blob: efc4fe69f4f010693c0ff54607d1639d398dd7cd [file] [log] [blame]
David Howellsec268152007-04-26 15:49:28 -07001/* AFS superblock handling
2 *
David Howells08e0e7c2007-04-26 15:55:03 -07003 * Copyright (c) 2002, 2007 Red Hat, Inc. All rights reserved.
Linus Torvalds1da177e2005-04-16 15:20:36 -07004 *
5 * This software may be freely redistributed under the terms of the
6 * GNU General Public License.
7 *
8 * You should have received a copy of the GNU General Public License
9 * along with this program; if not, write to the Free Software
10 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
11 *
12 * Authors: David Howells <dhowells@redhat.com>
David Howellsec268152007-04-26 15:49:28 -070013 * David Woodhouse <dwmw2@redhat.com>
Linus Torvalds1da177e2005-04-16 15:20:36 -070014 *
15 */
16
17#include <linux/kernel.h>
18#include <linux/module.h>
19#include <linux/init.h>
20#include <linux/slab.h>
21#include <linux/fs.h>
22#include <linux/pagemap.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070023#include "internal.h"
24
25#define AFS_FS_MAGIC 0x6B414653 /* 'kAFS' */
26
27struct afs_mount_params {
28 int rwpath;
29 struct afs_cell *default_cell;
30 struct afs_volume *volume;
31};
32
Christoph Lametere18b8902006-12-06 20:33:20 -080033static void afs_i_init_once(void *foo, struct kmem_cache *cachep,
Linus Torvalds1da177e2005-04-16 15:20:36 -070034 unsigned long flags);
35
David Howells454e2392006-06-23 02:02:57 -070036static int afs_get_sb(struct file_system_type *fs_type,
37 int flags, const char *dev_name,
38 void *data, struct vfsmount *mnt);
Linus Torvalds1da177e2005-04-16 15:20:36 -070039
40static struct inode *afs_alloc_inode(struct super_block *sb);
41
42static void afs_put_super(struct super_block *sb);
43
44static void afs_destroy_inode(struct inode *inode);
45
Trond Myklebust1f5ce9e2006-06-09 09:34:16 -040046struct file_system_type afs_fs_type = {
Linus Torvalds1da177e2005-04-16 15:20:36 -070047 .owner = THIS_MODULE,
48 .name = "afs",
49 .get_sb = afs_get_sb,
50 .kill_sb = kill_anon_super,
51 .fs_flags = FS_BINARY_MOUNTDATA,
52};
53
Josef 'Jeff' Sipekee9b6d62007-02-12 00:55:41 -080054static const struct super_operations afs_super_ops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -070055 .statfs = simple_statfs,
56 .alloc_inode = afs_alloc_inode,
57 .drop_inode = generic_delete_inode,
58 .destroy_inode = afs_destroy_inode,
59 .clear_inode = afs_clear_inode,
David Howells08e0e7c2007-04-26 15:55:03 -070060 .umount_begin = afs_umount_begin,
Linus Torvalds1da177e2005-04-16 15:20:36 -070061 .put_super = afs_put_super,
62};
63
Christoph Lametere18b8902006-12-06 20:33:20 -080064static struct kmem_cache *afs_inode_cachep;
Linus Torvalds1da177e2005-04-16 15:20:36 -070065static atomic_t afs_count_active_inodes;
66
Linus Torvalds1da177e2005-04-16 15:20:36 -070067/*
68 * initialise the filesystem
69 */
70int __init afs_fs_init(void)
71{
72 int ret;
73
74 _enter("");
75
Linus Torvalds1da177e2005-04-16 15:20:36 -070076 /* create ourselves an inode cache */
77 atomic_set(&afs_count_active_inodes, 0);
78
79 ret = -ENOMEM;
80 afs_inode_cachep = kmem_cache_create("afs_inode_cache",
81 sizeof(struct afs_vnode),
82 0,
83 SLAB_HWCACHE_ALIGN,
84 afs_i_init_once,
85 NULL);
86 if (!afs_inode_cachep) {
87 printk(KERN_NOTICE "kAFS: Failed to allocate inode cache\n");
88 return ret;
89 }
90
91 /* now export our filesystem to lesser mortals */
92 ret = register_filesystem(&afs_fs_type);
93 if (ret < 0) {
94 kmem_cache_destroy(afs_inode_cachep);
David Howells08e0e7c2007-04-26 15:55:03 -070095 _leave(" = %d", ret);
Linus Torvalds1da177e2005-04-16 15:20:36 -070096 return ret;
97 }
98
David Howells08e0e7c2007-04-26 15:55:03 -070099 _leave(" = 0");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700100 return 0;
David Howellsec268152007-04-26 15:49:28 -0700101}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700102
Linus Torvalds1da177e2005-04-16 15:20:36 -0700103/*
104 * clean up the filesystem
105 */
106void __exit afs_fs_exit(void)
107{
David Howells08e0e7c2007-04-26 15:55:03 -0700108 _enter("");
109
110 afs_mntpt_kill_timer();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700111 unregister_filesystem(&afs_fs_type);
112
113 if (atomic_read(&afs_count_active_inodes) != 0) {
114 printk("kAFS: %d active inode objects still present\n",
115 atomic_read(&afs_count_active_inodes));
116 BUG();
117 }
118
119 kmem_cache_destroy(afs_inode_cachep);
David Howells08e0e7c2007-04-26 15:55:03 -0700120 _leave("");
David Howellsec268152007-04-26 15:49:28 -0700121}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700122
Linus Torvalds1da177e2005-04-16 15:20:36 -0700123/*
124 * check that an argument has a value
125 */
126static int want_arg(char **_value, const char *option)
127{
128 if (!_value || !*_value || !**_value) {
129 printk(KERN_NOTICE "kAFS: %s: argument missing\n", option);
130 return 0;
131 }
132 return 1;
David Howellsec268152007-04-26 15:49:28 -0700133}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700134
Linus Torvalds1da177e2005-04-16 15:20:36 -0700135/*
136 * check that there's no subsequent value
137 */
138static int want_no_value(char *const *_value, const char *option)
139{
140 if (*_value && **_value) {
141 printk(KERN_NOTICE "kAFS: %s: Invalid argument: %s\n",
142 option, *_value);
143 return 0;
144 }
145 return 1;
David Howellsec268152007-04-26 15:49:28 -0700146}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700147
Linus Torvalds1da177e2005-04-16 15:20:36 -0700148/*
149 * parse the mount options
150 * - this function has been shamelessly adapted from the ext3 fs which
151 * shamelessly adapted it from the msdos fs
152 */
153static int afs_super_parse_options(struct afs_mount_params *params,
David Howells08e0e7c2007-04-26 15:55:03 -0700154 char *options, const char **devname)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700155{
David Howells08e0e7c2007-04-26 15:55:03 -0700156 struct afs_cell *cell;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700157 char *key, *value;
158 int ret;
159
160 _enter("%s", options);
161
162 options[PAGE_SIZE - 1] = 0;
163
164 ret = 0;
David Howells08e0e7c2007-04-26 15:55:03 -0700165 while ((key = strsep(&options, ","))) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700166 value = strchr(key, '=');
167 if (value)
168 *value++ = 0;
169
David Howells08e0e7c2007-04-26 15:55:03 -0700170 _debug("kAFS: KEY: %s, VAL:%s", key, value ?: "-");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700171
172 if (strcmp(key, "rwpath") == 0) {
173 if (!want_no_value(&value, "rwpath"))
174 return -EINVAL;
175 params->rwpath = 1;
David Howellsec268152007-04-26 15:49:28 -0700176 } else if (strcmp(key, "vol") == 0) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700177 if (!want_arg(&value, "vol"))
178 return -EINVAL;
179 *devname = value;
David Howellsec268152007-04-26 15:49:28 -0700180 } else if (strcmp(key, "cell") == 0) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700181 if (!want_arg(&value, "cell"))
182 return -EINVAL;
David Howells08e0e7c2007-04-26 15:55:03 -0700183 cell = afs_cell_lookup(value, strlen(value));
184 if (IS_ERR(cell))
185 return PTR_ERR(cell);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700186 afs_put_cell(params->default_cell);
David Howells08e0e7c2007-04-26 15:55:03 -0700187 params->default_cell = cell;
188 } else {
189 printk("kAFS: Unknown mount option: '%s'\n", key);
190 ret = -EINVAL;
191 goto error;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700192 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700193 }
194
195 ret = 0;
David Howellsec268152007-04-26 15:49:28 -0700196error:
Linus Torvalds1da177e2005-04-16 15:20:36 -0700197 _leave(" = %d", ret);
198 return ret;
David Howellsec268152007-04-26 15:49:28 -0700199}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700200
Linus Torvalds1da177e2005-04-16 15:20:36 -0700201/*
202 * check a superblock to see if it's the one we're looking for
203 */
204static int afs_test_super(struct super_block *sb, void *data)
205{
206 struct afs_mount_params *params = data;
207 struct afs_super_info *as = sb->s_fs_info;
208
209 return as->volume == params->volume;
David Howellsec268152007-04-26 15:49:28 -0700210}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700211
Linus Torvalds1da177e2005-04-16 15:20:36 -0700212/*
213 * fill in the superblock
214 */
215static int afs_fill_super(struct super_block *sb, void *data, int silent)
216{
217 struct afs_mount_params *params = data;
218 struct afs_super_info *as = NULL;
219 struct afs_fid fid;
220 struct dentry *root = NULL;
221 struct inode *inode = NULL;
222 int ret;
223
David Howells08e0e7c2007-04-26 15:55:03 -0700224 _enter("");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700225
226 /* allocate a superblock info record */
Yan Burmanb593e482006-12-06 20:40:32 -0800227 as = kzalloc(sizeof(struct afs_super_info), GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700228 if (!as) {
229 _leave(" = -ENOMEM");
230 return -ENOMEM;
231 }
232
Linus Torvalds1da177e2005-04-16 15:20:36 -0700233 afs_get_volume(params->volume);
234 as->volume = params->volume;
235
236 /* fill in the superblock */
237 sb->s_blocksize = PAGE_CACHE_SIZE;
238 sb->s_blocksize_bits = PAGE_CACHE_SHIFT;
239 sb->s_magic = AFS_FS_MAGIC;
240 sb->s_op = &afs_super_ops;
241 sb->s_fs_info = as;
242
243 /* allocate the root inode and dentry */
244 fid.vid = as->volume->vid;
245 fid.vnode = 1;
246 fid.unique = 1;
David Howells08e0e7c2007-04-26 15:55:03 -0700247 inode = afs_iget(sb, &fid);
248 if (IS_ERR(inode))
249 goto error_inode;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700250
251 ret = -ENOMEM;
252 root = d_alloc_root(inode);
253 if (!root)
254 goto error;
255
256 sb->s_root = root;
257
David Howells08e0e7c2007-04-26 15:55:03 -0700258 _leave(" = 0");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700259 return 0;
260
David Howells08e0e7c2007-04-26 15:55:03 -0700261error_inode:
262 ret = PTR_ERR(inode);
263 inode = NULL;
David Howellsec268152007-04-26 15:49:28 -0700264error:
Linus Torvalds1da177e2005-04-16 15:20:36 -0700265 iput(inode);
266 afs_put_volume(as->volume);
267 kfree(as);
268
269 sb->s_fs_info = NULL;
270
David Howells08e0e7c2007-04-26 15:55:03 -0700271 _leave(" = %d", ret);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700272 return ret;
David Howellsec268152007-04-26 15:49:28 -0700273}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700274
Linus Torvalds1da177e2005-04-16 15:20:36 -0700275/*
276 * get an AFS superblock
277 * - TODO: don't use get_sb_nodev(), but rather call sget() directly
278 */
David Howells454e2392006-06-23 02:02:57 -0700279static int afs_get_sb(struct file_system_type *fs_type,
280 int flags,
281 const char *dev_name,
282 void *options,
283 struct vfsmount *mnt)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700284{
285 struct afs_mount_params params;
286 struct super_block *sb;
David Howells08e0e7c2007-04-26 15:55:03 -0700287 struct afs_volume *vol;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700288 int ret;
289
290 _enter(",,%s,%p", dev_name, options);
291
292 memset(&params, 0, sizeof(params));
293
Linus Torvalds1da177e2005-04-16 15:20:36 -0700294 /* parse the options */
295 if (options) {
296 ret = afs_super_parse_options(&params, options, &dev_name);
297 if (ret < 0)
298 goto error;
299 if (!dev_name) {
300 printk("kAFS: no volume name specified\n");
301 ret = -EINVAL;
302 goto error;
303 }
304 }
305
306 /* parse the device name */
David Howells08e0e7c2007-04-26 15:55:03 -0700307 vol = afs_volume_lookup(dev_name, params.default_cell, params.rwpath);
308 if (IS_ERR(vol)) {
309 ret = PTR_ERR(vol);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700310 goto error;
David Howells08e0e7c2007-04-26 15:55:03 -0700311 }
312
313 params.volume = vol;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700314
315 /* allocate a deviceless superblock */
316 sb = sget(fs_type, afs_test_super, set_anon_super, &params);
David Howells08e0e7c2007-04-26 15:55:03 -0700317 if (IS_ERR(sb)) {
318 ret = PTR_ERR(sb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700319 goto error;
David Howells08e0e7c2007-04-26 15:55:03 -0700320 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700321
322 sb->s_flags = flags;
323
Theodore Ts'o9b04c992006-03-24 03:15:10 -0800324 ret = afs_fill_super(sb, &params, flags & MS_SILENT ? 1 : 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700325 if (ret < 0) {
326 up_write(&sb->s_umount);
327 deactivate_super(sb);
328 goto error;
329 }
330 sb->s_flags |= MS_ACTIVE;
David Howells454e2392006-06-23 02:02:57 -0700331 simple_set_mnt(mnt, sb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700332
333 afs_put_volume(params.volume);
334 afs_put_cell(params.default_cell);
David Howells08e0e7c2007-04-26 15:55:03 -0700335 _leave(" = 0 [%p]", sb);
David Howells454e2392006-06-23 02:02:57 -0700336 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700337
David Howellsec268152007-04-26 15:49:28 -0700338error:
Linus Torvalds1da177e2005-04-16 15:20:36 -0700339 afs_put_volume(params.volume);
340 afs_put_cell(params.default_cell);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700341 _leave(" = %d", ret);
David Howells454e2392006-06-23 02:02:57 -0700342 return ret;
David Howellsec268152007-04-26 15:49:28 -0700343}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700344
Linus Torvalds1da177e2005-04-16 15:20:36 -0700345/*
346 * finish the unmounting process on the superblock
347 */
348static void afs_put_super(struct super_block *sb)
349{
350 struct afs_super_info *as = sb->s_fs_info;
351
352 _enter("");
353
354 afs_put_volume(as->volume);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700355
356 _leave("");
David Howellsec268152007-04-26 15:49:28 -0700357}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700358
Linus Torvalds1da177e2005-04-16 15:20:36 -0700359/*
360 * initialise an inode cache slab element prior to any use
361 */
Christoph Lametere18b8902006-12-06 20:33:20 -0800362static void afs_i_init_once(void *_vnode, struct kmem_cache *cachep,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700363 unsigned long flags)
364{
David Howellsec268152007-04-26 15:49:28 -0700365 struct afs_vnode *vnode = _vnode;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700366
367 if ((flags & (SLAB_CTOR_VERIFY|SLAB_CTOR_CONSTRUCTOR)) ==
368 SLAB_CTOR_CONSTRUCTOR) {
369 memset(vnode, 0, sizeof(*vnode));
370 inode_init_once(&vnode->vfs_inode);
371 init_waitqueue_head(&vnode->update_waitq);
372 spin_lock_init(&vnode->lock);
David Howells08e0e7c2007-04-26 15:55:03 -0700373 INIT_WORK(&vnode->cb_broken_work, afs_broken_callback_work);
374 mutex_init(&vnode->cb_broken_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700375 }
David Howellsec268152007-04-26 15:49:28 -0700376}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700377
Linus Torvalds1da177e2005-04-16 15:20:36 -0700378/*
379 * allocate an AFS inode struct from our slab cache
380 */
381static struct inode *afs_alloc_inode(struct super_block *sb)
382{
383 struct afs_vnode *vnode;
384
David Howellsec268152007-04-26 15:49:28 -0700385 vnode = kmem_cache_alloc(afs_inode_cachep, GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700386 if (!vnode)
387 return NULL;
388
389 atomic_inc(&afs_count_active_inodes);
390
391 memset(&vnode->fid, 0, sizeof(vnode->fid));
392 memset(&vnode->status, 0, sizeof(vnode->status));
393
394 vnode->volume = NULL;
395 vnode->update_cnt = 0;
396 vnode->flags = 0;
David Howells08e0e7c2007-04-26 15:55:03 -0700397 vnode->cb_promised = false;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700398
399 return &vnode->vfs_inode;
David Howellsec268152007-04-26 15:49:28 -0700400}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700401
Linus Torvalds1da177e2005-04-16 15:20:36 -0700402/*
403 * destroy an AFS inode struct
404 */
405static void afs_destroy_inode(struct inode *inode)
406{
David Howells08e0e7c2007-04-26 15:55:03 -0700407 struct afs_vnode *vnode = AFS_FS_I(inode);
408
Linus Torvalds1da177e2005-04-16 15:20:36 -0700409 _enter("{%lu}", inode->i_ino);
410
David Howells08e0e7c2007-04-26 15:55:03 -0700411 _debug("DESTROY INODE %p", inode);
412
413 ASSERTCMP(vnode->server, ==, NULL);
414
415 kmem_cache_free(afs_inode_cachep, vnode);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700416 atomic_dec(&afs_count_active_inodes);
David Howellsec268152007-04-26 15:49:28 -0700417}