aboutsummaryrefslogtreecommitdiffstats
path: root/fs/ocfs2/symlink.c
diff options
context:
space:
mode:
authorAl Viro <viro@zeniv.linux.org.uk>2012-05-03 10:14:29 -0400
committerAl Viro <viro@zeniv.linux.org.uk>2012-05-29 23:28:40 -0400
commitea022dfb3c2a4680483b00eb2fecc9fc4f6091d1 (patch)
treea1fc74b921ef4ea32603492f2a500b016e15e377 /fs/ocfs2/symlink.c
parent408bd629badbd4353b238ab6f58001529b274d73 (diff)
ocfs: simplify symlink handling
seeing that "fast" symlinks still get allocation + copy, we might as well simply switch them to pagecache-based variant of ->follow_link(); just need an appropriate ->readpage() for them... Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
Diffstat (limited to 'fs/ocfs2/symlink.c')
-rw-r--r--fs/ocfs2/symlink.c115
1 files changed, 21 insertions, 94 deletions
diff --git a/fs/ocfs2/symlink.c b/fs/ocfs2/symlink.c
index 5d22872e2bb3..f1fbb4b552ad 100644
--- a/fs/ocfs2/symlink.c
+++ b/fs/ocfs2/symlink.c
@@ -54,101 +54,40 @@
54#include "buffer_head_io.h" 54#include "buffer_head_io.h"
55 55
56 56
57static char *ocfs2_fast_symlink_getlink(struct inode *inode, 57static int ocfs2_fast_symlink_readpage(struct file *unused, struct page *page)
58 struct buffer_head **bh)
59{ 58{
60 int status; 59 struct inode *inode = page->mapping->host;
61 char *link = NULL; 60 struct buffer_head *bh;
61 int status = ocfs2_read_inode_block(inode, &bh);
62 struct ocfs2_dinode *fe; 62 struct ocfs2_dinode *fe;
63 const char *link;
64 void *kaddr;
65 size_t len;
63 66
64 status = ocfs2_read_inode_block(inode, bh);
65 if (status < 0) { 67 if (status < 0) {
66 mlog_errno(status); 68 mlog_errno(status);
67 link = ERR_PTR(status); 69 return status;
68 goto bail;
69 } 70 }
70 71
71 fe = (struct ocfs2_dinode *) (*bh)->b_data; 72 fe = (struct ocfs2_dinode *) bh->b_data;
72 link = (char *) fe->id2.i_symlink; 73 link = (char *) fe->id2.i_symlink;
73bail: 74 /* will be less than a page size */
74 75 len = strnlen(link, ocfs2_fast_symlink_chars(inode->i_sb));
75 return link; 76 kaddr = kmap_atomic(page);
76} 77 memcpy(kaddr, link, len + 1);
77 78 kunmap_atomic(kaddr);
78static int ocfs2_readlink(struct dentry *dentry, 79 SetPageUptodate(page);
79 char __user *buffer, 80 unlock_page(page);
80 int buflen)
81{
82 int ret;
83 char *link;
84 struct buffer_head *bh = NULL;
85 struct inode *inode = dentry->d_inode;
86
87 link = ocfs2_fast_symlink_getlink(inode, &bh);
88 if (IS_ERR(link)) {
89 ret = PTR_ERR(link);
90 goto out;
91 }
92
93 /*
94 * Without vfsmount we can't update atime now,
95 * but we will update atime here ultimately.
96 */
97 ret = vfs_readlink(dentry, buffer, buflen, link);
98
99 brelse(bh); 81 brelse(bh);
100out: 82 return 0;
101 if (ret < 0)
102 mlog_errno(ret);
103 return ret;
104} 83}
105 84
106static void *ocfs2_fast_follow_link(struct dentry *dentry, 85const struct address_space_operations ocfs2_fast_symlink_aops = {
107 struct nameidata *nd) 86 .readpage = ocfs2_fast_symlink_readpage,
108{ 87};
109 int status = 0;
110 int len;
111 char *target, *link = ERR_PTR(-ENOMEM);
112 struct inode *inode = dentry->d_inode;
113 struct buffer_head *bh = NULL;
114
115 BUG_ON(!ocfs2_inode_is_fast_symlink(inode));
116 target = ocfs2_fast_symlink_getlink(inode, &bh);
117 if (IS_ERR(target)) {
118 status = PTR_ERR(target);
119 mlog_errno(status);
120 goto bail;
121 }
122
123 /* Fast symlinks can't be large */
124 len = strnlen(target, ocfs2_fast_symlink_chars(inode->i_sb));
125 link = kzalloc(len + 1, GFP_NOFS);
126 if (!link) {
127 status = -ENOMEM;
128 mlog_errno(status);
129 goto bail;
130 }
131
132 memcpy(link, target, len);
133
134bail:
135 nd_set_link(nd, status ? ERR_PTR(status) : link);
136 brelse(bh);
137
138 if (status)
139 mlog_errno(status);
140 return NULL;
141}
142
143static void ocfs2_fast_put_link(struct dentry *dentry, struct nameidata *nd, void *cookie)
144{
145 char *link = nd_get_link(nd);
146 if (!IS_ERR(link))
147 kfree(link);
148}
149 88
150const struct inode_operations ocfs2_symlink_inode_operations = { 89const struct inode_operations ocfs2_symlink_inode_operations = {
151 .readlink = page_readlink, 90 .readlink = generic_readlink,
152 .follow_link = page_follow_link_light, 91 .follow_link = page_follow_link_light,
153 .put_link = page_put_link, 92 .put_link = page_put_link,
154 .getattr = ocfs2_getattr, 93 .getattr = ocfs2_getattr,
@@ -159,15 +98,3 @@ const struct inode_operations ocfs2_symlink_inode_operations = {
159 .removexattr = generic_removexattr, 98 .removexattr = generic_removexattr,
160 .fiemap = ocfs2_fiemap, 99 .fiemap = ocfs2_fiemap,
161}; 100};
162const struct inode_operations ocfs2_fast_symlink_inode_operations = {
163 .readlink = ocfs2_readlink,
164 .follow_link = ocfs2_fast_follow_link,
165 .put_link = ocfs2_fast_put_link,
166 .getattr = ocfs2_getattr,
167 .setattr = ocfs2_setattr,
168 .setxattr = generic_setxattr,
169 .getxattr = generic_getxattr,
170 .listxattr = ocfs2_listxattr,
171 .removexattr = generic_removexattr,
172 .fiemap = ocfs2_fiemap,
173};