diff options
| -rw-r--r-- | fs/fat/fat.h | 1 | ||||
| -rw-r--r-- | fs/fat/inode.c | 15 | ||||
| -rw-r--r-- | fs/fat/nfs.c | 41 |
3 files changed, 52 insertions, 5 deletions
diff --git a/fs/fat/fat.h b/fs/fat/fat.h index c517fc066cf7..413eaaf30ed5 100644 --- a/fs/fat/fat.h +++ b/fs/fat/fat.h | |||
| @@ -75,6 +75,7 @@ struct msdos_sb_info { | |||
| 75 | unsigned long root_cluster; /* first cluster of the root directory */ | 75 | unsigned long root_cluster; /* first cluster of the root directory */ |
| 76 | unsigned long fsinfo_sector; /* sector number of FAT32 fsinfo */ | 76 | unsigned long fsinfo_sector; /* sector number of FAT32 fsinfo */ |
| 77 | struct mutex fat_lock; | 77 | struct mutex fat_lock; |
| 78 | struct mutex nfs_build_inode_lock; | ||
| 78 | struct mutex s_lock; | 79 | struct mutex s_lock; |
| 79 | unsigned int prev_free; /* previously allocated cluster number */ | 80 | unsigned int prev_free; /* previously allocated cluster number */ |
| 80 | unsigned int free_clusters; /* -1 if undefined */ | 81 | unsigned int free_clusters; /* -1 if undefined */ |
diff --git a/fs/fat/inode.c b/fs/fat/inode.c index 27f49f4fa0d3..971ba7d549da 100644 --- a/fs/fat/inode.c +++ b/fs/fat/inode.c | |||
| @@ -443,12 +443,25 @@ static int fat_fill_inode(struct inode *inode, struct msdos_dir_entry *de) | |||
| 443 | return 0; | 443 | return 0; |
| 444 | } | 444 | } |
| 445 | 445 | ||
| 446 | static inline void fat_lock_build_inode(struct msdos_sb_info *sbi) | ||
| 447 | { | ||
| 448 | if (sbi->options.nfs == FAT_NFS_NOSTALE_RO) | ||
| 449 | mutex_lock(&sbi->nfs_build_inode_lock); | ||
| 450 | } | ||
| 451 | |||
| 452 | static inline void fat_unlock_build_inode(struct msdos_sb_info *sbi) | ||
| 453 | { | ||
| 454 | if (sbi->options.nfs == FAT_NFS_NOSTALE_RO) | ||
| 455 | mutex_unlock(&sbi->nfs_build_inode_lock); | ||
| 456 | } | ||
| 457 | |||
| 446 | struct inode *fat_build_inode(struct super_block *sb, | 458 | struct inode *fat_build_inode(struct super_block *sb, |
| 447 | struct msdos_dir_entry *de, loff_t i_pos) | 459 | struct msdos_dir_entry *de, loff_t i_pos) |
| 448 | { | 460 | { |
| 449 | struct inode *inode; | 461 | struct inode *inode; |
| 450 | int err; | 462 | int err; |
| 451 | 463 | ||
| 464 | fat_lock_build_inode(MSDOS_SB(sb)); | ||
| 452 | inode = fat_iget(sb, i_pos); | 465 | inode = fat_iget(sb, i_pos); |
| 453 | if (inode) | 466 | if (inode) |
| 454 | goto out; | 467 | goto out; |
| @@ -468,6 +481,7 @@ struct inode *fat_build_inode(struct super_block *sb, | |||
| 468 | fat_attach(inode, i_pos); | 481 | fat_attach(inode, i_pos); |
| 469 | insert_inode_hash(inode); | 482 | insert_inode_hash(inode); |
| 470 | out: | 483 | out: |
| 484 | fat_unlock_build_inode(MSDOS_SB(sb)); | ||
| 471 | return inode; | 485 | return inode; |
| 472 | } | 486 | } |
| 473 | 487 | ||
| @@ -1247,6 +1261,7 @@ int fat_fill_super(struct super_block *sb, void *data, int silent, int isvfat, | |||
| 1247 | sb->s_magic = MSDOS_SUPER_MAGIC; | 1261 | sb->s_magic = MSDOS_SUPER_MAGIC; |
| 1248 | sb->s_op = &fat_sops; | 1262 | sb->s_op = &fat_sops; |
| 1249 | sb->s_export_op = &fat_export_ops; | 1263 | sb->s_export_op = &fat_export_ops; |
| 1264 | mutex_init(&sbi->nfs_build_inode_lock); | ||
| 1250 | ratelimit_state_init(&sbi->ratelimit, DEFAULT_RATELIMIT_INTERVAL, | 1265 | ratelimit_state_init(&sbi->ratelimit, DEFAULT_RATELIMIT_INTERVAL, |
| 1251 | DEFAULT_RATELIMIT_BURST); | 1266 | DEFAULT_RATELIMIT_BURST); |
| 1252 | 1267 | ||
diff --git a/fs/fat/nfs.c b/fs/fat/nfs.c index d59c02543a10..0748196889b2 100644 --- a/fs/fat/nfs.c +++ b/fs/fat/nfs.c | |||
| @@ -50,19 +50,50 @@ static struct inode *fat_dget(struct super_block *sb, int i_logstart) | |||
| 50 | return inode; | 50 | return inode; |
| 51 | } | 51 | } |
| 52 | 52 | ||
| 53 | static struct inode *fat_ilookup(struct super_block *sb, u64 ino, loff_t i_pos) | ||
| 54 | { | ||
| 55 | if (MSDOS_SB(sb)->options.nfs == FAT_NFS_NOSTALE_RO) | ||
| 56 | return fat_iget(sb, i_pos); | ||
| 57 | |||
| 58 | else { | ||
| 59 | if ((ino < MSDOS_ROOT_INO) || (ino == MSDOS_FSINFO_INO)) | ||
| 60 | return NULL; | ||
| 61 | return ilookup(sb, ino); | ||
| 62 | } | ||
| 63 | } | ||
| 64 | |||
| 53 | static struct inode *__fat_nfs_get_inode(struct super_block *sb, | 65 | static struct inode *__fat_nfs_get_inode(struct super_block *sb, |
| 54 | u64 ino, u32 generation, loff_t i_pos) | 66 | u64 ino, u32 generation, loff_t i_pos) |
| 55 | { | 67 | { |
| 56 | struct inode *inode; | 68 | struct inode *inode = fat_ilookup(sb, ino, i_pos); |
| 57 | |||
| 58 | if ((ino < MSDOS_ROOT_INO) || (ino == MSDOS_FSINFO_INO)) | ||
| 59 | return NULL; | ||
| 60 | 69 | ||
| 61 | inode = ilookup(sb, ino); | ||
| 62 | if (inode && generation && (inode->i_generation != generation)) { | 70 | if (inode && generation && (inode->i_generation != generation)) { |
| 63 | iput(inode); | 71 | iput(inode); |
| 64 | inode = NULL; | 72 | inode = NULL; |
| 65 | } | 73 | } |
| 74 | if (inode == NULL && MSDOS_SB(sb)->options.nfs == FAT_NFS_NOSTALE_RO) { | ||
| 75 | struct buffer_head *bh = NULL; | ||
| 76 | struct msdos_dir_entry *de ; | ||
| 77 | sector_t blocknr; | ||
| 78 | int offset; | ||
| 79 | fat_get_blknr_offset(MSDOS_SB(sb), i_pos, &blocknr, &offset); | ||
| 80 | bh = sb_bread(sb, blocknr); | ||
| 81 | if (!bh) { | ||
| 82 | fat_msg(sb, KERN_ERR, | ||
| 83 | "unable to read block(%llu) for building NFS inode", | ||
| 84 | (llu)blocknr); | ||
| 85 | return inode; | ||
| 86 | } | ||
| 87 | de = (struct msdos_dir_entry *)bh->b_data; | ||
| 88 | /* If a file is deleted on server and client is not updated | ||
| 89 | * yet, we must not build the inode upon a lookup call. | ||
| 90 | */ | ||
| 91 | if (IS_FREE(de[offset].name)) | ||
| 92 | inode = NULL; | ||
| 93 | else | ||
| 94 | inode = fat_build_inode(sb, &de[offset], i_pos); | ||
| 95 | brelse(bh); | ||
| 96 | } | ||
| 66 | 97 | ||
| 67 | return inode; | 98 | return inode; |
| 68 | } | 99 | } |
