VirtualBox

source: vbox/trunk/src/VBox/Additions/linux/sharedfolders/dirops.c@ 4924

Last change on this file since 4924 was 4860, checked in by vboxsync, 18 years ago

Activated the full backdoor logger in release builds of the Linux Guest Additions kernel modules

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 18.1 KB
Line 
1/** @file
2 *
3 * vboxvfs -- VirtualBox Guest Additions for Linux:
4 * Directory inode and file operations
5 */
6
7/*
8 * Copyright (C) 2006-2007 innotek GmbH
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License as published by the Free Software Foundation,
14 * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
15 * distribution. VirtualBox OSE is distributed in the hope that it will
16 * be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18
19#include "vfsmod.h"
20
21static int
22sf_dir_open (struct inode *inode, struct file *file)
23{
24 int rc;
25 int err;
26 struct sf_glob_info *sf_g = GET_GLOB_INFO (inode->i_sb);
27 struct sf_dir_info *sf_d;
28 struct sf_inode_info *sf_i = GET_INODE_INFO (inode);
29 SHFLCREATEPARMS params;
30
31 TRACE ();
32 BUG_ON (!sf_g);
33 BUG_ON (!sf_i);
34
35 if (file->private_data) {
36 LogFunc(("dir_open called on already opened directory %s\n",
37 sf_i->path->String.utf8));
38 return 0;
39 }
40
41 sf_d = sf_dir_info_alloc ();
42
43 if (!sf_d) {
44 LogRelFunc(("could not allocate directory info for %s\n",
45 sf_i->path->String.utf8));
46 return -ENOMEM;
47 }
48
49 params.CreateFlags = 0
50 | SHFL_CF_DIRECTORY
51 | SHFL_CF_ACT_OPEN_IF_EXISTS
52 | SHFL_CF_ACCESS_READ
53 ;
54
55 LogFunc(("sf_dir_open: calling vboxCallCreate, folder %s, flags %#x\n",
56 sf_i->path->String.utf8, params.CreateFlags));
57 rc = vboxCallCreate (&client_handle, &sf_g->map, sf_i->path, &params);
58 if (VBOX_FAILURE (rc)) {
59 LogFunc(("vboxCallCreate(%s) failed rc=%Vrc\n",
60 sf_i->path->String.utf8, rc));
61 sf_dir_info_free (sf_d);
62 return -EPERM;
63 }
64
65 if (params.Result != SHFL_FILE_EXISTS) {
66 LogFunc(("directory %s does not exist\n", sf_i->path->String.utf8));
67 sf_dir_info_free (sf_d);
68 return -ENOENT;
69 }
70
71 err = sf_dir_read_all (sf_g, sf_i, sf_d, params.Handle);
72 if (err) {
73 rc = vboxCallClose (&client_handle, &sf_g->map, params.Handle);
74 if (VBOX_FAILURE (rc)) {
75 LogFunc(("vboxCallClose(%s) after err=%d failed rc=%Vrc\n",
76 sf_i->path->String.utf8, err, rc));
77 }
78 sf_dir_info_free (sf_d);
79 return err;
80 }
81
82
83 rc = vboxCallClose (&client_handle, &sf_g->map, params.Handle);
84 if (VBOX_FAILURE (rc)) {
85 LogFunc(("vboxCallClose(%s) failed rc=%Vrc\n",
86 sf_i->path->String.utf8, rc));
87 }
88
89 file->private_data = sf_d;
90 return 0;
91}
92
93/* This is called when reference count of [file] goes to zero. Notify
94 the host that it can free whatever is associated with this
95 directory and deallocate our own internal buffers */
96static int
97sf_dir_release (struct inode *inode, struct file *file)
98{
99 struct sf_inode_info *sf_i = GET_INODE_INFO (inode);
100
101 TRACE ();
102 BUG_ON (!sf_i);
103
104 if (file->private_data) {
105 sf_dir_info_free (file->private_data);
106 }
107 return 0;
108}
109
110/* Extract element ([dir]->f_pos) from the directory [dir] into
111 [d_name], return:
112 0 all ok
113 1 end reached
114 -errno some form of error*/
115int
116sf_getdent (struct file *dir, char d_name[NAME_MAX])
117{
118 loff_t cur;
119 struct sf_glob_info *sf_g;
120 struct sf_dir_info *sf_d;
121 struct list_head *pos, *list;
122
123 TRACE ();
124 sf_g = GET_GLOB_INFO (dir->f_dentry->d_inode->i_sb);
125 sf_d = dir->private_data;
126
127 BUG_ON (!sf_g);
128 BUG_ON (!sf_d);
129
130 cur = 0;
131 list = &sf_d->info_list;
132 list_for_each (pos, list) {
133 struct sf_dir_buf *b;
134 SHFLDIRINFO *info;
135 loff_t i;
136 size_t name_len;
137 char *name_ptr;
138
139 b = list_entry (pos, struct sf_dir_buf, head);
140 if (dir->f_pos >= cur + b->nb_entries) {
141 cur += b->nb_entries;
142 continue;
143 }
144
145 for (i = 0, info = b->buf; i < dir->f_pos - cur; ++i) {
146 size_t size;
147
148 size = offsetof (SHFLDIRINFO, name.String) + info->name.u16Size;
149 info = (SHFLDIRINFO *) ((uintptr_t) info + size);
150 }
151
152 name_ptr = info->name.String.utf8;
153 name_len = info->name.u16Length;
154
155 return sf_nlscpy (sf_g, d_name, NAME_MAX, name_ptr, name_len);
156 }
157 return 1;
158}
159
160/* This is called when vfs wants to populate internal buffers with
161 directory [dir]s contents. [opaque] is an argument to the
162 [filldir]. [filldir] magically modifies it's argument - [opaque]
163 and takes following additional arguments (which i in turn get from
164 the host via sf_getdent):
165
166 name : name of the entry (i must also supply it's length huh?)
167 type : type of the entry (FILE | DIR | etc) (i ellect to use DT_UNKNOWN)
168 pos : position/index of the entry
169 ino : inode number of the entry (i fake those)
170
171 [dir] contains:
172 f_pos : cursor into the directory listing
173 private_data : mean of communcation with the host side
174
175 Extract elements from the directory listing (incrementing f_pos
176 along the way) and feed them to [filldir] until:
177
178 a. there are no more entries (i.e. sf_getdent set done to 1)
179 b. failure to compute fake inode number
180 c. filldir returns an error (see comment on that) */
181static int
182sf_dir_read (struct file *dir, void *opaque, filldir_t filldir)
183{
184 TRACE ();
185 for (;;) {
186 int err;
187 ino_t fake_ino;
188 loff_t sanity;
189 char d_name[NAME_MAX];
190
191 err = sf_getdent (dir, d_name);
192 switch (err) {
193 case 1:
194 return 0;
195
196 case 0:
197 break;
198
199 case -1:
200 default:
201 /* skip erroneous entry and proceed */
202 LogFunc(("sf_getdent error %d\n", err));
203 dir->f_pos += 1;
204 continue;
205 }
206
207 /* d_name now contains valid entry name */
208
209 sanity = dir->f_pos + 0xbeef;
210 fake_ino = sanity;
211 if (sanity - fake_ino) {
212 LogRelFunc(("can not compute ino\n"));
213 return -EINVAL;
214 }
215
216 err = filldir (opaque, d_name, strlen (d_name),
217 dir->f_pos, fake_ino, DT_UNKNOWN);
218 if (err) {
219 LogFunc(("filldir returned error %d\n", err));
220 /* Rely on the fact that filldir returns error
221 only when it runs out of space in opaque */
222 return 0;
223 }
224
225 dir->f_pos += 1;
226 }
227
228 BUG ();
229}
230
231struct file_operations sf_dir_fops = {
232 .open = sf_dir_open,
233 .readdir = sf_dir_read,
234 .release = sf_dir_release,
235 .read = generic_read_dir
236};
237
238
239/* iops */
240
241/* This is called when vfs failed to locate dentry in the cache. The
242 job of this function is to allocate inode and link it to dentry.
243 [dentry] contains the name to be looked in the [parent] directory.
244 Failure to locate the name is not a "hard" error, in this case NULL
245 inode is added to [dentry] and vfs should proceed trying to create
246 the entry via other means. NULL(or "positive" pointer) ought to be
247 returned in case of succes and "negative" pointer on error */
248static struct dentry *
249sf_lookup (struct inode *parent, struct dentry *dentry
250#if LINUX_VERSION_CODE >= KERNEL_VERSION (2, 6, 0)
251 , struct nameidata *nd
252#endif
253 )
254{
255 int err;
256 struct sf_inode_info *sf_i, *sf_new_i;
257 struct sf_glob_info *sf_g;
258 SHFLSTRING *path;
259 struct inode *inode;
260 ino_t ino;
261 RTFSOBJINFO fsinfo;
262
263 TRACE ();
264 sf_g = GET_GLOB_INFO (parent->i_sb);
265 sf_i = GET_INODE_INFO (parent);
266
267 BUG_ON (!sf_g);
268 BUG_ON (!sf_i);
269
270 err = sf_path_from_dentry (__func__, sf_g, sf_i, dentry, &path);
271 if (err) {
272 goto fail0;
273 }
274
275 err = sf_stat (__func__, sf_g, path, &fsinfo, 1);
276 if (err) {
277 if (err == -ENOENT) {
278 /* -ENOENT add NULL inode to dentry so it later can be
279 created via call to create/mkdir/open */
280 inode = NULL;
281 }
282 else goto fail1;
283 }
284 else {
285 sf_new_i = kmalloc (sizeof (*sf_new_i), GFP_KERNEL);
286 if (!sf_new_i) {
287 LogRelFunc(("could not allocate memory for new inode info\n"));
288 err = -ENOMEM;
289 goto fail1;
290 }
291
292 ino = iunique (parent->i_sb, 1);
293 inode = iget (parent->i_sb, ino);
294 if (!inode) {
295 LogFunc(("iget failed\n"));
296 err = -ENOMEM; /* XXX: ??? */
297 goto fail2;
298 }
299
300 SET_INODE_INFO (inode, sf_new_i);
301 sf_init_inode (sf_g, inode, &fsinfo);
302 sf_new_i->path = path;
303 }
304
305 sf_i->force_restat = 0;
306 dentry->d_time = jiffies;
307 dentry->d_op = &sf_dentry_ops;
308 d_add (dentry, inode);
309 return NULL;
310
311 fail2:
312 kfree (sf_new_i);
313 fail1:
314 kfree (path);
315 fail0:
316 return ERR_PTR (err);
317}
318
319/* This should allocate memory for sf_inode_info, compute unique inode
320 number, get an inode from vfs, initialize inode info, instantiate
321 dentry */
322static int
323sf_instantiate (const char *caller, struct inode *parent,
324 struct dentry *dentry, SHFLSTRING *path,
325 RTFSOBJINFO *info)
326{
327 int err;
328 ino_t ino;
329 struct inode *inode;
330 struct sf_inode_info *sf_new_i;
331 struct sf_glob_info *sf_g = GET_GLOB_INFO (parent->i_sb);
332
333 TRACE ();
334 BUG_ON (!sf_g);
335
336 sf_new_i = kmalloc (sizeof (*sf_new_i), GFP_KERNEL);
337 if (!sf_new_i) {
338 LogRelFunc(("could not allocate inode info. caller=%s\n", caller));
339 err = -ENOMEM;
340 goto fail0;
341 }
342
343 ino = iunique (parent->i_sb, 1);
344 inode = iget (parent->i_sb, ino);
345 if (!inode) {
346 LogFunc(("iget failed. caller=%s\n", caller));
347 err = -ENOMEM;
348 goto fail1;
349 }
350
351 sf_init_inode (sf_g, inode, info);
352 sf_new_i->path = path;
353 SET_INODE_INFO (inode, sf_new_i);
354
355 dentry->d_time = jiffies;
356 dentry->d_op = &sf_dentry_ops;
357 sf_new_i->force_restat = 1;
358
359 d_instantiate (dentry, inode);
360 return 0;
361
362 fail1:
363 kfree (sf_new_i);
364 fail0:
365 return err;
366
367}
368
369static int
370sf_create_aux (struct inode *parent, struct dentry *dentry, int dirop)
371{
372 int rc, err;
373 SHFLCREATEPARMS params;
374 SHFLSTRING *path;
375 struct sf_inode_info *sf_i = GET_INODE_INFO (parent);
376 struct sf_glob_info *sf_g = GET_GLOB_INFO (parent->i_sb);
377
378 TRACE ();
379 BUG_ON (!sf_i);
380 BUG_ON (!sf_g);
381
382#if 0
383 printk ("create_aux %s/%s\n", sf_i->path->String.utf8,
384 dentry->d_name.name);
385#endif
386
387 params.CreateFlags = 0
388 | SHFL_CF_ACT_CREATE_IF_NEW
389 | SHFL_CF_ACT_OVERWRITE_IF_EXISTS
390 | SHFL_CF_ACCESS_READWRITE
391 | (dirop ? SHFL_CF_DIRECTORY : 0)
392 ;
393
394 params.Info.Attr.fMode = 0
395 | (dirop ? RTFS_TYPE_DIRECTORY : RTFS_TYPE_FILE)
396 | RTFS_UNIX_IRUSR
397 | RTFS_UNIX_IWUSR
398 | RTFS_UNIX_IXUSR
399 ;
400
401 params.Info.Attr.enmAdditional = RTFSOBJATTRADD_NOTHING;
402
403 err = sf_path_from_dentry (__func__, sf_g, sf_i, dentry, &path);
404 if (err) {
405 goto fail0;
406 }
407
408 LogFunc(("calling vboxCallCreate, folder %s, flags %#x\n",
409 path->String.utf8, params.CreateFlags));
410 rc = vboxCallCreate (&client_handle, &sf_g->map, path, &params);
411 if (VBOX_FAILURE (rc)) {
412 err = -EPROTO;
413 LogFunc(("(%d): vboxCallCreate(%s) failed rc=%Vrc\n", dirop,
414 sf_i->path->String.utf8, rc));
415 goto fail0;
416 }
417
418 if (params.Result != SHFL_FILE_CREATED) {
419 err = -EPERM;
420 LogFunc(("(%d): could not create file %s result=%d\n", dirop,
421 sf_i->path->String.utf8, params.Result));
422 goto fail0;
423 }
424
425 err = sf_instantiate (__func__, parent, dentry, path, &params.Info);
426 if (err) {
427 LogFunc(("(%d): could not instantiate dentry for %s err=%d\n",
428 dirop, sf_i->path->String.utf8, err));
429 goto fail1;
430 }
431
432 rc = vboxCallClose (&client_handle, &sf_g->map, params.Handle);
433 if (VBOX_FAILURE (rc)) {
434 LogFunc(("(%d): vboxCallClose failed rc=%Vrc\n", dirop, rc));
435 }
436
437 sf_i->force_restat = 1;
438 return 0;
439
440 fail1:
441 rc = vboxCallClose (&client_handle, &sf_g->map, params.Handle);
442 if (VBOX_FAILURE (rc)) {
443 LogFunc(("(%d): vboxCallClose failed rc=%Vrc\n", dirop, rc));
444 }
445
446 fail0:
447 kfree (path);
448 return err;
449}
450
451static int
452sf_create (struct inode *parent, struct dentry *dentry, int mode
453#if LINUX_VERSION_CODE >= KERNEL_VERSION (2, 6, 0)
454 , struct nameidata *nd
455#endif
456 )
457{
458 TRACE ();
459 return sf_create_aux (parent, dentry, 0);
460}
461
462static int
463sf_mkdir (struct inode *parent, struct dentry *dentry, int mode)
464{
465 TRACE ();
466 return sf_create_aux (parent, dentry, 1);
467}
468
469static int
470sf_unlink_aux (struct inode *parent, struct dentry *dentry, int dirop)
471{
472 int rc, err;
473 struct sf_glob_info *sf_g = GET_GLOB_INFO (parent->i_sb);
474 struct sf_inode_info *sf_i = GET_INODE_INFO (parent);
475 SHFLSTRING *path;
476
477 TRACE ();
478 BUG_ON (!sf_g);
479
480 err = sf_path_from_dentry (__func__, sf_g, sf_i, dentry, &path);
481 if (err) {
482 goto fail0;
483 }
484
485 rc = vboxCallRemove (&client_handle, &sf_g->map, path,
486 dirop ? SHFL_REMOVE_DIR : SHFL_REMOVE_FILE);
487 if (VBOX_FAILURE (rc)) {
488 LogFunc(("(%d): vboxCallRemove(%s) failed rc=%Vrc\n", dirop,
489 path->String.utf8, rc));
490 err = -RTErrConvertToErrno (rc);
491 goto fail1;
492 }
493
494 kfree (path);
495 return 0;
496
497 fail1:
498 kfree (path);
499 fail0:
500 return err;
501}
502
503static int
504sf_unlink (struct inode *parent, struct dentry *dentry)
505{
506 TRACE ();
507 return sf_unlink_aux (parent, dentry, 0);
508}
509
510static int
511sf_rmdir (struct inode *parent, struct dentry *dentry)
512{
513 TRACE ();
514 return sf_unlink_aux (parent, dentry, 1);
515}
516
517static int
518sf_rename (struct inode *old_parent, struct dentry *old_dentry,
519 struct inode *new_parent, struct dentry *new_dentry)
520{
521 int err = 0, rc = VINF_SUCCESS;
522 struct sf_glob_info *sf_g = GET_GLOB_INFO (old_parent->i_sb);
523
524 TRACE ();
525
526 if (sf_g != GET_GLOB_INFO (new_parent->i_sb)) {
527 LogFunc(("rename with different roots\n"));
528 err = -EINVAL;
529 } else {
530 struct sf_inode_info *sf_old_i = GET_INODE_INFO (old_parent);
531 struct sf_inode_info *sf_new_i = GET_INODE_INFO (new_parent);
532 /* As we save the relative path inside the inode structure, we need to change
533 this if the rename is successful. */
534 struct sf_inode_info *sf_file_i = GET_INODE_INFO (old_dentry->d_inode);
535 SHFLSTRING *old_path;
536 SHFLSTRING *new_path;
537
538 BUG_ON (!sf_old_i);
539 BUG_ON (!sf_new_i);
540 BUG_ON (!sf_file_i);
541
542 old_path = sf_file_i->path;
543 err = sf_path_from_dentry (__func__, sf_g, sf_new_i,
544 new_dentry, &new_path);
545 if (err) {
546 LogFunc(("failed to create new path\n"));
547 } else {
548 int is_dir = ((old_dentry->d_inode->i_mode & S_IFDIR) != 0);
549
550 rc = vboxCallRename (&client_handle, &sf_g->map, old_path,
551 new_path, is_dir ? 0 : SHFL_RENAME_FILE | SHFL_RENAME_REPLACE_IF_EXISTS);
552 if (RT_SUCCESS(rc)) {
553 kfree (old_path);
554 sf_new_i->force_restat = 1;
555 sf_old_i->force_restat = 1; /* XXX: needed? */
556 /* Set the new relative path in the inode. */
557 sf_file_i->path = new_path;
558 } else {
559 LogFunc(("vboxCallRename failed rc=%Vrc\n", rc));
560 err = -RTErrConvertToErrno (err);
561 }
562 if (0 != err) {
563 kfree (new_path);
564 }
565 }
566 }
567 return err;
568}
569
570struct inode_operations sf_dir_iops = {
571 .lookup = sf_lookup,
572 .create = sf_create,
573 .mkdir = sf_mkdir,
574 .rmdir = sf_rmdir,
575 .unlink = sf_unlink,
576 .rename = sf_rename,
577#if LINUX_VERSION_CODE < KERNEL_VERSION (2, 6, 0)
578 .revalidate = sf_inode_revalidate
579#else
580 .getattr = sf_getattr
581#endif
582};
Note: See TracBrowser for help on using the repository browser.

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