diff options
Diffstat (limited to 'fs/fat')
| -rw-r--r-- | fs/fat/inode.c | 11 | ||||
| -rw-r--r-- | fs/fat/namei_vfat.c | 27 |
2 files changed, 27 insertions, 11 deletions
diff --git a/fs/fat/inode.c b/fs/fat/inode.c index 14da530b05ca..0ce143bd7d56 100644 --- a/fs/fat/inode.c +++ b/fs/fat/inode.c | |||
| @@ -558,7 +558,7 @@ static int fat_statfs(struct dentry *dentry, struct kstatfs *buf) | |||
| 558 | buf->f_bavail = sbi->free_clusters; | 558 | buf->f_bavail = sbi->free_clusters; |
| 559 | buf->f_fsid.val[0] = (u32)id; | 559 | buf->f_fsid.val[0] = (u32)id; |
| 560 | buf->f_fsid.val[1] = (u32)(id >> 32); | 560 | buf->f_fsid.val[1] = (u32)(id >> 32); |
| 561 | buf->f_namelen = sbi->options.isvfat ? 260 : 12; | 561 | buf->f_namelen = sbi->options.isvfat ? FAT_LFN_LEN : 12; |
| 562 | 562 | ||
| 563 | return 0; | 563 | return 0; |
| 564 | } | 564 | } |
| @@ -577,7 +577,7 @@ static inline loff_t fat_i_pos_read(struct msdos_sb_info *sbi, | |||
| 577 | return i_pos; | 577 | return i_pos; |
| 578 | } | 578 | } |
| 579 | 579 | ||
| 580 | static int fat_write_inode(struct inode *inode, int wait) | 580 | static int __fat_write_inode(struct inode *inode, int wait) |
| 581 | { | 581 | { |
| 582 | struct super_block *sb = inode->i_sb; | 582 | struct super_block *sb = inode->i_sb; |
| 583 | struct msdos_sb_info *sbi = MSDOS_SB(sb); | 583 | struct msdos_sb_info *sbi = MSDOS_SB(sb); |
| @@ -634,9 +634,14 @@ retry: | |||
| 634 | return err; | 634 | return err; |
| 635 | } | 635 | } |
| 636 | 636 | ||
| 637 | static int fat_write_inode(struct inode *inode, struct writeback_control *wbc) | ||
| 638 | { | ||
| 639 | return __fat_write_inode(inode, wbc->sync_mode == WB_SYNC_ALL); | ||
| 640 | } | ||
| 641 | |||
| 637 | int fat_sync_inode(struct inode *inode) | 642 | int fat_sync_inode(struct inode *inode) |
| 638 | { | 643 | { |
| 639 | return fat_write_inode(inode, 1); | 644 | return __fat_write_inode(inode, 1); |
| 640 | } | 645 | } |
| 641 | 646 | ||
| 642 | EXPORT_SYMBOL_GPL(fat_sync_inode); | 647 | EXPORT_SYMBOL_GPL(fat_sync_inode); |
diff --git a/fs/fat/namei_vfat.c b/fs/fat/namei_vfat.c index f565f24019b5..c1ef50154868 100644 --- a/fs/fat/namei_vfat.c +++ b/fs/fat/namei_vfat.c | |||
| @@ -502,14 +502,14 @@ xlate_to_uni(const unsigned char *name, int len, unsigned char *outname, | |||
| 502 | *outlen = utf8s_to_utf16s(name, len, (wchar_t *)outname); | 502 | *outlen = utf8s_to_utf16s(name, len, (wchar_t *)outname); |
| 503 | if (*outlen < 0) | 503 | if (*outlen < 0) |
| 504 | return *outlen; | 504 | return *outlen; |
| 505 | else if (*outlen > 255) | 505 | else if (*outlen > FAT_LFN_LEN) |
| 506 | return -ENAMETOOLONG; | 506 | return -ENAMETOOLONG; |
| 507 | 507 | ||
| 508 | op = &outname[*outlen * sizeof(wchar_t)]; | 508 | op = &outname[*outlen * sizeof(wchar_t)]; |
| 509 | } else { | 509 | } else { |
| 510 | if (nls) { | 510 | if (nls) { |
| 511 | for (i = 0, ip = name, op = outname, *outlen = 0; | 511 | for (i = 0, ip = name, op = outname, *outlen = 0; |
| 512 | i < len && *outlen <= 255; | 512 | i < len && *outlen <= FAT_LFN_LEN; |
| 513 | *outlen += 1) | 513 | *outlen += 1) |
| 514 | { | 514 | { |
| 515 | if (escape && (*ip == ':')) { | 515 | if (escape && (*ip == ':')) { |
| @@ -549,7 +549,7 @@ xlate_to_uni(const unsigned char *name, int len, unsigned char *outname, | |||
| 549 | return -ENAMETOOLONG; | 549 | return -ENAMETOOLONG; |
| 550 | } else { | 550 | } else { |
| 551 | for (i = 0, ip = name, op = outname, *outlen = 0; | 551 | for (i = 0, ip = name, op = outname, *outlen = 0; |
| 552 | i < len && *outlen <= 255; | 552 | i < len && *outlen <= FAT_LFN_LEN; |
| 553 | i++, *outlen += 1) | 553 | i++, *outlen += 1) |
| 554 | { | 554 | { |
| 555 | *op++ = *ip++; | 555 | *op++ = *ip++; |
| @@ -701,6 +701,15 @@ static int vfat_find(struct inode *dir, struct qstr *qname, | |||
| 701 | return fat_search_long(dir, qname->name, len, sinfo); | 701 | return fat_search_long(dir, qname->name, len, sinfo); |
| 702 | } | 702 | } |
| 703 | 703 | ||
| 704 | /* | ||
| 705 | * (nfsd's) anonymous disconnected dentry? | ||
| 706 | * NOTE: !IS_ROOT() is not anonymous (I.e. d_splice_alias() did the job). | ||
| 707 | */ | ||
| 708 | static int vfat_d_anon_disconn(struct dentry *dentry) | ||
| 709 | { | ||
| 710 | return IS_ROOT(dentry) && (dentry->d_flags & DCACHE_DISCONNECTED); | ||
| 711 | } | ||
| 712 | |||
| 704 | static struct dentry *vfat_lookup(struct inode *dir, struct dentry *dentry, | 713 | static struct dentry *vfat_lookup(struct inode *dir, struct dentry *dentry, |
| 705 | struct nameidata *nd) | 714 | struct nameidata *nd) |
| 706 | { | 715 | { |
| @@ -729,11 +738,11 @@ static struct dentry *vfat_lookup(struct inode *dir, struct dentry *dentry, | |||
| 729 | } | 738 | } |
| 730 | 739 | ||
| 731 | alias = d_find_alias(inode); | 740 | alias = d_find_alias(inode); |
| 732 | if (alias && !(alias->d_flags & DCACHE_DISCONNECTED)) { | 741 | if (alias && !vfat_d_anon_disconn(alias)) { |
| 733 | /* | 742 | /* |
| 734 | * This inode has non DCACHE_DISCONNECTED dentry. This | 743 | * This inode has non anonymous-DCACHE_DISCONNECTED |
| 735 | * means, the user did ->lookup() by an another name | 744 | * dentry. This means, the user did ->lookup() by an |
| 736 | * (longname vs 8.3 alias of it) in past. | 745 | * another name (longname vs 8.3 alias of it) in past. |
| 737 | * | 746 | * |
| 738 | * Switch to new one for reason of locality if possible. | 747 | * Switch to new one for reason of locality if possible. |
| 739 | */ | 748 | */ |
| @@ -743,7 +752,9 @@ static struct dentry *vfat_lookup(struct inode *dir, struct dentry *dentry, | |||
| 743 | iput(inode); | 752 | iput(inode); |
| 744 | unlock_super(sb); | 753 | unlock_super(sb); |
| 745 | return alias; | 754 | return alias; |
| 746 | } | 755 | } else |
| 756 | dput(alias); | ||
| 757 | |||
| 747 | out: | 758 | out: |
| 748 | unlock_super(sb); | 759 | unlock_super(sb); |
| 749 | dentry->d_op = sb->s_root->d_op; | 760 | dentry->d_op = sb->s_root->d_op; |
