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

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*
  2.  *  linux/fs/affs/namei.c
  3.  *
  4.  *  (c) 1996  Hans-Joachim Widmaier - Rewritten
  5.  *
  6.  *  (C) 1993  Ray Burr - Modified for Amiga FFS filesystem.
  7.  *
  8.  *  (C) 1991  Linus Torvalds - minix filesystem
  9.  */
  10. #include <linux/sched.h>
  11. #include <linux/affs_fs.h>
  12. #include <linux/kernel.h>
  13. #include <linux/string.h>
  14. #include <linux/stat.h>
  15. #include <linux/fcntl.h>
  16. #include <linux/locks.h>
  17. #include <linux/amigaffs.h>
  18. #include <asm/uaccess.h>
  19. #include <linux/errno.h>
  20. typedef int (*toupper_t)(int);
  21. extern struct inode_operations affs_symlink_inode_operations;
  22. static int  affs_toupper(int ch);
  23. static int  affs_hash_dentry(struct dentry *, struct qstr *);
  24. static int       affs_compare_dentry(struct dentry *, struct qstr *, struct qstr *);
  25. static int  affs_intl_toupper(int ch);
  26. static int  affs_intl_hash_dentry(struct dentry *, struct qstr *);
  27. static int       affs_intl_compare_dentry(struct dentry *, struct qstr *, struct qstr *);
  28. struct dentry_operations affs_dentry_operations = {
  29. d_hash: affs_hash_dentry,
  30. d_compare: affs_compare_dentry,
  31. };
  32. struct dentry_operations affs_intl_dentry_operations = {
  33. d_hash: affs_intl_hash_dentry,
  34. d_compare: affs_intl_compare_dentry,
  35. };
  36. /* Simple toupper() for DOS1 */
  37. static int
  38. affs_toupper(int ch)
  39. {
  40. return ch >= 'a' && ch <= 'z' ? ch -= ('a' - 'A') : ch;
  41. }
  42. /* International toupper() for DOS3 ("international") */
  43. static int
  44. affs_intl_toupper(int ch)
  45. {
  46. return (ch >= 'a' && ch <= 'z') || (ch >= 0xE0
  47. && ch <= 0xFE && ch != 0xF7) ?
  48. ch - ('a' - 'A') : ch;
  49. }
  50. static inline toupper_t
  51. affs_get_toupper(struct super_block *sb)
  52. {
  53. return AFFS_SB->s_flags & SF_INTL ? affs_intl_toupper : affs_toupper;
  54. }
  55. /*
  56.  * Note: the dentry argument is the parent dentry.
  57.  */
  58. static inline int
  59. __affs_hash_dentry(struct dentry *dentry, struct qstr *qstr, toupper_t toupper)
  60. {
  61. const u8 *name = qstr->name;
  62. unsigned long hash;
  63. int i;
  64. i = affs_check_name(qstr->name,qstr->len);
  65. if (i)
  66. return i;
  67. hash = init_name_hash();
  68. i = min(qstr->len, 30u);
  69. for (; i > 0; name++, i--)
  70. hash = partial_name_hash(toupper(*name), hash);
  71. qstr->hash = end_name_hash(hash);
  72. return 0;
  73. }
  74. static int
  75. affs_hash_dentry(struct dentry *dentry, struct qstr *qstr)
  76. {
  77. return __affs_hash_dentry(dentry, qstr, affs_toupper);
  78. }
  79. static int
  80. affs_intl_hash_dentry(struct dentry *dentry, struct qstr *qstr)
  81. {
  82. return __affs_hash_dentry(dentry, qstr, affs_intl_toupper);
  83. }
  84. static inline int
  85. __affs_compare_dentry(struct dentry *dentry, struct qstr *a, struct qstr *b, toupper_t toupper)
  86. {
  87. const u8 *aname = a->name;
  88. const u8 *bname = b->name;
  89. int len;
  90. /* 'a' is the qstr of an already existing dentry, so the name
  91.  * must be valid. 'b' must be validated first.
  92.  */
  93. if (affs_check_name(b->name,b->len))
  94. return 1;
  95. /* If the names are longer than the allowed 30 chars,
  96.  * the excess is ignored, so their length may differ.
  97.  */
  98. len = a->len;
  99. if (len >= 30) {
  100. if (b->len < 30)
  101. return 1;
  102. len = 30;
  103. } else if (len != b->len)
  104. return 1;
  105. for (; len > 0; len--)
  106. if (toupper(*aname++) != toupper(*bname++))
  107. return 1;
  108. return 0;
  109. }
  110. static int
  111. affs_compare_dentry(struct dentry *dentry, struct qstr *a, struct qstr *b)
  112. {
  113. return __affs_compare_dentry(dentry, a, b, affs_toupper);
  114. }
  115. static int
  116. affs_intl_compare_dentry(struct dentry *dentry, struct qstr *a, struct qstr *b)
  117. {
  118. return __affs_compare_dentry(dentry, a, b, affs_intl_toupper);
  119. }
  120. /*
  121.  * NOTE! unlike strncmp, affs_match returns 1 for success, 0 for failure.
  122.  */
  123. static inline int
  124. affs_match(struct dentry *dentry, const u8 *name2, toupper_t toupper)
  125. {
  126. const u8 *name = dentry->d_name.name;
  127. int len = dentry->d_name.len;
  128. if (len >= 30) {
  129. if (*name2 < 30)
  130. return 0;
  131. len = 30;
  132. } else if (len != *name2)
  133. return 0;
  134. for (name2++; len > 0; len--)
  135. if (toupper(*name++) != toupper(*name2++))
  136. return 0;
  137. return 1;
  138. }
  139. int
  140. affs_hash_name(struct super_block *sb, const u8 *name, unsigned int len)
  141. {
  142. toupper_t toupper = affs_get_toupper(sb);
  143. int hash;
  144. hash = len = min(len, 30u);
  145. for (; len > 0; len--)
  146. hash = (hash * 13 + toupper(*name++)) & 0x7ff;
  147. return hash % AFFS_SB->s_hashsize;
  148. }
  149. static struct buffer_head *
  150. affs_find_entry(struct inode *dir, struct dentry *dentry)
  151. {
  152. struct super_block *sb = dir->i_sb;
  153. struct buffer_head *bh;
  154. toupper_t toupper = affs_get_toupper(sb);
  155. u32 key;
  156. pr_debug("AFFS: find_entry("%.*s")n", (int)dentry->d_name.len, dentry->d_name.name);
  157. bh = affs_bread(sb, dir->i_ino);
  158. if (!bh)
  159. return ERR_PTR(-EIO);
  160. key = be32_to_cpu(AFFS_HEAD(bh)->table[affs_hash_name(sb, dentry->d_name.name, dentry->d_name.len)]);
  161. for (;;) {
  162. affs_brelse(bh);
  163. if (key == 0)
  164. return NULL;
  165. bh = affs_bread(sb, key);
  166. if (!bh)
  167. return ERR_PTR(-EIO);
  168. if (affs_match(dentry, AFFS_TAIL(sb, bh)->name, toupper))
  169. return bh;
  170. key = be32_to_cpu(AFFS_TAIL(sb, bh)->hash_chain);
  171. }
  172. }
  173. struct dentry *
  174. affs_lookup(struct inode *dir, struct dentry *dentry)
  175. {
  176. struct super_block *sb = dir->i_sb;
  177. struct buffer_head *bh;
  178. struct inode *inode = NULL;
  179. pr_debug("AFFS: lookup("%.*s")n",(int)dentry->d_name.len,dentry->d_name.name);
  180. affs_lock_dir(dir);
  181. bh = affs_find_entry(dir, dentry);
  182. affs_unlock_dir(dir);
  183. if (IS_ERR(bh))
  184. return ERR_PTR(PTR_ERR(bh));
  185. if (bh) {
  186. u32 ino = bh->b_blocknr;
  187. /* store the real header ino in d_fsdata for faster lookups */
  188. dentry->d_fsdata = (void *)(long)ino;
  189. switch (be32_to_cpu(AFFS_TAIL(sb, bh)->stype)) {
  190. //link to dirs disabled
  191. //case ST_LINKDIR:
  192. case ST_LINKFILE:
  193. ino = be32_to_cpu(AFFS_TAIL(sb, bh)->original);
  194. }
  195. affs_brelse(bh);
  196. inode = iget(sb, ino);
  197. if (!inode)
  198. return ERR_PTR(-EACCES);
  199. }
  200. dentry->d_op = AFFS_SB->s_flags & SF_INTL ? &affs_intl_dentry_operations : &affs_dentry_operations;
  201. d_add(dentry, inode);
  202. return NULL;
  203. }
  204. int
  205. affs_unlink(struct inode *dir, struct dentry *dentry)
  206. {
  207. pr_debug("AFFS: unlink(dir=%d, "%.*s")n", (u32)dir->i_ino,
  208.  (int)dentry->d_name.len, dentry->d_name.name);
  209. if (!dentry->d_inode)
  210. return -ENOENT;
  211. return affs_remove_header(dentry);
  212. }
  213. int
  214. affs_create(struct inode *dir, struct dentry *dentry, int mode)
  215. {
  216. struct super_block *sb = dir->i_sb;
  217. struct inode *inode;
  218. int  error;
  219. pr_debug("AFFS: create(%lu,"%.*s",0%o)n",dir->i_ino,(int)dentry->d_name.len,
  220.  dentry->d_name.name,mode);
  221. inode = affs_new_inode(dir);
  222. if (!inode)
  223. return -ENOSPC;
  224. inode->i_mode = mode;
  225. mode_to_prot(inode);
  226. mark_inode_dirty(inode);
  227. inode->i_op = &affs_file_inode_operations;
  228. inode->i_fop = &affs_file_operations;
  229. inode->i_mapping->a_ops = (AFFS_SB->s_flags & SF_OFS) ? &affs_aops_ofs : &affs_aops;
  230. error = affs_add_entry(dir, inode, dentry, ST_FILE);
  231. if (error) {
  232. inode->i_nlink = 0;
  233. iput(inode);
  234. return error;
  235. }
  236. return 0;
  237. }
  238. int
  239. affs_mkdir(struct inode *dir, struct dentry *dentry, int mode)
  240. {
  241. struct inode *inode;
  242. int  error;
  243. pr_debug("AFFS: mkdir(%lu,"%.*s",0%o)n",dir->i_ino,
  244.  (int)dentry->d_name.len,dentry->d_name.name,mode);
  245. inode = affs_new_inode(dir);
  246. if (!inode)
  247. return -ENOSPC;
  248. inode->i_mode = S_IFDIR | mode;
  249. mode_to_prot(inode);
  250. inode->i_op = &affs_dir_inode_operations;
  251. inode->i_fop = &affs_dir_operations;
  252. error = affs_add_entry(dir, inode, dentry, ST_USERDIR);
  253. if (error) {
  254. inode->i_nlink = 0;
  255. mark_inode_dirty(inode);
  256. iput(inode);
  257. return error;
  258. }
  259. return 0;
  260. }
  261. int
  262. affs_rmdir(struct inode *dir, struct dentry *dentry)
  263. {
  264. pr_debug("AFFS: rmdir(dir=%u, "%.*s")n", (u32)dir->i_ino,
  265.  (int)dentry->d_name.len, dentry->d_name.name);
  266. if (!dentry->d_inode)
  267. return -ENOENT;
  268. return affs_remove_header(dentry);
  269. }
  270. int
  271. affs_symlink(struct inode *dir, struct dentry *dentry, const char *symname)
  272. {
  273. struct super_block *sb = dir->i_sb;
  274. struct buffer_head *bh;
  275. struct inode *inode;
  276. char *p;
  277. int  i, maxlen, error;
  278. char  c, lc;
  279. pr_debug("AFFS: symlink(%lu,"%.*s" -> "%s")n",dir->i_ino,
  280.  (int)dentry->d_name.len,dentry->d_name.name,symname);
  281. maxlen = AFFS_SB->s_hashsize * sizeof(u32) - 1;
  282. error = -ENOSPC;
  283. inode  = affs_new_inode(dir);
  284. if (!inode)
  285. return -ENOSPC;
  286. inode->i_op = &affs_symlink_inode_operations;
  287. inode->i_data.a_ops = &affs_symlink_aops;
  288. inode->i_mode = S_IFLNK | 0777;
  289. mode_to_prot(inode);
  290. error = -EIO;
  291. bh = affs_bread(sb, inode->i_ino);
  292. if (!bh)
  293. goto err;
  294. i  = 0;
  295. p  = (char *)AFFS_HEAD(bh)->table;
  296. lc = '/';
  297. if (*symname == '/') {
  298. while (*symname == '/')
  299. symname++;
  300. while (AFFS_SB->s_volume[i]) /* Cannot overflow */
  301. *p++ = AFFS_SB->s_volume[i++];
  302. }
  303. while (i < maxlen && (c = *symname++)) {
  304. if (c == '.' && lc == '/' && *symname == '.' && symname[1] == '/') {
  305. *p++ = '/';
  306. i++;
  307. symname += 2;
  308. lc = '/';
  309. } else if (c == '.' && lc == '/' && *symname == '/') {
  310. symname++;
  311. lc = '/';
  312. } else {
  313. *p++ = c;
  314. lc   = c;
  315. i++;
  316. }
  317. if (lc == '/')
  318. while (*symname == '/')
  319. symname++;
  320. }
  321. *p = 0;
  322. mark_buffer_dirty_inode(bh, inode);
  323. affs_brelse(bh);
  324. mark_inode_dirty(inode);
  325. error = affs_add_entry(dir, inode, dentry, ST_SOFTLINK);
  326. if (error)
  327. goto err;
  328. return 0;
  329. err:
  330. inode->i_nlink = 0;
  331. mark_inode_dirty(inode);
  332. iput(inode);
  333. return error;
  334. }
  335. int
  336. affs_link(struct dentry *old_dentry, struct inode *dir, struct dentry *dentry)
  337. {
  338. struct inode *inode = old_dentry->d_inode;
  339. int error;
  340. pr_debug("AFFS: link(%u, %u, "%.*s")n", (u32)inode->i_ino, (u32)dir->i_ino,
  341.  (int)dentry->d_name.len,dentry->d_name.name);
  342. if (S_ISDIR(inode->i_mode))
  343. return -EPERM;
  344. error = affs_add_entry(dir, inode, dentry, S_ISDIR(inode->i_mode) ? ST_LINKDIR : ST_LINKFILE);
  345. if (error) {
  346. inode->i_nlink = 0;
  347. mark_inode_dirty(inode);
  348. iput(inode);
  349. return error;
  350. }
  351. return 0;
  352. }
  353. int
  354. affs_rename(struct inode *old_dir, struct dentry *old_dentry,
  355.     struct inode *new_dir, struct dentry *new_dentry)
  356. {
  357. struct super_block *sb = old_dir->i_sb;
  358. struct buffer_head *bh = NULL;
  359. int retval;
  360. pr_debug("AFFS: rename(old=%u,"%*s" to new=%u,"%*s")n",
  361.  (u32)old_dir->i_ino, (int)old_dentry->d_name.len, old_dentry->d_name.name,
  362.  (u32)new_dir->i_ino, (int)new_dentry->d_name.len, new_dentry->d_name.name);
  363. if ((retval = affs_check_name(new_dentry->d_name.name,new_dentry->d_name.len)))
  364. goto done;
  365. /* Unlink destination if it already exists */
  366. if (new_dentry->d_inode) {
  367. retval = affs_remove_header(new_dentry);
  368. if (retval)
  369. return retval;
  370. }
  371. retval = -EIO;
  372. bh = affs_bread(sb, old_dentry->d_inode->i_ino);
  373. if (!bh)
  374. goto done;
  375. /* Remove header from its parent directory. */
  376. affs_lock_dir(old_dir);
  377. retval = affs_remove_hash(old_dir, bh);
  378. affs_unlock_dir(old_dir);
  379. if (retval)
  380. goto done;
  381. /* And insert it into the new directory with the new name. */
  382. affs_copy_name(AFFS_TAIL(sb, bh)->name, new_dentry);
  383. affs_fix_checksum(sb, bh);
  384. affs_lock_dir(new_dir);
  385. retval = affs_insert_hash(new_dir, bh);
  386. affs_unlock_dir(new_dir);
  387. /* TODO: move it back to old_dir, if error? */
  388. done:
  389. mark_buffer_dirty_inode(bh, retval ? old_dir : new_dir);
  390. affs_brelse(bh);
  391. return retval;
  392. }