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

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*
  2.  *  linux/fs/affs/inode.c
  3.  *
  4.  *  (c) 1996  Hans-Joachim Widmaier - Rewritten
  5.  *
  6.  *  (C) 1993  Ray Burr - Modified for Amiga FFS filesystem.
  7.  *
  8.  *  (C) 1992  Eric Youngdale Modified for ISO9660 filesystem.
  9.  *
  10.  *  (C) 1991  Linus Torvalds - minix filesystem
  11.  */
  12. #include <asm/div64.h>
  13. #include <linux/errno.h>
  14. #include <linux/fs.h>
  15. #include <linux/slab.h>
  16. #include <linux/stat.h>
  17. #include <linux/sched.h>
  18. #include <linux/affs_fs.h>
  19. #include <linux/kernel.h>
  20. #include <linux/mm.h>
  21. #include <linux/string.h>
  22. #include <linux/locks.h>
  23. #include <linux/genhd.h>
  24. #include <linux/amigaffs.h>
  25. #include <linux/major.h>
  26. #include <linux/blkdev.h>
  27. #include <linux/init.h>
  28. #include <linux/smp_lock.h>
  29. #include <asm/system.h>
  30. #include <asm/uaccess.h>
  31. #include <linux/module.h>
  32. extern struct inode_operations affs_symlink_inode_operations;
  33. extern struct timezone sys_tz;
  34. void
  35. affs_read_inode(struct inode *inode)
  36. {
  37. struct super_block *sb = inode->i_sb;
  38. struct buffer_head *bh;
  39. struct affs_head *head;
  40. struct affs_tail *tail;
  41. u32  block;
  42. u32  size;
  43. u32  prot;
  44. u16  id;
  45. pr_debug("AFFS: read_inode(%lu)n",inode->i_ino);
  46. block = inode->i_ino;
  47. bh = affs_bread(sb, block);
  48. if (!bh) {
  49. affs_warning(sb, "read_inode", "Cannot read block %d", block);
  50. goto bad_inode;
  51. }
  52. if (affs_checksum_block(sb, bh) || be32_to_cpu(AFFS_HEAD(bh)->ptype) != T_SHORT) {
  53. affs_warning(sb,"read_inode",
  54.    "Checksum or type (ptype=%d) error on inode %d",
  55.    AFFS_HEAD(bh)->ptype, block);
  56. goto bad_inode;
  57. }
  58. head = AFFS_HEAD(bh);
  59. tail = AFFS_TAIL(sb, bh);
  60. prot = be32_to_cpu(tail->protect);
  61. inode->i_size = 0;
  62. inode->i_nlink = 1;
  63. inode->i_mode = 0;
  64. memset(AFFS_INODE, 0, sizeof(*AFFS_INODE));
  65. init_MUTEX(&AFFS_INODE->i_link_lock);
  66. init_MUTEX(&AFFS_INODE->i_ext_lock);
  67. AFFS_INODE->i_extcnt = 1;
  68. AFFS_INODE->i_ext_last = ~1;
  69. AFFS_INODE->i_protect = prot;
  70. if (AFFS_SB->s_flags & SF_SETMODE)
  71. inode->i_mode = AFFS_SB->s_mode;
  72. else
  73. inode->i_mode = prot_to_mode(prot);
  74. id = be16_to_cpu(tail->uid);
  75. if (id == 0 || AFFS_SB->s_flags & SF_SETUID)
  76. inode->i_uid = AFFS_SB->s_uid;
  77. else if (id == 0xFFFF && AFFS_SB->s_flags & SF_MUFS)
  78. inode->i_uid = 0;
  79. else
  80. inode->i_uid = id;
  81. id = be16_to_cpu(tail->gid);
  82. if (id == 0 || AFFS_SB->s_flags & SF_SETGID)
  83. inode->i_gid = AFFS_SB->s_gid;
  84. else if (id == 0xFFFF && AFFS_SB->s_flags & SF_MUFS)
  85. inode->i_gid = 0;
  86. else
  87. inode->i_gid = id;
  88. switch (be32_to_cpu(tail->stype)) {
  89. case ST_ROOT:
  90. inode->i_uid = AFFS_SB->s_uid;
  91. inode->i_gid = AFFS_SB->s_gid;
  92. /* fall through */
  93. case ST_USERDIR:
  94. if (be32_to_cpu(tail->stype) == ST_USERDIR ||
  95.     AFFS_SB->s_flags & SF_SETMODE) {
  96. if (inode->i_mode & S_IRUSR)
  97. inode->i_mode |= S_IXUSR;
  98. if (inode->i_mode & S_IRGRP)
  99. inode->i_mode |= S_IXGRP;
  100. if (inode->i_mode & S_IROTH)
  101. inode->i_mode |= S_IXOTH;
  102. inode->i_mode |= S_IFDIR;
  103. } else
  104. inode->i_mode = S_IRUGO | S_IXUGO | S_IWUSR | S_IFDIR;
  105. if (tail->link_chain)
  106. inode->i_nlink = 2;
  107. /* Maybe it should be controlled by mount parameter? */
  108. //inode->i_mode |= S_ISVTX;
  109. inode->i_op = &affs_dir_inode_operations;
  110. inode->i_fop = &affs_dir_operations;
  111. break;
  112. case ST_LINKDIR:
  113. #if 0
  114. affs_warning(sb, "read_inode", "inode is LINKDIR");
  115. goto bad_inode;
  116. #else
  117. inode->i_mode |= S_IFDIR;
  118. inode->i_op = NULL;
  119. inode->i_fop = NULL;
  120. break;
  121. #endif
  122. case ST_LINKFILE:
  123. affs_warning(sb, "read_inode", "inode is LINKFILE");
  124. goto bad_inode;
  125. case ST_FILE:
  126. size = be32_to_cpu(tail->size);
  127. inode->i_mode |= S_IFREG;
  128. AFFS_INODE->mmu_private = inode->i_size = size;
  129. if (inode->i_size) {
  130. AFFS_INODE->i_blkcnt = (size - 1) /
  131.        AFFS_SB->s_data_blksize + 1;
  132. AFFS_INODE->i_extcnt = (AFFS_INODE->i_blkcnt - 1) /
  133.        AFFS_SB->s_hashsize + 1;
  134. }
  135. if (tail->link_chain)
  136. inode->i_nlink = 2;
  137. inode->i_mapping->a_ops = (AFFS_SB->s_flags & SF_OFS) ? &affs_aops_ofs : &affs_aops;
  138. inode->i_op = &affs_file_inode_operations;
  139. inode->i_fop = &affs_file_operations;
  140. break;
  141. case ST_SOFTLINK:
  142. inode->i_mode |= S_IFLNK;
  143. inode->i_op = &affs_symlink_inode_operations;
  144. inode->i_data.a_ops = &affs_symlink_aops;
  145. break;
  146. }
  147. inode->i_mtime = inode->i_atime = inode->i_ctime
  148.        = (be32_to_cpu(tail->change.days) * (24 * 60 * 60) +
  149.          be32_to_cpu(tail->change.mins) * 60 +
  150.  be32_to_cpu(tail->change.ticks) / 50 +
  151.  ((8 * 365 + 2) * 24 * 60 * 60)) +
  152.  sys_tz.tz_minuteswest * 60;
  153. affs_brelse(bh);
  154. return;
  155. bad_inode:
  156. make_bad_inode(inode);
  157. affs_brelse(bh);
  158. return;
  159. }
  160. void
  161. affs_write_inode(struct inode *inode, int unused)
  162. {
  163. struct super_block *sb = inode->i_sb;
  164. struct buffer_head *bh;
  165. struct affs_tail *tail;
  166. uid_t  uid;
  167. gid_t  gid;
  168. pr_debug("AFFS: write_inode(%lu)n",inode->i_ino);
  169. if (!inode->i_nlink)
  170. // possibly free block
  171. return;
  172. lock_kernel();
  173. bh = affs_bread(sb, inode->i_ino);
  174. if (!bh) {
  175. affs_error(sb,"write_inode","Cannot read block %lu",inode->i_ino);
  176. unlock_kernel();
  177. return;
  178. }
  179. tail = AFFS_TAIL(sb, bh);
  180. if (tail->stype == be32_to_cpu(ST_ROOT)) {
  181. secs_to_datestamp(inode->i_mtime,&AFFS_ROOT_TAIL(sb, bh)->root_change);
  182. } else {
  183. tail->protect = cpu_to_be32(AFFS_INODE->i_protect);
  184. tail->size = cpu_to_be32(inode->i_size);
  185. secs_to_datestamp(inode->i_mtime,&tail->change);
  186. if (!(inode->i_ino == AFFS_SB->s_root_block)) {
  187. uid = inode->i_uid;
  188. gid = inode->i_gid;
  189. if (sb->u.affs_sb.s_flags & SF_MUFS) {
  190. if (inode->i_uid == 0 || inode->i_uid == 0xFFFF)
  191. uid = inode->i_uid ^ ~0;
  192. if (inode->i_gid == 0 || inode->i_gid == 0xFFFF)
  193. gid = inode->i_gid ^ ~0;
  194. }
  195. if (!(sb->u.affs_sb.s_flags & SF_SETUID))
  196. tail->uid = cpu_to_be16(uid);
  197. if (!(sb->u.affs_sb.s_flags & SF_SETGID))
  198. tail->gid = cpu_to_be16(gid);
  199. }
  200. }
  201. affs_fix_checksum(sb, bh);
  202. mark_buffer_dirty_inode(bh, inode);
  203. affs_brelse(bh);
  204. unlock_kernel();
  205. }
  206. int
  207. affs_notify_change(struct dentry *dentry, struct iattr *attr)
  208. {
  209. struct inode *inode = dentry->d_inode;
  210. int error;
  211. pr_debug("AFFS: notify_change(%lu,0x%x)n",inode->i_ino,attr->ia_valid);
  212. error = inode_change_ok(inode,attr);
  213. if (error)
  214. goto out;
  215. if (((attr->ia_valid & ATTR_UID) && (inode->i_sb->u.affs_sb.s_flags & SF_SETUID)) ||
  216.     ((attr->ia_valid & ATTR_GID) && (inode->i_sb->u.affs_sb.s_flags & SF_SETGID)) ||
  217.     ((attr->ia_valid & ATTR_MODE) &&
  218.      (inode->i_sb->u.affs_sb.s_flags & (SF_SETMODE | SF_IMMUTABLE)))) {
  219. if (!(inode->i_sb->u.affs_sb.s_flags & SF_QUIET))
  220. error = -EPERM;
  221. goto out;
  222. }
  223. inode_setattr(inode, attr);
  224. if (!error && (attr->ia_valid & ATTR_MODE))
  225. mode_to_prot(inode);
  226. out:
  227. return error;
  228. }
  229. void
  230. affs_put_inode(struct inode *inode)
  231. {
  232. pr_debug("AFFS: put_inode(ino=%lu, nlink=%u)n", inode->i_ino, inode->i_nlink);
  233. lock_kernel();
  234. affs_free_prealloc(inode);
  235. if (atomic_read(&inode->i_count) == 1) {
  236. if (inode->i_size != AFFS_INODE->mmu_private)
  237. affs_truncate(inode);
  238. //if (inode->i_nlink)
  239. // affs_clear_inode(inode);
  240. }
  241. unlock_kernel();
  242. }
  243. void
  244. affs_delete_inode(struct inode *inode)
  245. {
  246. pr_debug("AFFS: delete_inode(ino=%lu, nlink=%u)n", inode->i_ino, inode->i_nlink);
  247. lock_kernel();
  248. inode->i_size = 0;
  249. if (S_ISREG(inode->i_mode))
  250. affs_truncate(inode);
  251. clear_inode(inode);
  252. affs_free_block(inode->i_sb, inode->i_ino);
  253. unlock_kernel();
  254. }
  255. void
  256. affs_clear_inode(struct inode *inode)
  257. {
  258. unsigned long cache_page = (unsigned long) inode->u.affs_i.i_lc;
  259. pr_debug("AFFS: clear_inode(ino=%lu, nlink=%u)n", inode->i_ino, inode->i_nlink);
  260. if (cache_page) {
  261. pr_debug("AFFS: freeing ext cachen");
  262. inode->u.affs_i.i_lc = NULL;
  263. inode->u.affs_i.i_ac = NULL;
  264. free_page(cache_page);
  265. }
  266. affs_brelse(AFFS_INODE->i_ext_bh);
  267. AFFS_INODE->i_ext_last = ~1;
  268. AFFS_INODE->i_ext_bh = NULL;
  269. }
  270. struct inode *
  271. affs_new_inode(struct inode *dir)
  272. {
  273. struct super_block *sb = dir->i_sb;
  274. struct inode *inode;
  275. u32  block;
  276. struct buffer_head *bh;
  277. if (!(inode = new_inode(sb)))
  278. goto err_inode;
  279. if (!(block = affs_alloc_block(dir, dir->i_ino)))
  280. goto err_block;
  281. bh = affs_getzeroblk(sb, block);
  282. if (!bh)
  283. goto err_bh;
  284. mark_buffer_dirty_inode(bh, inode);
  285. affs_brelse(bh);
  286. inode->i_uid     = current->fsuid;
  287. inode->i_gid     = current->fsgid;
  288. inode->i_ino     = block;
  289. inode->i_nlink   = 1;
  290. inode->i_mtime   = inode->i_atime = inode->i_ctime = CURRENT_TIME;
  291. memset(AFFS_INODE, 0, sizeof(*AFFS_INODE));
  292. AFFS_INODE->i_extcnt = 1;
  293. AFFS_INODE->i_ext_last = ~1;
  294. init_MUTEX(&AFFS_INODE->i_link_lock);
  295. init_MUTEX(&AFFS_INODE->i_ext_lock);
  296. insert_inode_hash(inode);
  297. return inode;
  298. err_bh:
  299. affs_free_block(sb, block);
  300. err_block:
  301. iput(inode);
  302. err_inode:
  303. return NULL;
  304. }
  305. /*
  306.  * Add an entry to a directory. Create the header block
  307.  * and insert it into the hash table.
  308.  */
  309. int
  310. affs_add_entry(struct inode *dir, struct inode *inode, struct dentry *dentry, s32 type)
  311. {
  312. struct super_block *sb = dir->i_sb;
  313. struct buffer_head *inode_bh = NULL;
  314. struct buffer_head *bh = NULL;
  315. u32 block = 0;
  316. int retval;
  317. pr_debug("AFFS: add_entry(dir=%u, inode=%u, "%*s", type=%d)n", (u32)dir->i_ino,
  318.          (u32)inode->i_ino, (int)dentry->d_name.len, dentry->d_name.name, type);
  319. retval = -EIO;
  320. bh = affs_bread(sb, inode->i_ino);
  321. if (!bh)
  322. goto done;
  323. affs_lock_link(inode);
  324. switch (type) {
  325. case ST_LINKFILE:
  326. case ST_LINKDIR:
  327. inode_bh = bh;
  328. retval = -ENOSPC;
  329. block = affs_alloc_block(dir, dir->i_ino);
  330. if (!block)
  331. goto err;
  332. retval = -EIO;
  333. bh = affs_getzeroblk(sb, block);
  334. if (!bh)
  335. goto err;
  336. break;
  337. default:
  338. break;
  339. }
  340. AFFS_HEAD(bh)->ptype = cpu_to_be32(T_SHORT);
  341. AFFS_HEAD(bh)->key = cpu_to_be32(bh->b_blocknr);
  342. affs_copy_name(AFFS_TAIL(sb, bh)->name, dentry);
  343. AFFS_TAIL(sb, bh)->stype = cpu_to_be32(type);
  344. AFFS_TAIL(sb, bh)->parent = cpu_to_be32(dir->i_ino);
  345. if (inode_bh) {
  346. u32 chain;
  347.         chain = AFFS_TAIL(sb, inode_bh)->link_chain;
  348. AFFS_TAIL(sb, bh)->original = cpu_to_be32(inode->i_ino);
  349. AFFS_TAIL(sb, bh)->link_chain = chain;
  350. AFFS_TAIL(sb, inode_bh)->link_chain = cpu_to_be32(block);
  351. affs_adjust_checksum(inode_bh, block - be32_to_cpu(chain));
  352. mark_buffer_dirty_inode(inode_bh, inode);
  353. inode->i_nlink = 2;
  354. atomic_inc(&inode->i_count);
  355. }
  356. affs_fix_checksum(sb, bh);
  357. mark_buffer_dirty_inode(bh, inode);
  358. dentry->d_fsdata = (void *)bh->b_blocknr;
  359. affs_lock_dir(dir);
  360. retval = affs_insert_hash(dir, bh);
  361. mark_buffer_dirty_inode(bh, inode);
  362. affs_unlock_dir(dir);
  363. affs_unlock_link(inode);
  364. d_instantiate(dentry, inode);
  365. done:
  366. affs_brelse(inode_bh);
  367. affs_brelse(bh);
  368. return retval;
  369. err:
  370. if (block)
  371. affs_free_block(sb, block);
  372. affs_unlock_link(inode);
  373. goto done;
  374. }
  375. MODULE_LICENSE("GPL");