diff options
Diffstat (limited to 'fs/nfs/namespace.c')
-rw-r--r-- | fs/nfs/namespace.c | 103 |
1 files changed, 19 insertions, 84 deletions
diff --git a/fs/nfs/namespace.c b/fs/nfs/namespace.c index d51868e5683c..08b9c93675da 100644 --- a/fs/nfs/namespace.c +++ b/fs/nfs/namespace.c | |||
@@ -26,11 +26,6 @@ static LIST_HEAD(nfs_automount_list); | |||
26 | static DECLARE_DELAYED_WORK(nfs_automount_task, nfs_expire_automounts); | 26 | static DECLARE_DELAYED_WORK(nfs_automount_task, nfs_expire_automounts); |
27 | int nfs_mountpoint_expiry_timeout = 500 * HZ; | 27 | int nfs_mountpoint_expiry_timeout = 500 * HZ; |
28 | 28 | ||
29 | static struct vfsmount *nfs_do_submount(struct dentry *dentry, | ||
30 | struct nfs_fh *fh, | ||
31 | struct nfs_fattr *fattr, | ||
32 | rpc_authflavor_t authflavor); | ||
33 | |||
34 | /* | 29 | /* |
35 | * nfs_path - reconstruct the path given an arbitrary dentry | 30 | * nfs_path - reconstruct the path given an arbitrary dentry |
36 | * @base - used to return pointer to the end of devname part of path | 31 | * @base - used to return pointer to the end of devname part of path |
@@ -118,64 +113,6 @@ Elong: | |||
118 | return ERR_PTR(-ENAMETOOLONG); | 113 | return ERR_PTR(-ENAMETOOLONG); |
119 | } | 114 | } |
120 | 115 | ||
121 | #ifdef CONFIG_NFS_V4 | ||
122 | rpc_authflavor_t nfs_find_best_sec(struct nfs4_secinfo_flavors *flavors) | ||
123 | { | ||
124 | struct gss_api_mech *mech; | ||
125 | struct xdr_netobj oid; | ||
126 | int i; | ||
127 | rpc_authflavor_t pseudoflavor = RPC_AUTH_UNIX; | ||
128 | |||
129 | for (i = 0; i < flavors->num_flavors; i++) { | ||
130 | struct nfs4_secinfo_flavor *flavor; | ||
131 | flavor = &flavors->flavors[i]; | ||
132 | |||
133 | if (flavor->flavor == RPC_AUTH_NULL || flavor->flavor == RPC_AUTH_UNIX) { | ||
134 | pseudoflavor = flavor->flavor; | ||
135 | break; | ||
136 | } else if (flavor->flavor == RPC_AUTH_GSS) { | ||
137 | oid.len = flavor->gss.sec_oid4.len; | ||
138 | oid.data = flavor->gss.sec_oid4.data; | ||
139 | mech = gss_mech_get_by_OID(&oid); | ||
140 | if (!mech) | ||
141 | continue; | ||
142 | pseudoflavor = gss_svc_to_pseudoflavor(mech, flavor->gss.service); | ||
143 | gss_mech_put(mech); | ||
144 | break; | ||
145 | } | ||
146 | } | ||
147 | |||
148 | return pseudoflavor; | ||
149 | } | ||
150 | |||
151 | static struct rpc_clnt *nfs_lookup_mountpoint(struct inode *dir, | ||
152 | struct qstr *name, | ||
153 | struct nfs_fh *fh, | ||
154 | struct nfs_fattr *fattr) | ||
155 | { | ||
156 | int err; | ||
157 | |||
158 | if (NFS_PROTO(dir)->version == 4) | ||
159 | return nfs4_proc_lookup_mountpoint(dir, name, fh, fattr); | ||
160 | |||
161 | err = NFS_PROTO(dir)->lookup(NFS_SERVER(dir)->client, dir, name, fh, fattr); | ||
162 | if (err) | ||
163 | return ERR_PTR(err); | ||
164 | return rpc_clone_client(NFS_SERVER(dir)->client); | ||
165 | } | ||
166 | #else /* CONFIG_NFS_V4 */ | ||
167 | static inline struct rpc_clnt *nfs_lookup_mountpoint(struct inode *dir, | ||
168 | struct qstr *name, | ||
169 | struct nfs_fh *fh, | ||
170 | struct nfs_fattr *fattr) | ||
171 | { | ||
172 | int err = NFS_PROTO(dir)->lookup(NFS_SERVER(dir)->client, dir, name, fh, fattr); | ||
173 | if (err) | ||
174 | return ERR_PTR(err); | ||
175 | return rpc_clone_client(NFS_SERVER(dir)->client); | ||
176 | } | ||
177 | #endif /* CONFIG_NFS_V4 */ | ||
178 | |||
179 | /* | 116 | /* |
180 | * nfs_d_automount - Handle crossing a mountpoint on the server | 117 | * nfs_d_automount - Handle crossing a mountpoint on the server |
181 | * @path - The mountpoint | 118 | * @path - The mountpoint |
@@ -191,10 +128,9 @@ static inline struct rpc_clnt *nfs_lookup_mountpoint(struct inode *dir, | |||
191 | struct vfsmount *nfs_d_automount(struct path *path) | 128 | struct vfsmount *nfs_d_automount(struct path *path) |
192 | { | 129 | { |
193 | struct vfsmount *mnt; | 130 | struct vfsmount *mnt; |
194 | struct dentry *parent; | 131 | struct nfs_server *server = NFS_SERVER(path->dentry->d_inode); |
195 | struct nfs_fh *fh = NULL; | 132 | struct nfs_fh *fh = NULL; |
196 | struct nfs_fattr *fattr = NULL; | 133 | struct nfs_fattr *fattr = NULL; |
197 | struct rpc_clnt *client; | ||
198 | 134 | ||
199 | dprintk("--> nfs_d_automount()\n"); | 135 | dprintk("--> nfs_d_automount()\n"); |
200 | 136 | ||
@@ -210,21 +146,7 @@ struct vfsmount *nfs_d_automount(struct path *path) | |||
210 | 146 | ||
211 | dprintk("%s: enter\n", __func__); | 147 | dprintk("%s: enter\n", __func__); |
212 | 148 | ||
213 | /* Look it up again to get its attributes */ | 149 | mnt = server->nfs_client->rpc_ops->submount(server, path->dentry, fh, fattr); |
214 | parent = dget_parent(path->dentry); | ||
215 | client = nfs_lookup_mountpoint(parent->d_inode, &path->dentry->d_name, fh, fattr); | ||
216 | dput(parent); | ||
217 | if (IS_ERR(client)) { | ||
218 | mnt = ERR_CAST(client); | ||
219 | goto out; | ||
220 | } | ||
221 | |||
222 | if (fattr->valid & NFS_ATTR_FATTR_V4_REFERRAL) | ||
223 | mnt = nfs_do_refmount(client, path->dentry); | ||
224 | else | ||
225 | mnt = nfs_do_submount(path->dentry, fh, fattr, client->cl_auth->au_flavor); | ||
226 | rpc_shutdown_client(client); | ||
227 | |||
228 | if (IS_ERR(mnt)) | 150 | if (IS_ERR(mnt)) |
229 | goto out; | 151 | goto out; |
230 | 152 | ||
@@ -297,10 +219,8 @@ static struct vfsmount *nfs_do_clone_mount(struct nfs_server *server, | |||
297 | * @authflavor - security flavor to use when performing the mount | 219 | * @authflavor - security flavor to use when performing the mount |
298 | * | 220 | * |
299 | */ | 221 | */ |
300 | static struct vfsmount *nfs_do_submount(struct dentry *dentry, | 222 | struct vfsmount *nfs_do_submount(struct dentry *dentry, struct nfs_fh *fh, |
301 | struct nfs_fh *fh, | 223 | struct nfs_fattr *fattr, rpc_authflavor_t authflavor) |
302 | struct nfs_fattr *fattr, | ||
303 | rpc_authflavor_t authflavor) | ||
304 | { | 224 | { |
305 | struct nfs_clone_mount mountdata = { | 225 | struct nfs_clone_mount mountdata = { |
306 | .sb = dentry->d_sb, | 226 | .sb = dentry->d_sb, |
@@ -333,3 +253,18 @@ out: | |||
333 | dprintk("<-- nfs_do_submount() = %p\n", mnt); | 253 | dprintk("<-- nfs_do_submount() = %p\n", mnt); |
334 | return mnt; | 254 | return mnt; |
335 | } | 255 | } |
256 | |||
257 | struct vfsmount *nfs_submount(struct nfs_server *server, struct dentry *dentry, | ||
258 | struct nfs_fh *fh, struct nfs_fattr *fattr) | ||
259 | { | ||
260 | int err; | ||
261 | struct dentry *parent = dget_parent(dentry); | ||
262 | |||
263 | /* Look it up again to get its attributes */ | ||
264 | err = server->nfs_client->rpc_ops->lookup(parent->d_inode, &dentry->d_name, fh, fattr); | ||
265 | dput(parent); | ||
266 | if (err != 0) | ||
267 | return ERR_PTR(err); | ||
268 | |||
269 | return nfs_do_submount(dentry, fh, fattr, server->client->cl_auth->au_flavor); | ||
270 | } | ||