VirtualBox

source: vbox/trunk/src/VBox/Additions/linux/sharedfolders/vfsmod.c@ 10759

Last change on this file since 10759 was 8155, checked in by vboxsync, 17 years ago

The Big Sun Rebranding Header Change

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 13.2 KB
Line 
1/** @file
2 *
3 * vboxvfs -- VirtualBox Guest Additions for Linux:
4 * Virtual File System for VirtualBox Shared Folders
5 *
6 * Module initialization/finalization
7 * File system registration/deregistration
8 * Superblock reading
9 * Few utility functions
10 */
11
12/*
13 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
14 *
15 * This file is part of VirtualBox Open Source Edition (OSE), as
16 * available from http://www.virtualbox.org. This file is free software;
17 * you can redistribute it and/or modify it under the terms of the GNU
18 * General Public License (GPL) as published by the Free Software
19 * Foundation, in version 2 as it comes in the "COPYING" file of the
20 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
21 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
22 *
23 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
24 * Clara, CA 95054 USA or visit http://www.sun.com if you need
25 * additional information or have any questions.
26 */
27
28/**
29 * @note Anyone wishing to make changes here might wish to take a look at
30 * http://www.atnf.csiro.au/people/rgooch/linux/vfs.txt
31 * which seems to be the closest there is to official documentation on
32 * writing filesystem drivers for Linux.
33 */
34
35/*
36 * Suppress the definition of wchar_t from stddef.h that occurs below.
37 * This makes (at least) RHEL3U5 happy.
38 */
39#if 0
40#if LINUX_VERSION_CODE < KERNEL_VERSION(2, 5, 0)
41# define _WCHAR_T
42#endif
43#endif
44
45#include "vfsmod.h"
46
47// #define wchar_t linux_wchar_t
48
49#define xstr(s) str(s)
50#define str(s) #s
51
52MODULE_DESCRIPTION ("Host file system access VFS for VirtualBox");
53MODULE_AUTHOR ("Sun Microsystems, Inc.");
54MODULE_LICENSE ("GPL");
55#ifdef MODULE_VERSION
56MODULE_VERSION(VBOX_VERSION_STRING " (interface " xstr(VMMDEV_VERSION) ")");
57#endif
58
59/* globals */
60VBSFCLIENT client_handle;
61
62/* forward declarations */
63static struct super_operations sf_super_ops;
64
65// #include "utils.c"
66// #include "dirops.c"
67// #include "regops.c"
68
69/* allocate global info, try to map host share */
70static int
71sf_glob_alloc (struct vbsf_mount_info *info, struct sf_glob_info **sf_gp)
72{
73 int err, rc;
74 SHFLSTRING *str_name;
75 size_t name_len, str_len;
76 struct sf_glob_info *sf_g;
77
78 TRACE ();
79 sf_g = kmalloc (sizeof (*sf_g), GFP_KERNEL);
80 if (!sf_g) {
81 err = -ENOMEM;
82 LogRelFunc(("could not allocate memory for global info\n"));
83 goto fail0;
84 }
85
86 info->name[sizeof (info->name) - 1] = 0;
87 info->nls_name[sizeof (info->nls_name) - 1] = 0;
88
89 name_len = strlen (info->name);
90 if (name_len > 0xfffe) {
91 err = -ENAMETOOLONG;
92 LogFunc(("map name too big\n"));
93 goto fail1;
94 }
95
96 str_len = offsetof (SHFLSTRING, String.utf8) + name_len + 1;
97 str_name = kmalloc (str_len, GFP_KERNEL);
98 if (!str_name) {
99 err = -ENOMEM;
100 LogRelFunc(("could not allocate memory for host name\n"));
101 goto fail1;
102 }
103
104 str_name->u16Length = name_len;
105 str_name->u16Size = name_len + 1;
106 memcpy (str_name->String.utf8, info->name, name_len + 1);
107
108 if (info->nls_name[0] && strcmp (info->nls_name, "utf8")) {
109 sf_g->nls = load_nls (info->nls_name);
110 if (!sf_g->nls) {
111 err = -EINVAL;
112 LogFunc(("failed to load nls %s\n", info->nls_name));
113 goto fail1;
114 }
115 }
116 else {
117 sf_g->nls = NULL;
118 }
119
120 rc = vboxCallMapFolder (&client_handle, str_name, &sf_g->map);
121 kfree (str_name);
122
123 if (VBOX_FAILURE (rc)) {
124 err = -EPROTO;
125 LogFunc(("vboxCallMapFolder failed rc=%d\n", rc));
126 goto fail2;
127 }
128
129 sf_g->ttl = info->ttl;
130 sf_g->uid = info->uid;
131 sf_g->gid = info->gid;
132
133 *sf_gp = sf_g;
134 return 0;
135
136 fail2:
137 if (sf_g->nls) {
138 unload_nls (sf_g->nls);
139 }
140 fail1:
141 kfree (sf_g);
142 fail0:
143 return err;
144}
145
146/* unmap the share and free global info [sf_g] */
147static void
148sf_glob_free (struct sf_glob_info *sf_g)
149{
150 int rc;
151
152 TRACE ();
153 rc = vboxCallUnmapFolder (&client_handle, &sf_g->map);
154 if (VBOX_FAILURE (rc)) {
155 LogFunc(("vboxCallUnmapFolder failed rc=%d\n", rc));
156 }
157
158 if (sf_g->nls) {
159 unload_nls (sf_g->nls);
160 }
161 kfree (sf_g);
162}
163
164/* this is called (by sf_read_super_[24|26] when vfs mounts the fs and
165 wants to read super_block.
166
167 calls [sf_glob_alloc] to map the folder and allocate global
168 information structure.
169
170 initializes [sb], initializes root inode and dentry.
171
172 should respect [flags] */
173static int
174sf_read_super_aux (struct super_block *sb, void *data, int flags)
175{
176 int err;
177 struct dentry *droot;
178 struct inode *iroot;
179 struct sf_inode_info *sf_i;
180 struct sf_glob_info *sf_g;
181 RTFSOBJINFO fsinfo;
182 struct vbsf_mount_info *info;
183
184 TRACE ();
185 if (!data) {
186 LogFunc(("no mount info specified\n"));
187 return -EINVAL;
188 }
189
190 info = data;
191
192 if (flags & MS_REMOUNT) {
193 LogFunc(("remounting is not supported\n"));
194 return -ENOSYS;
195 }
196
197 err = sf_glob_alloc (info, &sf_g);
198 if (err) {
199 goto fail0;
200 }
201
202 sf_i = kmalloc (sizeof (*sf_i), GFP_KERNEL);
203 if (!sf_i) {
204 err = -ENOMEM;
205 LogRelFunc (("could not allocate memory for root inode info\n"));
206 goto fail1;
207 }
208
209 sf_i->path = kmalloc (sizeof (SHFLSTRING) + 1, GFP_KERNEL);
210 if (!sf_i->path) {
211 err = -ENOMEM;
212 LogRelFunc (("could not allocate memory for root inode path\n"));
213 goto fail2;
214 }
215
216 sf_i->path->u16Length = 1;
217 sf_i->path->u16Size = 2;
218 sf_i->path->String.utf8[0] = '/';
219 sf_i->path->String.utf8[1] = 0;
220
221 err = sf_stat (__func__, sf_g, sf_i->path, &fsinfo, 0);
222 if (err) {
223 LogFunc(("could not stat root of share\n"));
224 goto fail3;
225 }
226
227 sb->s_magic = 0xface;
228 sb->s_blocksize = 1024;
229 sb->s_op = &sf_super_ops;
230
231#if LINUX_VERSION_CODE >= KERNEL_VERSION (2, 4, 25)
232 iroot = iget_locked (sb, 0);
233#else
234 iroot = iget (sb, 0);
235#endif
236 if (!iroot) {
237 err = -ENOMEM; /* XXX */
238 LogFunc(("could not get root inode\n"));
239 goto fail3;
240 }
241
242 sf_init_inode (sf_g, iroot, &fsinfo);
243 SET_INODE_INFO (iroot, sf_i);
244
245#if LINUX_VERSION_CODE >= KERNEL_VERSION (2, 4, 25)
246 unlock_new_inode(iroot);
247#endif
248
249 droot = d_alloc_root (iroot);
250 if (!droot) {
251 err = -ENOMEM; /* XXX */
252 LogFunc(("d_alloc_root failed\n"));
253 goto fail4;
254 }
255
256 sb->s_root = droot;
257 SET_GLOB_INFO (sb, sf_g);
258 return 0;
259
260 fail4:
261 iput (iroot);
262 fail3:
263 kfree (sf_i->path);
264 fail2:
265 kfree (sf_i);
266 fail1:
267 sf_glob_free (sf_g);
268 fail0:
269 return err;
270}
271
272#if LINUX_VERSION_CODE < KERNEL_VERSION (2, 6, 0)
273static struct super_block *
274sf_read_super_24 (struct super_block *sb, void *data, int flags)
275{
276 int err;
277
278 TRACE ();
279 err = sf_read_super_aux (sb, data, flags);
280 if (err) {
281 return NULL;
282 }
283
284 return sb;
285}
286#endif
287
288/* this is called when vfs is about to destroy the [inode]. all
289 resources associated with this [inode] must be cleared here */
290static void
291sf_clear_inode (struct inode *inode)
292{
293 struct sf_inode_info *sf_i;
294
295 TRACE ();
296 sf_i = GET_INODE_INFO (inode);
297 if (!sf_i) {
298 return;
299 }
300
301 BUG_ON (!sf_i->path);
302 kfree (sf_i->path);
303 kfree (sf_i);
304 SET_INODE_INFO (inode, NULL);
305}
306
307/* this is called by vfs when it wants to populate [inode] with data.
308 the only thing that is known about inode at this point is its index
309 hence we can't do anything here, and let lookup/whatever with the
310 job to properly fill then [inode] */
311#if LINUX_VERSION_CODE < KERNEL_VERSION (2, 6, 25)
312static void
313sf_read_inode (struct inode *inode)
314{
315}
316#endif
317
318/* vfs is done with [sb] (umount called) call [sf_glob_free] to unmap
319 the folder and free [sf_g] */
320static void
321sf_put_super (struct super_block *sb)
322{
323 struct sf_glob_info *sf_g;
324
325 sf_g = GET_GLOB_INFO (sb);
326 BUG_ON (!sf_g);
327 sf_glob_free (sf_g);
328}
329
330#if LINUX_VERSION_CODE < KERNEL_VERSION (2, 6, 18)
331static int
332sf_statfs (struct super_block *sb, STRUCT_STATFS *stat)
333{
334 return sf_get_volume_info(sb, stat);
335}
336#else
337static int
338sf_statfs (struct dentry *dentry, STRUCT_STATFS *stat)
339{
340 struct super_block *sb = dentry->d_inode->i_sb;
341 return sf_get_volume_info(sb, stat);
342}
343#endif
344
345static int
346sf_remount_fs (struct super_block *sb, int *flags, char *data)
347{
348 TRACE ();
349 return -ENOSYS;
350}
351
352static struct super_operations sf_super_ops = {
353 .clear_inode = sf_clear_inode,
354#if LINUX_VERSION_CODE < KERNEL_VERSION (2, 6, 25)
355 .read_inode = sf_read_inode,
356#endif
357 .put_super = sf_put_super,
358 .statfs = sf_statfs,
359 .remount_fs = sf_remount_fs
360};
361
362#if LINUX_VERSION_CODE < KERNEL_VERSION (2, 6, 0)
363static DECLARE_FSTYPE (vboxsf_fs_type, "vboxsf", sf_read_super_24, 0);
364#else
365static int
366sf_read_super_26 (struct super_block *sb, void *data, int flags)
367{
368 int err;
369
370 TRACE ();
371 err = sf_read_super_aux (sb, data, flags);
372 if (err) {
373 printk (KERN_DEBUG "sf_read_super_aux err=%d\n", err);
374 }
375 return err;
376}
377
378#if LINUX_VERSION_CODE < KERNEL_VERSION (2, 6, 18)
379static struct super_block *
380sf_get_sb (struct file_system_type *fs_type, int flags,
381 const char *dev_name, void *data)
382{
383 TRACE ();
384 return get_sb_nodev (fs_type, flags, data, sf_read_super_26);
385}
386#else
387static int
388sf_get_sb (struct file_system_type *fs_type, int flags,
389 const char *dev_name, void *data, struct vfsmount *mnt)
390{
391 TRACE ();
392 return get_sb_nodev (fs_type, flags, data, sf_read_super_26, mnt);
393}
394#endif
395
396static struct file_system_type vboxsf_fs_type = {
397 .owner = THIS_MODULE,
398 .name = "vboxsf",
399 .get_sb = sf_get_sb,
400 .kill_sb = kill_anon_super
401};
402#endif
403
404extern int CMC_API
405vboxadd_cmc_ctl_guest_filter_mask (uint32_t or_mask, uint32_t not_mask);
406
407/* Module initialization/finalization handlers */
408static int __init
409init (void)
410{
411 int rcVBox;
412 int rcRet = 0;
413 int err;
414
415 TRACE ();
416
417 if (sizeof (struct vbsf_mount_info) > PAGE_SIZE) {
418 printk (KERN_ERR
419 "Mount information structure is too large %lu\n"
420 "Must be less than or equal to %lu\n",
421 (unsigned long)sizeof (struct vbsf_mount_info),
422 PAGE_SIZE);
423 return -EINVAL;
424 }
425
426 err = register_filesystem (&vboxsf_fs_type);
427 if (err) {
428 LogFunc(("register_filesystem err=%d\n", err));
429 return err;
430 }
431
432 if (vboxadd_cmc_ctl_guest_filter_mask (VMMDEV_EVENT_HGCM, 0)) {
433 rcRet = -EINVAL;
434 goto fail0;
435 }
436
437 rcVBox = vboxInit ();
438 if (VBOX_FAILURE (rcVBox)) {
439 LogRelFunc (("vboxInit failed, rc=%d\n", rcVBox));
440 rcRet = -EPROTO;
441 goto fail0;
442 }
443
444 rcVBox = vboxConnect (&client_handle);
445 if (VBOX_FAILURE (rcVBox)) {
446 LogRelFunc (("vboxConnect failed, rc=%d\n", rcVBox));
447 rcRet = -EPROTO;
448 goto fail1;
449 }
450
451 rcVBox = vboxCallSetUtf8 (&client_handle);
452 if (VBOX_FAILURE (rcVBox)) {
453 LogRelFunc (("vboxCallSetUtf8 failed, rc=%d\n", rcVBox));
454 rcRet = -EPROTO;
455 goto fail2;
456 }
457
458 printk(KERN_DEBUG
459 "vboxvfs: Successfully loaded version " VBOX_VERSION_STRING
460 " (interface " xstr(VMMDEV_VERSION) ")\n");
461
462 return 0;
463
464 fail2:
465 vboxDisconnect (&client_handle);
466 fail1:
467 vboxUninit ();
468 fail0:
469 vboxadd_cmc_ctl_guest_filter_mask (0, VMMDEV_EVENT_HGCM);
470 unregister_filesystem (&vboxsf_fs_type);
471 return rcRet;
472}
473
474static void __exit
475fini (void)
476{
477 TRACE ();
478
479 vboxDisconnect (&client_handle);
480 vboxUninit ();
481 vboxadd_cmc_ctl_guest_filter_mask (0, VMMDEV_EVENT_HGCM);
482 unregister_filesystem (&vboxsf_fs_type);
483}
484
485module_init (init);
486module_exit (fini);
487
488/* C++ hack */
489int __gxx_personality_v0 = 0xdeadbeef;
490
491#if 0
492/* long long hacks (as far as i can see, gcc emits the refs to those
493 symbols, notwithstanding the fact that those aren't referenced
494 anywhere in the module) */
495void __divdi3 (void)
496{
497 elog ("called from %p\n", __builtin_return_address (0));
498 BUG ();
499}
500
501void __moddi3 (void)
502{
503 elog ("called from %p\n", __builtin_return_address (0));
504 BUG ();
505}
506#endif /* 0 */
507
508/*
509 * Local Variables:
510 * c-mode: linux
511 * indent-tabs-mode: nil
512 * c-basic-offset: 8
513 * End:
514 */
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette