inode.c
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:17k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*
  2.  *  inode.c
  3.  *
  4.  *  Copyright (C) 1995, 1996 by Paal-Kr. Engstad and Volker Lendecke
  5.  *  Copyright (C) 1997 by Volker Lendecke
  6.  *
  7.  *  Please add a note about your changes to smbfs in the ChangeLog file.
  8.  */
  9. #include <linux/config.h>
  10. #include <linux/module.h>
  11. #include <linux/sched.h>
  12. #include <linux/kernel.h>
  13. #include <linux/mm.h>
  14. #include <linux/string.h>
  15. #include <linux/stat.h>
  16. #include <linux/errno.h>
  17. #include <linux/locks.h>
  18. #include <linux/slab.h>
  19. #include <linux/init.h>
  20. #include <linux/file.h>
  21. #include <linux/dcache.h>
  22. #include <linux/smp_lock.h>
  23. #include <linux/nls.h>
  24. #include <linux/seq_file.h>
  25. #include <linux/smb_fs.h>
  26. #include <linux/smbno.h>
  27. #include <linux/smb_mount.h>
  28. #include <asm/system.h>
  29. #include <asm/uaccess.h>
  30. #include "smb_debug.h"
  31. #include "getopt.h"
  32. #include "proto.h"
  33. /* Always pick a default string */
  34. #ifdef CONFIG_SMB_NLS_REMOTE
  35. #define SMB_NLS_REMOTE CONFIG_SMB_NLS_REMOTE
  36. #else
  37. #define SMB_NLS_REMOTE ""
  38. #endif
  39. #define SMB_TTL_DEFAULT 1000
  40. #define SMB_TIMEO_DEFAULT 30
  41. static void smb_delete_inode(struct inode *);
  42. static void smb_put_super(struct super_block *);
  43. static int  smb_statfs(struct super_block *, struct statfs *);
  44. static int  smb_show_options(struct seq_file *, struct vfsmount *);
  45. static struct super_operations smb_sops =
  46. {
  47. put_inode: force_delete,
  48. delete_inode: smb_delete_inode,
  49. put_super: smb_put_super,
  50. statfs: smb_statfs,
  51. show_options: smb_show_options,
  52. };
  53. /* We are always generating a new inode here */
  54. struct inode *
  55. smb_iget(struct super_block *sb, struct smb_fattr *fattr)
  56. {
  57. struct inode *result;
  58. DEBUG1("smb_iget: %pn", fattr);
  59. result = new_inode(sb);
  60. if (!result)
  61. return result;
  62. result->i_ino = fattr->f_ino;
  63. memset(&(result->u.smbfs_i), 0, sizeof(result->u.smbfs_i));
  64. smb_set_inode_attr(result, fattr);
  65. if (S_ISREG(result->i_mode)) {
  66. result->i_op = &smb_file_inode_operations;
  67. result->i_fop = &smb_file_operations;
  68. result->i_data.a_ops = &smb_file_aops;
  69. } else if (S_ISDIR(result->i_mode)) {
  70. result->i_op = &smb_dir_inode_operations;
  71. result->i_fop = &smb_dir_operations;
  72. }
  73. insert_inode_hash(result);
  74. return result;
  75. }
  76. /*
  77.  * Copy the inode data to a smb_fattr structure.
  78.  */
  79. void
  80. smb_get_inode_attr(struct inode *inode, struct smb_fattr *fattr)
  81. {
  82. memset(fattr, 0, sizeof(struct smb_fattr));
  83. fattr->f_mode = inode->i_mode;
  84. fattr->f_nlink = inode->i_nlink;
  85. fattr->f_ino = inode->i_ino;
  86. fattr->f_uid = inode->i_uid;
  87. fattr->f_gid = inode->i_gid;
  88. fattr->f_rdev = inode->i_rdev;
  89. fattr->f_size = inode->i_size;
  90. fattr->f_mtime = inode->i_mtime;
  91. fattr->f_ctime = inode->i_ctime;
  92. fattr->f_atime = inode->i_atime;
  93. fattr->f_blksize= inode->i_blksize;
  94. fattr->f_blocks = inode->i_blocks;
  95. fattr->attr = inode->u.smbfs_i.attr;
  96. /*
  97.  * Keep the attributes in sync with the inode permissions.
  98.  */
  99. if (fattr->f_mode & S_IWUSR)
  100. fattr->attr &= ~aRONLY;
  101. else
  102. fattr->attr |= aRONLY;
  103. }
  104. /*
  105.  * Update the inode, possibly causing it to invalidate its pages if mtime/size
  106.  * is different from last time.
  107.  */
  108. void
  109. smb_set_inode_attr(struct inode *inode, struct smb_fattr *fattr)
  110. {
  111. /*
  112.  * A size change should have a different mtime, or same mtime
  113.  * but different size.
  114.  */
  115. time_t last_time = inode->i_mtime;
  116. loff_t last_sz = inode->i_size;
  117. inode->i_mode = fattr->f_mode;
  118. inode->i_nlink = fattr->f_nlink;
  119. inode->i_uid = fattr->f_uid;
  120. inode->i_gid = fattr->f_gid;
  121. inode->i_rdev = fattr->f_rdev;
  122. inode->i_ctime = fattr->f_ctime;
  123. inode->i_blksize= fattr->f_blksize;
  124. inode->i_blocks = fattr->f_blocks;
  125. inode->i_size = fattr->f_size;
  126. inode->i_mtime = fattr->f_mtime;
  127. inode->i_atime = fattr->f_atime;
  128. inode->u.smbfs_i.attr = fattr->attr;
  129. /*
  130.  * Update the "last time refreshed" field for revalidation.
  131.  */
  132. inode->u.smbfs_i.oldmtime = jiffies;
  133. if (inode->i_mtime != last_time || inode->i_size != last_sz) {
  134. VERBOSE("%ld changed, old=%ld, new=%ld, oz=%ld, nz=%ldn",
  135. inode->i_ino,
  136. (long) last_time, (long) inode->i_mtime,
  137. (long) last_sz, (long) inode->i_size);
  138. if (!S_ISDIR(inode->i_mode))
  139. invalidate_inode_pages(inode);
  140. }
  141. }
  142. /*
  143.  * This is called if the connection has gone bad ...
  144.  * try to kill off all the current inodes.
  145.  */
  146. void
  147. smb_invalidate_inodes(struct smb_sb_info *server)
  148. {
  149. VERBOSE("n");
  150. shrink_dcache_sb(SB_of(server));
  151. invalidate_inodes(SB_of(server));
  152. }
  153. /*
  154.  * This is called to update the inode attributes after
  155.  * we've made changes to a file or directory.
  156.  */
  157. static int
  158. smb_refresh_inode(struct dentry *dentry)
  159. {
  160. struct inode *inode = dentry->d_inode;
  161. int error;
  162. struct smb_fattr fattr;
  163. error = smb_proc_getattr(dentry, &fattr);
  164. if (!error) {
  165. smb_renew_times(dentry);
  166. /*
  167.  * Check whether the type part of the mode changed,
  168.  * and don't update the attributes if it did.
  169.  */
  170. if ((inode->i_mode & S_IFMT) == (fattr.f_mode & S_IFMT)) {
  171. smb_set_inode_attr(inode, &fattr);
  172. } else {
  173. /*
  174.  * Big trouble! The inode has become a new object,
  175.  * so any operations attempted on it are invalid.
  176.  *
  177.  * To limit damage, mark the inode as bad so that
  178.  * subsequent lookup validations will fail.
  179.  */
  180. PARANOIA("%s/%s changed mode, %07o to %07on",
  181.  DENTRY_PATH(dentry),
  182.  inode->i_mode, fattr.f_mode);
  183. fattr.f_mode = inode->i_mode; /* save mode */
  184. make_bad_inode(inode);
  185. inode->i_mode = fattr.f_mode; /* restore mode */
  186. /*
  187.  * No need to worry about unhashing the dentry: the
  188.  * lookup validation will see that the inode is bad.
  189.  * But we do want to invalidate the caches ...
  190.  */
  191. if (!S_ISDIR(inode->i_mode))
  192. invalidate_inode_pages(inode);
  193. else
  194. smb_invalid_dir_cache(inode);
  195. error = -EIO;
  196. }
  197. }
  198. return error;
  199. }
  200. /*
  201.  * This is called when we want to check whether the inode
  202.  * has changed on the server.  If it has changed, we must
  203.  * invalidate our local caches.
  204.  */
  205. int
  206. smb_revalidate_inode(struct dentry *dentry)
  207. {
  208. struct smb_sb_info *s = server_from_dentry(dentry);
  209. struct inode *inode = dentry->d_inode;
  210. int error = 0;
  211. DEBUG1("smb_revalidate_inoden");
  212. lock_kernel();
  213. /*
  214.  * Check whether we've recently refreshed the inode.
  215.  */
  216. if (time_before(jiffies, inode->u.smbfs_i.oldmtime + SMB_MAX_AGE(s))) {
  217. VERBOSE("up-to-date, ino=%ld, jiffies=%lu, oldtime=%lun",
  218. inode->i_ino, jiffies, inode->u.smbfs_i.oldmtime);
  219. goto out;
  220. }
  221. error = smb_refresh_inode(dentry);
  222. out:
  223. unlock_kernel();
  224. return error;
  225. }
  226. /*
  227.  * This routine is called when i_nlink == 0 and i_count goes to 0.
  228.  * All blocking cleanup operations need to go here to avoid races.
  229.  */
  230. static void
  231. smb_delete_inode(struct inode *ino)
  232. {
  233. DEBUG1("ino=%ldn", ino->i_ino);
  234. lock_kernel();
  235. if (smb_close(ino))
  236. PARANOIA("could not close inode %ldn", ino->i_ino);
  237. unlock_kernel();
  238. clear_inode(ino);
  239. }
  240. static struct option opts[] = {
  241. { "version", 0, 'v' },
  242. { "win95", SMB_MOUNT_WIN95, 1 },
  243. { "oldattr", SMB_MOUNT_OLDATTR, 1 },
  244. { "dirattr", SMB_MOUNT_DIRATTR, 1 },
  245. { "case", SMB_MOUNT_CASE, 1 },
  246. { "uid", 0, 'u' },
  247. { "gid", 0, 'g' },
  248. { "file_mode", 0, 'f' },
  249. { "dir_mode", 0, 'd' },
  250. { "iocharset", 0, 'i' },
  251. { "codepage", 0, 'c' },
  252. { "ttl", 0, 't' },
  253. { "timeo", 0, 'o' },
  254. { NULL, 0, 0}
  255. };
  256. static int
  257. parse_options(struct smb_mount_data_kernel *mnt, char *options)
  258. {
  259. int c;
  260. unsigned long flags;
  261. unsigned long value;
  262. char *optarg;
  263. char *optopt;
  264. flags = 0;
  265. while ( (c = smb_getopt("smbfs", &options, opts,
  266. &optopt, &optarg, &flags, &value)) > 0) {
  267. VERBOSE("'%s' -> '%s'n", optopt, optarg ? optarg : "<none>");
  268. switch (c) {
  269. case 1:
  270. /* got a "flag" option */
  271. break;
  272. case 'v':
  273. if (value != SMB_MOUNT_VERSION) {
  274. printk ("smbfs: Bad mount version %ld, expected %dn",
  275. value, SMB_MOUNT_VERSION);
  276. return 0;
  277. }
  278. mnt->version = value;
  279. break;
  280. case 'u':
  281. mnt->uid = value;
  282. break;
  283. case 'g':
  284. mnt->gid = value;
  285. break;
  286. case 'f':
  287. mnt->file_mode = (value & S_IRWXUGO) | S_IFREG;
  288. break;
  289. case 'd':
  290. mnt->dir_mode = (value & S_IRWXUGO) | S_IFDIR;
  291. break;
  292. case 'i':
  293. strncpy(mnt->codepage.local_name, optarg, 
  294. SMB_NLS_MAXNAMELEN);
  295. break;
  296. case 'c':
  297. strncpy(mnt->codepage.remote_name, optarg,
  298. SMB_NLS_MAXNAMELEN);
  299. break;
  300. case 't':
  301. mnt->ttl = value;
  302. break;
  303. case 'o':
  304. mnt->timeo = value;
  305. break;
  306. default:
  307. printk ("smbfs: Unrecognized mount option %sn",
  308. optopt);
  309. return -1;
  310. }
  311. }
  312. mnt->flags = flags;
  313. return c;
  314. }
  315. /*
  316.  * smb_show_options() is for displaying mount options in /proc/mounts.
  317.  * It tries to avoid showing settings that were not changed from their
  318.  * defaults.
  319.  */
  320. static int
  321. smb_show_options(struct seq_file *s, struct vfsmount *m)
  322. {
  323. struct smb_mount_data_kernel *mnt = m->mnt_sb->u.smbfs_sb.mnt;
  324. int i;
  325. for (i = 0; opts[i].name != NULL; i++)
  326. if (mnt->flags & opts[i].flag)
  327. seq_printf(s, ",%s", opts[i].name);
  328. if (mnt->uid != 0)
  329. seq_printf(s, ",uid=%d", mnt->uid);
  330. if (mnt->gid != 0)
  331. seq_printf(s, ",gid=%d", mnt->gid);
  332. if (mnt->mounted_uid != 0)
  333. seq_printf(s, ",mounted_uid=%d", mnt->mounted_uid);
  334. /* 
  335.  * Defaults for file_mode and dir_mode are unknown to us; they
  336.  * depend on the current umask of the user doing the mount.
  337.  */
  338. seq_printf(s, ",file_mode=%04o", mnt->file_mode & S_IRWXUGO);
  339. seq_printf(s, ",dir_mode=%04o", mnt->dir_mode & S_IRWXUGO);
  340. if (strcmp(mnt->codepage.local_name, CONFIG_NLS_DEFAULT))
  341. seq_printf(s, ",iocharset=%s", mnt->codepage.local_name);
  342. if (strcmp(mnt->codepage.remote_name, SMB_NLS_REMOTE))
  343. seq_printf(s, ",codepage=%s", mnt->codepage.remote_name);
  344. if (mnt->ttl != SMB_TTL_DEFAULT)
  345. seq_printf(s, ",ttl=%d", mnt->ttl);
  346. if (mnt->timeo != SMB_TIMEO_DEFAULT)
  347. seq_printf(s, ",timeo=%d", mnt->timeo);
  348. return 0;
  349. }
  350. static void
  351. smb_put_super(struct super_block *sb)
  352. {
  353. struct smb_sb_info *server = &(sb->u.smbfs_sb);
  354. if (server->sock_file) {
  355. smb_dont_catch_keepalive(server);
  356. fput(server->sock_file);
  357. }
  358. if (server->conn_pid)
  359.        kill_proc(server->conn_pid, SIGTERM, 1);
  360. smb_kfree(server->mnt);
  361. smb_kfree(server->temp_buf);
  362. if (server->packet)
  363. smb_vfree(server->packet);
  364. if (server->remote_nls) {
  365. unload_nls(server->remote_nls);
  366. server->remote_nls = NULL;
  367. }
  368. if (server->local_nls) {
  369. unload_nls(server->local_nls);
  370. server->local_nls = NULL;
  371. }
  372. }
  373. struct super_block *
  374. smb_read_super(struct super_block *sb, void *raw_data, int silent)
  375. {
  376. struct smb_sb_info *server = &sb->u.smbfs_sb;
  377. struct smb_mount_data_kernel *mnt;
  378. struct smb_mount_data *oldmnt;
  379. struct inode *root_inode;
  380. struct smb_fattr root;
  381. int ver;
  382. if (!raw_data)
  383. goto out_no_data;
  384. oldmnt = (struct smb_mount_data *) raw_data;
  385. ver = oldmnt->version;
  386. if (ver != SMB_MOUNT_OLDVERSION && cpu_to_be32(ver) != SMB_MOUNT_ASCII)
  387. goto out_wrong_data;
  388. sb->s_blocksize = 1024; /* Eh...  Is this correct? */
  389. sb->s_blocksize_bits = 10;
  390. sb->s_magic = SMB_SUPER_MAGIC;
  391. sb->s_op = &smb_sops;
  392. server->mnt = NULL;
  393. server->sock_file = NULL;
  394. init_MUTEX(&server->sem);
  395. init_waitqueue_head(&server->wait);
  396. server->conn_pid = 0;
  397. server->state = CONN_INVALID; /* no connection yet */
  398. server->generation = 0;
  399. server->packet_size = smb_round_length(SMB_INITIAL_PACKET_SIZE);
  400. server->packet = smb_vmalloc(server->packet_size);
  401. if (!server->packet)
  402. goto out_no_mem;
  403. /* Allocate the global temp buffer */
  404. server->temp_buf = smb_kmalloc(2*SMB_MAXPATHLEN+20, GFP_KERNEL);
  405. if (!server->temp_buf)
  406. goto out_no_temp;
  407. /* Setup NLS stuff */
  408. server->remote_nls = NULL;
  409. server->local_nls = NULL;
  410. server->name_buf = server->temp_buf + SMB_MAXPATHLEN + 20;
  411. /* Allocate the mount data structure */
  412. /* FIXME: merge this with the other malloc and get a whole page? */
  413. mnt = smb_kmalloc(sizeof(struct smb_mount_data_kernel), GFP_KERNEL);
  414. if (!mnt)
  415. goto out_no_mount;
  416. server->mnt = mnt;
  417. memset(mnt, 0, sizeof(struct smb_mount_data_kernel));
  418. strncpy(mnt->codepage.local_name, CONFIG_NLS_DEFAULT,
  419. SMB_NLS_MAXNAMELEN);
  420. strncpy(mnt->codepage.remote_name, SMB_NLS_REMOTE,
  421. SMB_NLS_MAXNAMELEN);
  422. mnt->ttl = SMB_TTL_DEFAULT;
  423. mnt->timeo = SMB_TIMEO_DEFAULT;
  424. if (ver == SMB_MOUNT_OLDVERSION) {
  425. mnt->version = oldmnt->version;
  426. /* FIXME: is this enough to convert uid/gid's ? */
  427. mnt->mounted_uid = oldmnt->mounted_uid;
  428. mnt->uid = oldmnt->uid;
  429. mnt->gid = oldmnt->gid;
  430. mnt->file_mode = (oldmnt->file_mode & S_IRWXUGO) | S_IFREG;
  431. mnt->dir_mode = (oldmnt->dir_mode & S_IRWXUGO) | S_IFDIR;
  432. mnt->flags = (oldmnt->file_mode >> 9);
  433. } else {
  434. if (parse_options(mnt, raw_data))
  435. goto out_bad_option;
  436. mnt->mounted_uid = current->uid;
  437. }
  438. smb_setcodepage(server, &mnt->codepage);
  439. /*
  440.  * Display the enabled options
  441.  * Note: smb_proc_getattr uses these in 2.4 (but was changed in 2.2)
  442.  */
  443. if (mnt->flags & SMB_MOUNT_OLDATTR)
  444. printk("SMBFS: Using core getattr (Win 95 speedup)n");
  445. else if (mnt->flags & SMB_MOUNT_DIRATTR)
  446. printk("SMBFS: Using dir ff getattrn");
  447. /*
  448.  * Keep the super block locked while we get the root inode.
  449.  */
  450. smb_init_root_dirent(server, &root);
  451. root_inode = smb_iget(sb, &root);
  452. if (!root_inode)
  453. goto out_no_root;
  454. sb->s_root = d_alloc_root(root_inode);
  455. if (!sb->s_root)
  456. goto out_no_root;
  457. smb_new_dentry(sb->s_root);
  458. return sb;
  459. out_no_root:
  460. iput(root_inode);
  461. out_bad_option:
  462. smb_kfree(server->mnt);
  463. out_no_mount:
  464. smb_kfree(server->temp_buf);
  465. out_no_temp:
  466. smb_vfree(server->packet);
  467. out_no_mem:
  468. if (!server->mnt)
  469. printk(KERN_ERR "smb_read_super: allocation failuren");
  470. goto out_fail;
  471. out_wrong_data:
  472. printk(KERN_ERR "smbfs: mount_data version %d is not supportedn", ver);
  473. goto out_fail;
  474. out_no_data:
  475. printk(KERN_ERR "smb_read_super: missing data argumentn");
  476. out_fail:
  477. return NULL;
  478. }
  479. static int
  480. smb_statfs(struct super_block *sb, struct statfs *buf)
  481. {
  482. int result = smb_proc_dskattr(sb, buf);
  483. buf->f_type = SMB_SUPER_MAGIC;
  484. buf->f_namelen = SMB_MAXPATHLEN;
  485. return result;
  486. }
  487. int
  488. smb_notify_change(struct dentry *dentry, struct iattr *attr)
  489. {
  490. struct inode *inode = dentry->d_inode;
  491. struct smb_sb_info *server = server_from_dentry(dentry);
  492. unsigned int mask = (S_IFREG | S_IFDIR | S_IRWXUGO);
  493. int error, changed, refresh = 0;
  494. struct smb_fattr fattr;
  495. error = smb_revalidate_inode(dentry);
  496. if (error)
  497. goto out;
  498. if ((error = inode_change_ok(inode, attr)) < 0)
  499. goto out;
  500. error = -EPERM;
  501. if ((attr->ia_valid & ATTR_UID) && (attr->ia_uid != server->mnt->uid))
  502. goto out;
  503. if ((attr->ia_valid & ATTR_GID) && (attr->ia_uid != server->mnt->gid))
  504. goto out;
  505. if ((attr->ia_valid & ATTR_MODE) && (attr->ia_mode & ~mask))
  506. goto out;
  507. if ((attr->ia_valid & ATTR_SIZE) != 0) {
  508. VERBOSE("changing %s/%s, old size=%ld, new size=%ldn",
  509. DENTRY_PATH(dentry),
  510. (long) inode->i_size, (long) attr->ia_size);
  511. filemap_fdatasync(inode->i_mapping);
  512. filemap_fdatawait(inode->i_mapping);
  513. error = smb_open(dentry, O_WRONLY);
  514. if (error)
  515. goto out;
  516. error = smb_proc_trunc(server, inode->u.smbfs_i.fileid,
  517.  attr->ia_size);
  518. if (error)
  519. goto out;
  520. error = vmtruncate(inode, attr->ia_size);
  521. if (error)
  522. goto out;
  523. refresh = 1;
  524. }
  525. /*
  526.  * Initialize the fattr and check for changed fields.
  527.  * Note: CTIME under SMB is creation time rather than
  528.  * change time, so we don't attempt to change it.
  529.  */
  530. smb_get_inode_attr(inode, &fattr);
  531. changed = 0;
  532. if ((attr->ia_valid & ATTR_MTIME) != 0) {
  533. fattr.f_mtime = attr->ia_mtime;
  534. changed = 1;
  535. }
  536. if ((attr->ia_valid & ATTR_ATIME) != 0) {
  537. fattr.f_atime = attr->ia_atime;
  538. /* Earlier protocols don't have an access time */
  539. if (server->opt.protocol >= SMB_PROTOCOL_LANMAN2)
  540. changed = 1;
  541. }
  542. if (changed) {
  543. error = smb_proc_settime(dentry, &fattr);
  544. if (error)
  545. goto out;
  546. refresh = 1;
  547. }
  548. /*
  549.  * Check for mode changes ... we're extremely limited in
  550.  * what can be set for SMB servers: just the read-only bit.
  551.  */
  552. if ((attr->ia_valid & ATTR_MODE) != 0) {
  553. VERBOSE("%s/%s mode change, old=%x, new=%xn",
  554. DENTRY_PATH(dentry), fattr.f_mode, attr->ia_mode);
  555. changed = 0;
  556. if (attr->ia_mode & S_IWUSR) {
  557. if (fattr.attr & aRONLY) {
  558. fattr.attr &= ~aRONLY;
  559. changed = 1;
  560. }
  561. } else {
  562. if (!(fattr.attr & aRONLY)) {
  563. fattr.attr |= aRONLY;
  564. changed = 1;
  565. }
  566. }
  567. if (changed) {
  568. error = smb_proc_setattr(dentry, &fattr);
  569. if (error)
  570. goto out;
  571. refresh = 1;
  572. }
  573. }
  574. error = 0;
  575. out:
  576. if (refresh)
  577. smb_refresh_inode(dentry);
  578. return error;
  579. }
  580. #ifdef DEBUG_SMB_MALLOC
  581. int smb_malloced;
  582. int smb_current_kmalloced;
  583. int smb_current_vmalloced;
  584. #endif
  585. static DECLARE_FSTYPE( smb_fs_type, "smbfs", smb_read_super, 0);
  586. static int __init init_smb_fs(void)
  587. {
  588. DEBUG1("registering ...n");
  589. #ifdef DEBUG_SMB_MALLOC
  590. smb_malloced = 0;
  591. smb_current_kmalloced = 0;
  592. smb_current_vmalloced = 0;
  593. #endif
  594. return register_filesystem(&smb_fs_type);
  595. }
  596. static void __exit exit_smb_fs(void)
  597. {
  598. DEBUG1("unregistering ...n");
  599. unregister_filesystem(&smb_fs_type);
  600. #ifdef DEBUG_SMB_MALLOC
  601. printk(KERN_DEBUG "smb_malloced: %dn", smb_malloced);
  602. printk(KERN_DEBUG "smb_current_kmalloced: %dn",smb_current_kmalloced);
  603. printk(KERN_DEBUG "smb_current_vmalloced: %dn",smb_current_vmalloced);
  604. #endif
  605. }
  606. EXPORT_NO_SYMBOLS;
  607. module_init(init_smb_fs)
  608. module_exit(exit_smb_fs)
  609. MODULE_LICENSE("GPL");