2 * linux/fs/affs/namei.c
4 * (c) 1996 Hans-Joachim Widmaier - Rewritten
6 * (C) 1993 Ray Burr - Modified for Amiga FFS filesystem.
8 * (C) 1991 Linus Torvalds - minix filesystem
12 #include <linux/exportfs.h>
14 typedef int (*toupper_t
)(int);
16 /* Simple toupper() for DOS\1 */
21 return ch
>= 'a' && ch
<= 'z' ? ch
-= ('a' - 'A') : ch
;
24 /* International toupper() for DOS\3 ("international") */
27 affs_intl_toupper(int ch
)
29 return (ch
>= 'a' && ch
<= 'z') || (ch
>= 0xE0
30 && ch
<= 0xFE && ch
!= 0xF7) ?
31 ch
- ('a' - 'A') : ch
;
34 static inline toupper_t
35 affs_get_toupper(struct super_block
*sb
)
37 return affs_test_opt(AFFS_SB(sb
)->s_flags
, SF_INTL
) ?
38 affs_intl_toupper
: affs_toupper
;
42 * Note: the dentry argument is the parent dentry.
45 __affs_hash_dentry(const struct dentry
*dentry
, struct qstr
*qstr
, toupper_t toupper
, bool notruncate
)
47 const u8
*name
= qstr
->name
;
52 retval
= affs_check_name(qstr
->name
, qstr
->len
, notruncate
);
56 hash
= init_name_hash(dentry
);
57 len
= min(qstr
->len
, AFFSNAMEMAX
);
58 for (; len
> 0; name
++, len
--)
59 hash
= partial_name_hash(toupper(*name
), hash
);
60 qstr
->hash
= end_name_hash(hash
);
66 affs_hash_dentry(const struct dentry
*dentry
, struct qstr
*qstr
)
68 return __affs_hash_dentry(dentry
, qstr
, affs_toupper
,
69 affs_nofilenametruncate(dentry
));
74 affs_intl_hash_dentry(const struct dentry
*dentry
, struct qstr
*qstr
)
76 return __affs_hash_dentry(dentry
, qstr
, affs_intl_toupper
,
77 affs_nofilenametruncate(dentry
));
81 static inline int __affs_compare_dentry(unsigned int len
,
82 const char *str
, const struct qstr
*name
, toupper_t toupper
,
85 const u8
*aname
= str
;
86 const u8
*bname
= name
->name
;
89 * 'str' is the name of an already existing dentry, so the name
90 * must be valid. 'name' must be validated first.
93 if (affs_check_name(name
->name
, name
->len
, notruncate
))
97 * If the names are longer than the allowed 30 chars,
98 * the excess is ignored, so their length may differ.
100 if (len
>= AFFSNAMEMAX
) {
101 if (name
->len
< AFFSNAMEMAX
)
104 } else if (len
!= name
->len
)
107 for (; len
> 0; len
--)
108 if (toupper(*aname
++) != toupper(*bname
++))
115 affs_compare_dentry(const struct dentry
*dentry
,
116 unsigned int len
, const char *str
, const struct qstr
*name
)
119 return __affs_compare_dentry(len
, str
, name
, affs_toupper
,
120 affs_nofilenametruncate(dentry
));
124 affs_intl_compare_dentry(const struct dentry
*dentry
,
125 unsigned int len
, const char *str
, const struct qstr
*name
)
127 return __affs_compare_dentry(len
, str
, name
, affs_intl_toupper
,
128 affs_nofilenametruncate(dentry
));
133 * NOTE! unlike strncmp, affs_match returns 1 for success, 0 for failure.
137 affs_match(struct dentry
*dentry
, const u8
*name2
, toupper_t toupper
)
139 const u8
*name
= dentry
->d_name
.name
;
140 int len
= dentry
->d_name
.len
;
142 if (len
>= AFFSNAMEMAX
) {
143 if (*name2
< AFFSNAMEMAX
)
146 } else if (len
!= *name2
)
149 for (name2
++; len
> 0; len
--)
150 if (toupper(*name
++) != toupper(*name2
++))
156 affs_hash_name(struct super_block
*sb
, const u8
*name
, unsigned int len
)
158 toupper_t toupper
= affs_get_toupper(sb
);
161 hash
= len
= min(len
, AFFSNAMEMAX
);
162 for (; len
> 0; len
--)
163 hash
= (hash
* 13 + toupper(*name
++)) & 0x7ff;
165 return hash
% AFFS_SB(sb
)->s_hashsize
;
168 static struct buffer_head
*
169 affs_find_entry(struct inode
*dir
, struct dentry
*dentry
)
171 struct super_block
*sb
= dir
->i_sb
;
172 struct buffer_head
*bh
;
173 toupper_t toupper
= affs_get_toupper(sb
);
176 pr_debug("%s(\"%pd\")\n", __func__
, dentry
);
178 bh
= affs_bread(sb
, dir
->i_ino
);
180 return ERR_PTR(-EIO
);
182 key
= be32_to_cpu(AFFS_HEAD(bh
)->table
[affs_hash_name(sb
, dentry
->d_name
.name
, dentry
->d_name
.len
)]);
188 bh
= affs_bread(sb
, key
);
190 return ERR_PTR(-EIO
);
191 if (affs_match(dentry
, AFFS_TAIL(sb
, bh
)->name
, toupper
))
193 key
= be32_to_cpu(AFFS_TAIL(sb
, bh
)->hash_chain
);
198 affs_lookup(struct inode
*dir
, struct dentry
*dentry
, unsigned int flags
)
200 struct super_block
*sb
= dir
->i_sb
;
201 struct buffer_head
*bh
;
202 struct inode
*inode
= NULL
;
204 pr_debug("%s(\"%pd\")\n", __func__
, dentry
);
207 bh
= affs_find_entry(dir
, dentry
);
208 affs_unlock_dir(dir
);
212 u32 ino
= bh
->b_blocknr
;
214 /* store the real header ino in d_fsdata for faster lookups */
215 dentry
->d_fsdata
= (void *)(long)ino
;
216 switch (be32_to_cpu(AFFS_TAIL(sb
, bh
)->stype
)) {
217 //link to dirs disabled
220 ino
= be32_to_cpu(AFFS_TAIL(sb
, bh
)->original
);
223 inode
= affs_iget(sb
, ino
);
225 return ERR_CAST(inode
);
227 d_add(dentry
, inode
);
232 affs_unlink(struct inode
*dir
, struct dentry
*dentry
)
234 pr_debug("%s(dir=%lu, %lu \"%pd\")\n", __func__
, dir
->i_ino
,
235 d_inode(dentry
)->i_ino
, dentry
);
237 return affs_remove_header(dentry
);
241 affs_create(struct inode
*dir
, struct dentry
*dentry
, umode_t mode
, bool excl
)
243 struct super_block
*sb
= dir
->i_sb
;
247 pr_debug("%s(%lu,\"%pd\",0%ho)\n",
248 __func__
, dir
->i_ino
, dentry
, mode
);
250 inode
= affs_new_inode(dir
);
254 inode
->i_mode
= mode
;
255 affs_mode_to_prot(inode
);
256 mark_inode_dirty(inode
);
258 inode
->i_op
= &affs_file_inode_operations
;
259 inode
->i_fop
= &affs_file_operations
;
260 inode
->i_mapping
->a_ops
= affs_test_opt(AFFS_SB(sb
)->s_flags
, SF_OFS
) ?
261 &affs_aops_ofs
: &affs_aops
;
262 error
= affs_add_entry(dir
, inode
, dentry
, ST_FILE
);
272 affs_mkdir(struct inode
*dir
, struct dentry
*dentry
, umode_t mode
)
277 pr_debug("%s(%lu,\"%pd\",0%ho)\n",
278 __func__
, dir
->i_ino
, dentry
, mode
);
280 inode
= affs_new_inode(dir
);
284 inode
->i_mode
= S_IFDIR
| mode
;
285 affs_mode_to_prot(inode
);
287 inode
->i_op
= &affs_dir_inode_operations
;
288 inode
->i_fop
= &affs_dir_operations
;
290 error
= affs_add_entry(dir
, inode
, dentry
, ST_USERDIR
);
293 mark_inode_dirty(inode
);
301 affs_rmdir(struct inode
*dir
, struct dentry
*dentry
)
303 pr_debug("%s(dir=%lu, %lu \"%pd\")\n", __func__
, dir
->i_ino
,
304 d_inode(dentry
)->i_ino
, dentry
);
306 return affs_remove_header(dentry
);
310 affs_symlink(struct inode
*dir
, struct dentry
*dentry
, const char *symname
)
312 struct super_block
*sb
= dir
->i_sb
;
313 struct buffer_head
*bh
;
316 int i
, maxlen
, error
;
319 pr_debug("%s(%lu,\"%pd\" -> \"%s\")\n",
320 __func__
, dir
->i_ino
, dentry
, symname
);
322 maxlen
= AFFS_SB(sb
)->s_hashsize
* sizeof(u32
) - 1;
323 inode
= affs_new_inode(dir
);
327 inode
->i_op
= &affs_symlink_inode_operations
;
328 inode_nohighmem(inode
);
329 inode
->i_data
.a_ops
= &affs_symlink_aops
;
330 inode
->i_mode
= S_IFLNK
| 0777;
331 affs_mode_to_prot(inode
);
334 bh
= affs_bread(sb
, inode
->i_ino
);
338 p
= (char *)AFFS_HEAD(bh
)->table
;
340 if (*symname
== '/') {
341 struct affs_sb_info
*sbi
= AFFS_SB(sb
);
342 while (*symname
== '/')
344 spin_lock(&sbi
->symlink_lock
);
345 while (sbi
->s_volume
[i
]) /* Cannot overflow */
346 *p
++ = sbi
->s_volume
[i
++];
347 spin_unlock(&sbi
->symlink_lock
);
349 while (i
< maxlen
&& (c
= *symname
++)) {
350 if (c
== '.' && lc
== '/' && *symname
== '.' && symname
[1] == '/') {
355 } else if (c
== '.' && lc
== '/' && *symname
== '/') {
364 while (*symname
== '/')
368 mark_buffer_dirty_inode(bh
, inode
);
370 mark_inode_dirty(inode
);
372 error
= affs_add_entry(dir
, inode
, dentry
, ST_SOFTLINK
);
380 mark_inode_dirty(inode
);
386 affs_link(struct dentry
*old_dentry
, struct inode
*dir
, struct dentry
*dentry
)
388 struct inode
*inode
= d_inode(old_dentry
);
390 pr_debug("%s(%lu, %lu, \"%pd\")\n", __func__
, inode
->i_ino
, dir
->i_ino
,
393 return affs_add_entry(dir
, inode
, dentry
, ST_LINKFILE
);
397 affs_rename(struct inode
*old_dir
, struct dentry
*old_dentry
,
398 struct inode
*new_dir
, struct dentry
*new_dentry
,
401 struct super_block
*sb
= old_dir
->i_sb
;
402 struct buffer_head
*bh
= NULL
;
405 if (flags
& ~RENAME_NOREPLACE
)
408 pr_debug("%s(old=%lu,\"%pd\" to new=%lu,\"%pd\")\n", __func__
,
409 old_dir
->i_ino
, old_dentry
, new_dir
->i_ino
, new_dentry
);
411 retval
= affs_check_name(new_dentry
->d_name
.name
,
412 new_dentry
->d_name
.len
,
413 affs_nofilenametruncate(old_dentry
));
418 /* Unlink destination if it already exists */
419 if (d_really_is_positive(new_dentry
)) {
420 retval
= affs_remove_header(new_dentry
);
425 bh
= affs_bread(sb
, d_inode(old_dentry
)->i_ino
);
429 /* Remove header from its parent directory. */
430 affs_lock_dir(old_dir
);
431 retval
= affs_remove_hash(old_dir
, bh
);
432 affs_unlock_dir(old_dir
);
436 /* And insert it into the new directory with the new name. */
437 affs_copy_name(AFFS_TAIL(sb
, bh
)->name
, new_dentry
);
438 affs_fix_checksum(sb
, bh
);
439 affs_lock_dir(new_dir
);
440 retval
= affs_insert_hash(new_dir
, bh
);
441 affs_unlock_dir(new_dir
);
442 /* TODO: move it back to old_dir, if error? */
445 mark_buffer_dirty_inode(bh
, retval
? old_dir
: new_dir
);
450 static struct dentry
*affs_get_parent(struct dentry
*child
)
452 struct inode
*parent
;
453 struct buffer_head
*bh
;
455 bh
= affs_bread(child
->d_sb
, d_inode(child
)->i_ino
);
457 return ERR_PTR(-EIO
);
459 parent
= affs_iget(child
->d_sb
,
460 be32_to_cpu(AFFS_TAIL(child
->d_sb
, bh
)->parent
));
463 return ERR_CAST(parent
);
465 return d_obtain_alias(parent
);
468 static struct inode
*affs_nfs_get_inode(struct super_block
*sb
, u64 ino
,
473 if (!affs_validblock(sb
, ino
))
474 return ERR_PTR(-ESTALE
);
476 inode
= affs_iget(sb
, ino
);
478 return ERR_CAST(inode
);
480 if (generation
&& inode
->i_generation
!= generation
) {
482 return ERR_PTR(-ESTALE
);
488 static struct dentry
*affs_fh_to_dentry(struct super_block
*sb
, struct fid
*fid
,
489 int fh_len
, int fh_type
)
491 return generic_fh_to_dentry(sb
, fid
, fh_len
, fh_type
,
495 static struct dentry
*affs_fh_to_parent(struct super_block
*sb
, struct fid
*fid
,
496 int fh_len
, int fh_type
)
498 return generic_fh_to_parent(sb
, fid
, fh_len
, fh_type
,
502 const struct export_operations affs_export_ops
= {
503 .fh_to_dentry
= affs_fh_to_dentry
,
504 .fh_to_parent
= affs_fh_to_parent
,
505 .get_parent
= affs_get_parent
,
508 const struct dentry_operations affs_dentry_operations
= {
509 .d_hash
= affs_hash_dentry
,
510 .d_compare
= affs_compare_dentry
,
513 const struct dentry_operations affs_intl_dentry_operations
= {
514 .d_hash
= affs_intl_hash_dentry
,
515 .d_compare
= affs_intl_compare_dentry
,