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

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*
  2.  * fs/bfs/inode.c
  3.  * BFS superblock and inode operations.
  4.  * Copyright (C) 1999,2000 Tigran Aivazian <tigran@veritas.com>
  5.  * From fs/minix, Copyright (C) 1991, 1992 Linus Torvalds.
  6.  */
  7. #include <linux/module.h>
  8. #include <linux/mm.h>
  9. #include <linux/slab.h>
  10. #include <linux/init.h>
  11. #include <linux/locks.h>
  12. #include <linux/bfs_fs.h>
  13. #include <linux/smp_lock.h>
  14. #include <asm/uaccess.h>
  15. #include "bfs_defs.h"
  16. MODULE_AUTHOR("Tigran A. Aivazian <tigran@veritas.com>");
  17. MODULE_DESCRIPTION("SCO UnixWare BFS filesystem for Linux");
  18. MODULE_LICENSE("GPL");
  19. EXPORT_NO_SYMBOLS;
  20. #undef DEBUG
  21. #ifdef DEBUG
  22. #define dprintf(x...) printf(x)
  23. #else
  24. #define dprintf(x...)
  25. #endif
  26. void dump_imap(const char *prefix, struct super_block * s);
  27. static void bfs_read_inode(struct inode * inode)
  28. {
  29. unsigned long ino = inode->i_ino;
  30. kdev_t dev = inode->i_dev;
  31. struct bfs_inode * di;
  32. struct buffer_head * bh;
  33. int block, off;
  34. if (ino < BFS_ROOT_INO || ino > inode->i_sb->su_lasti) {
  35. printf("Bad inode number %s:%08lxn", bdevname(dev), ino);
  36. make_bad_inode(inode);
  37. return;
  38. }
  39. block = (ino - BFS_ROOT_INO)/BFS_INODES_PER_BLOCK + 1;
  40. bh = sb_bread(inode->i_sb, block);
  41. if (!bh) {
  42. printf("Unable to read inode %s:%08lxn", bdevname(dev), ino);
  43. make_bad_inode(inode);
  44. return;
  45. }
  46. off = (ino - BFS_ROOT_INO) % BFS_INODES_PER_BLOCK;
  47. di = (struct bfs_inode *)bh->b_data + off;
  48. inode->i_mode = 0x0000FFFF & di->i_mode;
  49. if (di->i_vtype == BFS_VDIR) {
  50. inode->i_mode |= S_IFDIR;
  51. inode->i_op = &bfs_dir_inops;
  52. inode->i_fop = &bfs_dir_operations;
  53. } else if (di->i_vtype == BFS_VREG) {
  54. inode->i_mode |= S_IFREG;
  55. inode->i_op = &bfs_file_inops;
  56. inode->i_fop = &bfs_file_operations;
  57. inode->i_mapping->a_ops = &bfs_aops;
  58. }
  59. inode->i_uid = di->i_uid;
  60. inode->i_gid = di->i_gid;
  61. inode->i_nlink = di->i_nlink;
  62. inode->i_size = BFS_FILESIZE(di);
  63. inode->i_blocks = BFS_FILEBLOCKS(di);
  64. inode->i_blksize = PAGE_SIZE;
  65. inode->i_atime = di->i_atime;
  66. inode->i_mtime = di->i_mtime;
  67. inode->i_ctime = di->i_ctime;
  68. inode->iu_dsk_ino = di->i_ino; /* can be 0 so we store a copy */
  69. inode->iu_sblock = di->i_sblock;
  70. inode->iu_eblock = di->i_eblock;
  71. brelse(bh);
  72. }
  73. static void bfs_write_inode(struct inode * inode, int unused)
  74. {
  75. unsigned long ino = inode->i_ino;
  76. kdev_t dev = inode->i_dev;
  77. struct bfs_inode * di;
  78. struct buffer_head * bh;
  79. int block, off;
  80. if (ino < BFS_ROOT_INO || ino > inode->i_sb->su_lasti) {
  81. printf("Bad inode number %s:%08lxn", bdevname(dev), ino);
  82. return;
  83. }
  84. lock_kernel();
  85. block = (ino - BFS_ROOT_INO)/BFS_INODES_PER_BLOCK + 1;
  86. bh = sb_bread(inode->i_sb, block);
  87. if (!bh) {
  88. printf("Unable to read inode %s:%08lxn", bdevname(dev), ino);
  89. unlock_kernel();
  90. return;
  91. }
  92. off = (ino - BFS_ROOT_INO)%BFS_INODES_PER_BLOCK;
  93. di = (struct bfs_inode *)bh->b_data + off;
  94. if (inode->i_ino == BFS_ROOT_INO)
  95. di->i_vtype = BFS_VDIR;
  96. else
  97. di->i_vtype = BFS_VREG;
  98. di->i_ino = inode->i_ino;
  99. di->i_mode = inode->i_mode;
  100. di->i_uid = inode->i_uid;
  101. di->i_gid = inode->i_gid;
  102. di->i_nlink = inode->i_nlink;
  103. di->i_atime = inode->i_atime;
  104. di->i_mtime = inode->i_mtime;
  105. di->i_ctime = inode->i_ctime;
  106. di->i_sblock = inode->iu_sblock;
  107. di->i_eblock = inode->iu_eblock;
  108. di->i_eoffset = di->i_sblock * BFS_BSIZE + inode->i_size - 1;
  109. mark_buffer_dirty(bh);
  110. brelse(bh);
  111. unlock_kernel();
  112. }
  113. static void bfs_delete_inode(struct inode * inode)
  114. {
  115. unsigned long ino = inode->i_ino;
  116. kdev_t dev = inode->i_dev;
  117. struct bfs_inode * di;
  118. struct buffer_head * bh;
  119. int block, off;
  120. struct super_block * s = inode->i_sb;
  121. dprintf("ino=%08lxn", inode->i_ino);
  122. if (inode->i_ino < BFS_ROOT_INO || inode->i_ino > inode->i_sb->su_lasti) {
  123. printf("invalid ino=%08lxn", inode->i_ino);
  124. return;
  125. }
  126. inode->i_size = 0;
  127. inode->i_atime = inode->i_mtime = inode->i_ctime = CURRENT_TIME;
  128. lock_kernel();
  129. mark_inode_dirty(inode);
  130. block = (ino - BFS_ROOT_INO)/BFS_INODES_PER_BLOCK + 1;
  131. bh = sb_bread(s, block);
  132. if (!bh) {
  133. printf("Unable to read inode %s:%08lxn", bdevname(dev), ino);
  134. unlock_kernel();
  135. return;
  136. }
  137. off = (ino - BFS_ROOT_INO)%BFS_INODES_PER_BLOCK;
  138. di = (struct bfs_inode *)bh->b_data + off;
  139. if (di->i_ino) {
  140. s->su_freeb += BFS_FILEBLOCKS(di);
  141. s->su_freei++;
  142. clear_bit(di->i_ino, s->su_imap);
  143. dump_imap("delete_inode", s);
  144. }
  145. di->i_ino = 0;
  146. di->i_sblock = 0;
  147. mark_buffer_dirty(bh);
  148. brelse(bh);
  149. /* if this was the last file, make the previous 
  150.    block "last files last block" even if there is no real file there,
  151.    saves us 1 gap */
  152. if (s->su_lf_eblk == inode->iu_eblock) {
  153. s->su_lf_eblk = inode->iu_sblock - 1;
  154. mark_buffer_dirty(s->su_sbh);
  155. }
  156. unlock_kernel();
  157. clear_inode(inode);
  158. }
  159. static void bfs_put_super(struct super_block *s)
  160. {
  161. brelse(s->su_sbh);
  162. kfree(s->su_imap);
  163. }
  164. static int bfs_statfs(struct super_block *s, struct statfs *buf)
  165. {
  166. buf->f_type = BFS_MAGIC;
  167. buf->f_bsize = s->s_blocksize;
  168. buf->f_blocks = s->su_blocks;
  169. buf->f_bfree = buf->f_bavail = s->su_freeb;
  170. buf->f_files = s->su_lasti + 1 - BFS_ROOT_INO;
  171. buf->f_ffree = s->su_freei;
  172. buf->f_fsid.val[0] = kdev_t_to_nr(s->s_dev);
  173. buf->f_namelen = BFS_NAMELEN;
  174. return 0;
  175. }
  176. static void bfs_write_super(struct super_block *s)
  177. {
  178. if (!(s->s_flags & MS_RDONLY))
  179. mark_buffer_dirty(s->su_sbh);
  180. s->s_dirt = 0;
  181. }
  182. static struct super_operations bfs_sops = {
  183. read_inode: bfs_read_inode,
  184. write_inode: bfs_write_inode,
  185. delete_inode: bfs_delete_inode,
  186. put_super: bfs_put_super,
  187. write_super: bfs_write_super,
  188. statfs: bfs_statfs,
  189. };
  190. void dump_imap(const char *prefix, struct super_block * s)
  191. {
  192. #if 0
  193. int i;
  194. char *tmpbuf = (char *)get_free_page(GFP_KERNEL);
  195. if (!tmpbuf)
  196. return;
  197. for (i=s->su_lasti; i>=0; i--) {
  198. if (i>PAGE_SIZE-100) break;
  199. if (test_bit(i, s->su_imap))
  200. strcat(tmpbuf, "1");
  201. else
  202. strcat(tmpbuf, "0");
  203. }
  204. printk(KERN_ERR "BFS-fs: %s: lasti=%08lx <%s>n", prefix, s->su_lasti, tmpbuf);
  205. free_page((unsigned long)tmpbuf);
  206. #endif
  207. }
  208. static struct super_block * bfs_read_super(struct super_block * s, 
  209. void * data, int silent)
  210. {
  211. kdev_t dev;
  212. struct buffer_head * bh;
  213. struct bfs_super_block * bfs_sb;
  214. struct inode * inode;
  215. int i, imap_len;
  216. dev = s->s_dev;
  217. set_blocksize(dev, BFS_BSIZE);
  218. s->s_blocksize = BFS_BSIZE;
  219. s->s_blocksize_bits = BFS_BSIZE_BITS;
  220. bh = sb_bread(s, 0);
  221. if(!bh)
  222. goto out;
  223. bfs_sb = (struct bfs_super_block *)bh->b_data;
  224. if (bfs_sb->s_magic != BFS_MAGIC) {
  225. if (!silent)
  226. printf("No BFS filesystem on %s (magic=%08x)n", 
  227. bdevname(dev), bfs_sb->s_magic);
  228. goto out;
  229. }
  230. if (BFS_UNCLEAN(bfs_sb, s) && !silent)
  231. printf("%s is unclean, continuingn", bdevname(dev));
  232. s->s_magic = BFS_MAGIC;
  233. s->su_bfs_sb = bfs_sb;
  234. s->su_sbh = bh;
  235. s->su_lasti = (bfs_sb->s_start - BFS_BSIZE)/sizeof(struct bfs_inode) 
  236. + BFS_ROOT_INO - 1;
  237. imap_len = s->su_lasti/8 + 1;
  238. s->su_imap = kmalloc(imap_len, GFP_KERNEL);
  239. if (!s->su_imap)
  240. goto out;
  241. memset(s->su_imap, 0, imap_len);
  242. for (i=0; i<BFS_ROOT_INO; i++) 
  243. set_bit(i, s->su_imap);
  244. s->s_op = &bfs_sops;
  245. inode = iget(s, BFS_ROOT_INO);
  246. if (!inode) {
  247. kfree(s->su_imap);
  248. goto out;
  249. }
  250. s->s_root = d_alloc_root(inode);
  251. if (!s->s_root) {
  252. iput(inode);
  253. kfree(s->su_imap);
  254. goto out;
  255. }
  256. s->su_blocks = (bfs_sb->s_end + 1)>>BFS_BSIZE_BITS; /* for statfs(2) */
  257. s->su_freeb = (bfs_sb->s_end + 1 - bfs_sb->s_start)>>BFS_BSIZE_BITS;
  258. s->su_freei = 0;
  259. s->su_lf_eblk = 0;
  260. s->su_lf_sblk = 0;
  261. s->su_lf_ioff = 0;
  262. for (i=BFS_ROOT_INO; i<=s->su_lasti; i++) {
  263. inode = iget(s,i);
  264. if (inode->iu_dsk_ino == 0)
  265. s->su_freei++;
  266. else {
  267. set_bit(i, s->su_imap);
  268. s->su_freeb -= inode->i_blocks;
  269. if (inode->iu_eblock > s->su_lf_eblk) {
  270. s->su_lf_eblk = inode->iu_eblock;
  271. s->su_lf_sblk = inode->iu_sblock;
  272. s->su_lf_ioff = BFS_INO2OFF(i);
  273. }
  274. }
  275. iput(inode);
  276. }
  277. if (!(s->s_flags & MS_RDONLY)) {
  278. mark_buffer_dirty(bh);
  279. s->s_dirt = 1;
  280. dump_imap("read_super", s);
  281. return s;
  282. out:
  283. brelse(bh);
  284. return NULL;
  285. }
  286. static DECLARE_FSTYPE_DEV(bfs_fs_type, "bfs", bfs_read_super);
  287. static int __init init_bfs_fs(void)
  288. {
  289. return register_filesystem(&bfs_fs_type);
  290. }
  291. static void __exit exit_bfs_fs(void)
  292. {
  293. unregister_filesystem(&bfs_fs_type);
  294. }
  295. module_init(init_bfs_fs)
  296. module_exit(exit_bfs_fs)