aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--fs/nfs/dir.c54
-rw-r--r--fs/nfs/inode.c1
-rw-r--r--include/linux/nfs_fs.h3
3 files changed, 50 insertions, 8 deletions
diff --git a/fs/nfs/dir.c b/fs/nfs/dir.c
index c5c71cb62fbd..cda73814f666 100644
--- a/fs/nfs/dir.c
+++ b/fs/nfs/dir.c
@@ -44,6 +44,7 @@
44/* #define NFS_DEBUG_VERBOSE 1 */ 44/* #define NFS_DEBUG_VERBOSE 1 */
45 45
46static int nfs_opendir(struct inode *, struct file *); 46static int nfs_opendir(struct inode *, struct file *);
47static int nfs_closedir(struct inode *, struct file *);
47static int nfs_readdir(struct file *, void *, filldir_t); 48static int nfs_readdir(struct file *, void *, filldir_t);
48static struct dentry *nfs_lookup(struct inode *, struct dentry *, struct nameidata *); 49static struct dentry *nfs_lookup(struct inode *, struct dentry *, struct nameidata *);
49static int nfs_create(struct inode *, struct dentry *, int, struct nameidata *); 50static int nfs_create(struct inode *, struct dentry *, int, struct nameidata *);
@@ -64,7 +65,7 @@ const struct file_operations nfs_dir_operations = {
64 .read = generic_read_dir, 65 .read = generic_read_dir,
65 .readdir = nfs_readdir, 66 .readdir = nfs_readdir,
66 .open = nfs_opendir, 67 .open = nfs_opendir,
67 .release = nfs_release, 68 .release = nfs_closedir,
68 .fsync = nfs_fsync_dir, 69 .fsync = nfs_fsync_dir,
69}; 70};
70 71
@@ -133,13 +134,33 @@ const struct inode_operations nfs4_dir_inode_operations = {
133 134
134#endif /* CONFIG_NFS_V4 */ 135#endif /* CONFIG_NFS_V4 */
135 136
137static struct nfs_open_dir_context *alloc_nfs_open_dir_context(struct rpc_cred *cred)
138{
139 struct nfs_open_dir_context *ctx;
140 ctx = kmalloc(sizeof(*ctx), GFP_KERNEL);
141 if (ctx != NULL) {
142 ctx->dir_cookie = 0;
143 ctx->cred = get_rpccred(cred);
144 } else
145 ctx = ERR_PTR(-ENOMEM);
146 return ctx;
147}
148
149static void put_nfs_open_dir_context(struct nfs_open_dir_context *ctx)
150{
151 put_rpccred(ctx->cred);
152 kfree(ctx);
153}
154
136/* 155/*
137 * Open file 156 * Open file
138 */ 157 */
139static int 158static int
140nfs_opendir(struct inode *inode, struct file *filp) 159nfs_opendir(struct inode *inode, struct file *filp)
141{ 160{
142 int res; 161 int res = 0;
162 struct nfs_open_dir_context *ctx;
163 struct rpc_cred *cred;
143 164
144 dfprintk(FILE, "NFS: open dir(%s/%s)\n", 165 dfprintk(FILE, "NFS: open dir(%s/%s)\n",
145 filp->f_path.dentry->d_parent->d_name.name, 166 filp->f_path.dentry->d_parent->d_name.name,
@@ -147,8 +168,15 @@ nfs_opendir(struct inode *inode, struct file *filp)
147 168
148 nfs_inc_stats(inode, NFSIOS_VFSOPEN); 169 nfs_inc_stats(inode, NFSIOS_VFSOPEN);
149 170
150 /* Call generic open code in order to cache credentials */ 171 cred = rpc_lookup_cred();
151 res = nfs_open(inode, filp); 172 if (IS_ERR(cred))
173 return PTR_ERR(cred);
174 ctx = alloc_nfs_open_dir_context(cred);
175 if (IS_ERR(ctx)) {
176 res = PTR_ERR(ctx);
177 goto out;
178 }
179 filp->private_data = ctx;
152 if (filp->f_path.dentry == filp->f_path.mnt->mnt_root) { 180 if (filp->f_path.dentry == filp->f_path.mnt->mnt_root) {
153 /* This is a mountpoint, so d_revalidate will never 181 /* This is a mountpoint, so d_revalidate will never
154 * have been called, so we need to refresh the 182 * have been called, so we need to refresh the
@@ -156,9 +184,18 @@ nfs_opendir(struct inode *inode, struct file *filp)
156 */ 184 */
157 __nfs_revalidate_inode(NFS_SERVER(inode), inode); 185 __nfs_revalidate_inode(NFS_SERVER(inode), inode);
158 } 186 }
187out:
188 put_rpccred(cred);
159 return res; 189 return res;
160} 190}
161 191
192static int
193nfs_closedir(struct inode *inode, struct file *filp)
194{
195 put_nfs_open_dir_context(filp->private_data);
196 return 0;
197}
198
162struct nfs_cache_array_entry { 199struct nfs_cache_array_entry {
163 u64 cookie; 200 u64 cookie;
164 u64 ino; 201 u64 ino;
@@ -355,7 +392,8 @@ static
355int nfs_readdir_xdr_filler(struct page **pages, nfs_readdir_descriptor_t *desc, 392int nfs_readdir_xdr_filler(struct page **pages, nfs_readdir_descriptor_t *desc,
356 struct nfs_entry *entry, struct file *file, struct inode *inode) 393 struct nfs_entry *entry, struct file *file, struct inode *inode)
357{ 394{
358 struct rpc_cred *cred = nfs_file_cred(file); 395 struct nfs_open_dir_context *ctx = file->private_data;
396 struct rpc_cred *cred = ctx->cred;
359 unsigned long timestamp, gencount; 397 unsigned long timestamp, gencount;
360 int error; 398 int error;
361 399
@@ -786,6 +824,7 @@ static int nfs_readdir(struct file *filp, void *dirent, filldir_t filldir)
786 struct inode *inode = dentry->d_inode; 824 struct inode *inode = dentry->d_inode;
787 nfs_readdir_descriptor_t my_desc, 825 nfs_readdir_descriptor_t my_desc,
788 *desc = &my_desc; 826 *desc = &my_desc;
827 struct nfs_open_dir_context *dir_ctx = filp->private_data;
789 int res; 828 int res;
790 829
791 dfprintk(FILE, "NFS: readdir(%s/%s) starting at cookie %llu\n", 830 dfprintk(FILE, "NFS: readdir(%s/%s) starting at cookie %llu\n",
@@ -802,7 +841,7 @@ static int nfs_readdir(struct file *filp, void *dirent, filldir_t filldir)
802 memset(desc, 0, sizeof(*desc)); 841 memset(desc, 0, sizeof(*desc));
803 842
804 desc->file = filp; 843 desc->file = filp;
805 desc->dir_cookie = &nfs_file_open_context(filp)->dir_cookie; 844 desc->dir_cookie = &dir_ctx->dir_cookie;
806 desc->decode = NFS_PROTO(inode)->decode_dirent; 845 desc->decode = NFS_PROTO(inode)->decode_dirent;
807 desc->plus = NFS_USE_READDIRPLUS(inode); 846 desc->plus = NFS_USE_READDIRPLUS(inode);
808 847
@@ -854,6 +893,7 @@ static loff_t nfs_llseek_dir(struct file *filp, loff_t offset, int origin)
854{ 893{
855 struct dentry *dentry = filp->f_path.dentry; 894 struct dentry *dentry = filp->f_path.dentry;
856 struct inode *inode = dentry->d_inode; 895 struct inode *inode = dentry->d_inode;
896 struct nfs_open_dir_context *dir_ctx = filp->private_data;
857 897
858 dfprintk(FILE, "NFS: llseek dir(%s/%s, %lld, %d)\n", 898 dfprintk(FILE, "NFS: llseek dir(%s/%s, %lld, %d)\n",
859 dentry->d_parent->d_name.name, 899 dentry->d_parent->d_name.name,
@@ -873,7 +913,7 @@ static loff_t nfs_llseek_dir(struct file *filp, loff_t offset, int origin)
873 } 913 }
874 if (offset != filp->f_pos) { 914 if (offset != filp->f_pos) {
875 filp->f_pos = offset; 915 filp->f_pos = offset;
876 nfs_file_open_context(filp)->dir_cookie = 0; 916 dir_ctx->dir_cookie = 0;
877 } 917 }
878out: 918out:
879 mutex_unlock(&inode->i_mutex); 919 mutex_unlock(&inode->i_mutex);
diff --git a/fs/nfs/inode.c b/fs/nfs/inode.c
index 2f8e61816d75..477a2e512b39 100644
--- a/fs/nfs/inode.c
+++ b/fs/nfs/inode.c
@@ -639,7 +639,6 @@ struct nfs_open_context *alloc_nfs_open_context(struct path *path, struct rpc_cr
639 ctx->mode = f_mode; 639 ctx->mode = f_mode;
640 ctx->flags = 0; 640 ctx->flags = 0;
641 ctx->error = 0; 641 ctx->error = 0;
642 ctx->dir_cookie = 0;
643 nfs_init_lock_context(&ctx->lock_context); 642 nfs_init_lock_context(&ctx->lock_context);
644 ctx->lock_context.open_context = ctx; 643 ctx->lock_context.open_context = ctx;
645 INIT_LIST_HEAD(&ctx->list); 644 INIT_LIST_HEAD(&ctx->list);
diff --git a/include/linux/nfs_fs.h b/include/linux/nfs_fs.h
index cb2add401f25..4b851a02f7f3 100644
--- a/include/linux/nfs_fs.h
+++ b/include/linux/nfs_fs.h
@@ -95,7 +95,10 @@ struct nfs_open_context {
95 int error; 95 int error;
96 96
97 struct list_head list; 97 struct list_head list;
98};
98 99
100struct nfs_open_dir_context {
101 struct rpc_cred *cred;
99 __u64 dir_cookie; 102 __u64 dir_cookie;
100}; 103};
101 104