aboutsummaryrefslogtreecommitdiffstats
path: root/fs/configfs
diff options
context:
space:
mode:
authorAl Viro <viro@zeniv.linux.org.uk>2015-05-02 13:32:22 -0400
committerAl Viro <viro@zeniv.linux.org.uk>2015-05-10 22:19:45 -0400
commit680baacbca69d18a6d7315374ad83d05ac9c0977 (patch)
treea69822ef5234d4a308b780ff51c5d9b77bd3a89b /fs/configfs
parent46afd6f61cc33ae4b3a2aed4bb454d11d4114c27 (diff)
new ->follow_link() and ->put_link() calling conventions
a) instead of storing the symlink body (via nd_set_link()) and returning an opaque pointer later passed to ->put_link(), ->follow_link() _stores_ that opaque pointer (into void * passed by address by caller) and returns the symlink body. Returning ERR_PTR() on error, NULL on jump (procfs magic symlinks) and pointer to symlink body for normal symlinks. Stored pointer is ignored in all cases except the last one. Storing NULL for opaque pointer (or not storing it at all) means no call of ->put_link(). b) the body used to be passed to ->put_link() implicitly (via nameidata). Now only the opaque pointer is. In the cases when we used the symlink body to free stuff, ->follow_link() now should store it as opaque pointer in addition to returning it. Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
Diffstat (limited to 'fs/configfs')
-rw-r--r--fs/configfs/symlink.c28
1 files changed, 12 insertions, 16 deletions
diff --git a/fs/configfs/symlink.c b/fs/configfs/symlink.c
index cc9f2546ea4a..fac8e8517f33 100644
--- a/fs/configfs/symlink.c
+++ b/fs/configfs/symlink.c
@@ -279,30 +279,26 @@ static int configfs_getlink(struct dentry *dentry, char * path)
279 279
280} 280}
281 281
282static void *configfs_follow_link(struct dentry *dentry, struct nameidata *nd) 282static const char *configfs_follow_link(struct dentry *dentry, void **cookie, struct nameidata *nd)
283{ 283{
284 int error = -ENOMEM;
285 unsigned long page = get_zeroed_page(GFP_KERNEL); 284 unsigned long page = get_zeroed_page(GFP_KERNEL);
285 int error;
286 286
287 if (page) { 287 if (!page)
288 error = configfs_getlink(dentry, (char *)page); 288 return ERR_PTR(-ENOMEM);
289 if (!error) { 289
290 nd_set_link(nd, (char *)page); 290 error = configfs_getlink(dentry, (char *)page);
291 return (void *)page; 291 if (!error) {
292 } 292 return *cookie = (void *)page;
293 } 293 }
294 294
295 nd_set_link(nd, ERR_PTR(error)); 295 free_page(page);
296 return NULL; 296 return ERR_PTR(error);
297} 297}
298 298
299static void configfs_put_link(struct dentry *dentry, struct nameidata *nd, 299static void configfs_put_link(struct dentry *dentry, void *cookie)
300 void *cookie)
301{ 300{
302 if (cookie) { 301 free_page((unsigned long)cookie);
303 unsigned long page = (unsigned long)cookie;
304 free_page(page);
305 }
306} 302}
307 303
308const struct inode_operations configfs_symlink_inode_operations = { 304const struct inode_operations configfs_symlink_inode_operations = {