VirtualBox

source: vbox/trunk/src/VBox/Additions/linux/sharedfolders/regops.c@ 20707

Last change on this file since 20707 was 20707, checked in by vboxsync, 16 years ago

Linux shared folders: implemented some missing functions, xTracker 3764

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 18.0 KB
Line 
1/** @file
2 *
3 * vboxvfs -- VirtualBox Guest Additions for Linux:
4 * Regular file inode and file operations
5 */
6
7/*
8 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
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 (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 *
18 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
19 * Clara, CA 95054 USA or visit http://www.sun.com if you need
20 * additional information or have any questions.
21 */
22
23/*
24 * Limitations: only COW memory mapping is supported
25 */
26
27#include "vfsmod.h"
28
29#define CHUNK_SIZE 4096
30
31/* fops */
32static int
33sf_reg_read_aux (const char *caller, struct sf_glob_info *sf_g,
34 struct sf_reg_info *sf_r, void *buf, uint32_t *nread,
35 uint64_t pos)
36{
37 int rc = vboxCallRead (&client_handle, &sf_g->map, sf_r->handle,
38 pos, nread, buf, false /* already locked? */);
39 if (RT_FAILURE (rc)) {
40 LogFunc(("vboxCallRead(%s) failed. caller=%s, rc=%Rrc\n",
41 sf_i->path->String.utf8, caller, rc));
42 return -EPROTO;
43 }
44 return 0;
45}
46
47static int
48sf_reg_write_aux (const char *caller, struct sf_glob_info *sf_g,
49 struct sf_reg_info *sf_r, void *buf, uint32_t *nwritten,
50 uint64_t pos)
51{
52 int rc = vboxCallWrite (&client_handle, &sf_g->map, sf_r->handle,
53 pos, nwritten, buf, false /* already locked? */);
54 if (RT_FAILURE (rc)) {
55 LogFunc(("vboxCallWrite(%s) failed rc=%Rrc\n",
56 sf_i->path->String.utf8, caller, rc));
57 return -EPROTO;
58 }
59 return 0;
60}
61
62static ssize_t
63sf_reg_read (struct file *file, char *buf, size_t size, loff_t *off)
64{
65 int err;
66 void *tmp;
67 size_t left = size;
68 ssize_t total_bytes_read = 0;
69 struct inode *inode = file->f_dentry->d_inode;
70 struct sf_glob_info *sf_g = GET_GLOB_INFO (inode->i_sb);
71 struct sf_reg_info *sf_r = file->private_data;
72 loff_t pos = *off;
73
74 TRACE ();
75 if (!S_ISREG (inode->i_mode)) {
76 LogFunc(("read from non regular file %d\n", inode->i_mode));
77 return -EINVAL;
78 }
79
80 /** XXX Check read permission accoring to inode->i_mode! */
81
82 if (!size) {
83 return 0;
84 }
85
86 tmp = kmalloc (CHUNK_SIZE, GFP_KERNEL);
87 if (!tmp) {
88 LogRelFunc(("could not allocate bounce buffer memory %d bytes\n", CHUNK_SIZE));
89 return -ENOMEM;
90 }
91
92 while (left) {
93 uint32_t to_read, nread;
94
95 to_read = CHUNK_SIZE;
96 if (to_read > left) {
97 to_read = (uint32_t) left;
98 }
99 nread = to_read;
100
101 err = sf_reg_read_aux (__func__, sf_g, sf_r, tmp, &nread, pos);
102 if (err)
103 goto fail;
104
105 if (copy_to_user (buf, tmp, nread)) {
106 err = -EFAULT;
107 goto fail;
108 }
109
110 pos += nread;
111 left -= nread;
112 buf += nread;
113 total_bytes_read += nread;
114 if (nread != to_read) {
115 break;
116 }
117 }
118
119 *off += total_bytes_read;
120 kfree (tmp);
121 return total_bytes_read;
122
123 fail:
124 kfree (tmp);
125 return err;
126}
127
128static ssize_t
129sf_reg_write (struct file *file, const char *buf, size_t size, loff_t *off)
130{
131 int err;
132 void *tmp;
133 size_t left = size;
134 ssize_t total_bytes_written = 0;
135 struct inode *inode = file->f_dentry->d_inode;
136 struct sf_inode_info *sf_i = GET_INODE_INFO (inode);
137 struct sf_glob_info *sf_g = GET_GLOB_INFO (inode->i_sb);
138 struct sf_reg_info *sf_r = file->private_data;
139 loff_t pos;
140
141 TRACE ();
142 BUG_ON (!sf_i);
143 BUG_ON (!sf_g);
144 BUG_ON (!sf_r);
145
146 if (!S_ISREG (inode->i_mode)) {
147 LogFunc(("write to non regular file %d\n", inode->i_mode));
148 return -EINVAL;
149 }
150
151 pos = *off;
152 if (file->f_flags & O_APPEND)
153 pos += inode->i_size;
154
155 /** XXX Check write permission accoring to inode->i_mode! */
156
157 if (!size)
158 return 0;
159
160 tmp = kmalloc (CHUNK_SIZE, GFP_KERNEL);
161 if (!tmp) {
162 LogRelFunc(("could not allocate bounce buffer memory %d bytes\n", CHUNK_SIZE));
163 return -ENOMEM;
164 }
165
166 while (left) {
167 uint32_t to_write, nwritten;
168
169 to_write = CHUNK_SIZE;
170 if (to_write > left) {
171 to_write = (uint32_t) left;
172 }
173 nwritten = to_write;
174
175 if (copy_from_user (tmp, buf, to_write)) {
176 err = -EFAULT;
177 goto fail;
178 }
179
180 err = sf_reg_write_aux (__func__, sf_g, sf_r, tmp, &nwritten, pos);
181 if (err)
182 goto fail;
183
184 pos += nwritten;
185 left -= nwritten;
186 buf += nwritten;
187 total_bytes_written += nwritten;
188 if (nwritten != to_write)
189 break;
190 }
191
192#if 1 /* XXX: which way is correct? */
193 *off += total_bytes_written;
194#else
195 file->f_pos += total_bytes_written;
196#endif
197 sf_i->force_restat = 1;
198 kfree (tmp);
199 return total_bytes_written;
200
201 fail:
202 kfree (tmp);
203 return err;
204}
205
206static int
207sf_reg_open (struct inode *inode, struct file *file)
208{
209 int rc, rc_linux = 0;
210 struct sf_glob_info *sf_g = GET_GLOB_INFO (inode->i_sb);
211 struct sf_inode_info *sf_i = GET_INODE_INFO (inode);
212 struct sf_reg_info *sf_r;
213 SHFLCREATEPARMS params;
214
215 TRACE ();
216 BUG_ON (!sf_g);
217 BUG_ON (!sf_i);
218
219 LogFunc(("open %s\n", sf_i->path->String.utf8));
220
221 sf_r = kmalloc (sizeof (*sf_r), GFP_KERNEL);
222 if (!sf_r) {
223 LogRelFunc(("could not allocate reg info\n"));
224 return -ENOMEM;
225 }
226
227 memset(&params, 0, sizeof(params));
228 params.Handle = SHFL_HANDLE_NIL;
229 /* We check the value of params.Handle afterwards to find out if
230 * the call succeeded or failed, as the API does not seem to cleanly
231 * distinguish error and informational messages.
232 *
233 * Furthermore, we must set params.Handle to SHFL_HANDLE_NIL to
234 * make the shared folders host service use our fMode parameter */
235
236 if (file->f_flags & O_CREAT) {
237 LogFunc(("O_CREAT set\n"));
238 params.CreateFlags |= SHFL_CF_ACT_CREATE_IF_NEW;
239 /* We ignore O_EXCL, as the Linux kernel seems to call create
240 beforehand itself, so O_EXCL should always fail. */
241 if (file->f_flags & O_TRUNC) {
242 LogFunc(("O_TRUNC set\n"));
243 params.CreateFlags |= ( SHFL_CF_ACT_OVERWRITE_IF_EXISTS
244 | SHFL_CF_ACCESS_WRITE);
245 }
246 else {
247 params.CreateFlags |= SHFL_CF_ACT_OPEN_IF_EXISTS;
248 }
249 }
250 else {
251 params.CreateFlags |= SHFL_CF_ACT_FAIL_IF_NEW;
252 if (file->f_flags & O_TRUNC) {
253 LogFunc(("O_TRUNC set\n"));
254 params.CreateFlags |= ( SHFL_CF_ACT_OVERWRITE_IF_EXISTS
255 | SHFL_CF_ACCESS_WRITE);
256 }
257 }
258
259 if (!(params.CreateFlags & SHFL_CF_ACCESS_READWRITE)) {
260 switch (file->f_flags & O_ACCMODE) {
261 case O_RDONLY:
262 params.CreateFlags |= SHFL_CF_ACCESS_READ;
263 break;
264
265 case O_WRONLY:
266 params.CreateFlags |= SHFL_CF_ACCESS_WRITE;
267 break;
268
269 case O_RDWR:
270 params.CreateFlags |= SHFL_CF_ACCESS_READWRITE;
271 break;
272
273 default:
274 BUG ();
275 }
276 }
277
278 params.Info.Attr.fMode = inode->i_mode;
279 LogFunc(("sf_reg_open: calling vboxCallCreate, file %s, flags=%d, %#x\n",
280 sf_i->path->String.utf8 , file->f_flags, params.CreateFlags));
281 rc = vboxCallCreate (&client_handle, &sf_g->map, sf_i->path, &params);
282
283 if (RT_FAILURE (rc)) {
284 LogFunc(("vboxCallCreate failed flags=%d,%#x rc=%Rrc\n",
285 file->f_flags, params.CreateFlags, rc));
286 kfree (sf_r);
287 return -RTErrConvertToErrno(rc);
288 }
289
290 if (SHFL_HANDLE_NIL == params.Handle) {
291 switch (params.Result) {
292 case SHFL_PATH_NOT_FOUND:
293 case SHFL_FILE_NOT_FOUND:
294 rc_linux = -ENOENT;
295 break;
296 case SHFL_FILE_EXISTS:
297 rc_linux = -EEXIST;
298 break;
299 default:
300 break;
301 }
302 }
303
304 sf_i->force_restat = 1;
305 sf_r->handle = params.Handle;
306 sf_i->file = file;
307 file->private_data = sf_r;
308 return rc_linux;
309}
310
311static int
312sf_reg_release (struct inode *inode, struct file *file)
313{
314 int rc;
315 struct sf_reg_info *sf_r;
316 struct sf_glob_info *sf_g;
317 struct sf_inode_info *sf_i = GET_INODE_INFO (inode);
318
319 TRACE ();
320 sf_g = GET_GLOB_INFO (inode->i_sb);
321 sf_r = file->private_data;
322
323 BUG_ON (!sf_g);
324 BUG_ON (!sf_r);
325
326 rc = vboxCallClose (&client_handle, &sf_g->map, sf_r->handle);
327 if (RT_FAILURE (rc)) {
328 LogFunc(("vboxCallClose failed rc=%Rrc\n", rc));
329 }
330
331 kfree (sf_r);
332 sf_i->file = NULL;
333 file->private_data = NULL;
334 return 0;
335}
336
337#if LINUX_VERSION_CODE > KERNEL_VERSION (2, 6, 25)
338static int
339sf_reg_fault(struct vm_area_struct *vma, struct vm_fault *vmf)
340#elif LINUX_VERSION_CODE >= KERNEL_VERSION (2, 6, 0)
341static struct page *
342sf_reg_nopage (struct vm_area_struct *vma, unsigned long vaddr, int *type)
343# define SET_TYPE(t) *type = (t)
344#else /* LINUX_VERSION_CODE < KERNEL_VERSION (2, 6, 0) */
345static struct page *
346sf_reg_nopage (struct vm_area_struct *vma, unsigned long vaddr, int unused)
347# define SET_TYPE(t)
348#endif
349{
350 struct page *page;
351 char *buf;
352 loff_t off;
353 uint32_t nread = PAGE_SIZE;
354 int err;
355 struct file *file = vma->vm_file;
356 struct inode *inode = file->f_dentry->d_inode;
357 struct sf_glob_info *sf_g = GET_GLOB_INFO (inode->i_sb);
358 struct sf_reg_info *sf_r = file->private_data;
359
360 TRACE ();
361#if LINUX_VERSION_CODE > KERNEL_VERSION (2, 6, 25)
362 if (vmf->pgoff > vma->vm_end)
363 return VM_FAULT_SIGBUS;
364#else
365 if (vaddr > vma->vm_end) {
366 SET_TYPE (VM_FAULT_SIGBUS);
367 return NOPAGE_SIGBUS;
368 }
369#endif
370
371 page = alloc_page (GFP_HIGHUSER);
372 if (!page) {
373 LogRelFunc(("failed to allocate page\n"));
374#if LINUX_VERSION_CODE > KERNEL_VERSION (2, 6, 25)
375 return VM_FAULT_OOM;
376#else
377 SET_TYPE (VM_FAULT_OOM);
378 return NOPAGE_OOM;
379#endif
380 }
381
382 buf = kmap (page);
383#if LINUX_VERSION_CODE > KERNEL_VERSION (2, 6, 25)
384 off = (vmf->pgoff << PAGE_SHIFT);
385#else
386 off = (vaddr - vma->vm_start) + (vma->vm_pgoff << PAGE_SHIFT);
387#endif
388 err = sf_reg_read_aux (__func__, sf_g, sf_r, buf, &nread, off);
389 if (err) {
390 kunmap (page);
391 put_page (page);
392#if LINUX_VERSION_CODE > KERNEL_VERSION (2, 6, 25)
393 return VM_FAULT_SIGBUS;
394#else
395 SET_TYPE (VM_FAULT_SIGBUS);
396 return NOPAGE_SIGBUS;
397#endif
398 }
399
400 BUG_ON (nread > PAGE_SIZE);
401 if (!nread) {
402#if LINUX_VERSION_CODE > KERNEL_VERSION (2, 6, 25)
403 clear_user_page (page_address (page), vmf->pgoff, page);
404#elif LINUX_VERSION_CODE >= KERNEL_VERSION (2, 6, 0)
405 clear_user_page (page_address (page), vaddr, page);
406#else
407 clear_user_page (page_address (page), vaddr);
408#endif
409 }
410 else {
411 memset (buf + nread, 0, PAGE_SIZE - nread);
412 }
413
414 flush_dcache_page (page);
415 kunmap (page);
416#if LINUX_VERSION_CODE > KERNEL_VERSION (2, 6, 25)
417 vmf->page = page;
418 return 0;
419#else
420 SET_TYPE (VM_FAULT_MAJOR);
421 return page;
422#endif
423}
424
425static struct vm_operations_struct sf_vma_ops = {
426#if LINUX_VERSION_CODE > KERNEL_VERSION (2, 6, 25)
427 .fault = sf_reg_fault
428#else
429 .nopage = sf_reg_nopage
430#endif
431};
432
433static int
434sf_reg_mmap (struct file *file, struct vm_area_struct *vma)
435{
436 TRACE ();
437 if (vma->vm_flags & VM_SHARED) {
438 LogFunc(("shared mmapping not available\n"));
439 return -EINVAL;
440 }
441
442 vma->vm_ops = &sf_vma_ops;
443 return 0;
444}
445
446struct file_operations sf_reg_fops = {
447 .read = sf_reg_read,
448 .open = sf_reg_open,
449 .write = sf_reg_write,
450 .release = sf_reg_release,
451 .mmap = sf_reg_mmap,
452#if LINUX_VERSION_CODE >= KERNEL_VERSION (2, 6, 0)
453# if LINUX_VERSION_CODE >= KERNEL_VERSION (2, 6, 23)
454 .splice_read = generic_file_splice_read,
455# else
456 .sendfile = generic_file_sendfile,
457# endif
458 .aio_read = generic_file_aio_read,
459 .aio_write = generic_file_aio_write,
460 .fsync = simple_sync_file,
461 .llseek = generic_file_llseek,
462#endif
463};
464
465
466struct inode_operations sf_reg_iops = {
467#if LINUX_VERSION_CODE < KERNEL_VERSION (2, 6, 0)
468 .revalidate = sf_inode_revalidate
469#else
470 .getattr = sf_getattr,
471 .setattr = sf_setattr
472#endif
473};
474
475
476#if LINUX_VERSION_CODE >= KERNEL_VERSION (2, 6, 0)
477static int
478sf_readpage(struct file *file, struct page *page)
479{
480 struct inode *inode = file->f_dentry->d_inode;
481 struct sf_glob_info *sf_g = GET_GLOB_INFO (inode->i_sb);
482 struct sf_reg_info *sf_r = file->private_data;
483 uint32_t nread = PAGE_SIZE;
484 char *buf;
485 loff_t off = ((loff_t)page->index) << PAGE_SHIFT;
486 int ret;
487
488 TRACE ();
489
490 buf = kmap(page);
491 ret = sf_reg_read_aux (__func__, sf_g, sf_r, buf, &nread, off);
492 if (ret) {
493 kunmap (page);
494 if (PageLocked(page))
495 unlock_page(page);
496 return ret;
497 }
498 BUG_ON (nread > PAGE_SIZE);
499 memset(&buf[nread], 0, PAGE_SIZE - nread);
500 flush_dcache_page (page);
501 kunmap (page);
502 SetPageUptodate(page);
503 unlock_page(page);
504 return 0;
505}
506
507static int
508sf_writepage(struct page *page, struct writeback_control *wbc)
509{
510 struct address_space *mapping = page->mapping;
511 struct inode *inode = mapping->host;
512 struct sf_glob_info *sf_g = GET_GLOB_INFO (inode->i_sb);
513 struct sf_inode_info *sf_i = GET_INODE_INFO (inode);
514 struct file *file = sf_i->file;
515 struct sf_reg_info *sf_r = file->private_data;
516 char *buf;
517 uint32_t nwritten = PAGE_SIZE;
518 int end_index = inode->i_size >> PAGE_SHIFT;
519 loff_t off = ((loff_t) page->index) << PAGE_SHIFT;
520 int err;
521
522 TRACE ();
523
524 if (page->index >= end_index)
525 nwritten = inode->i_size & (PAGE_SIZE-1);
526
527 buf = kmap(page);
528
529 err = sf_reg_write_aux (__func__, sf_g, sf_r, buf, &nwritten, off);
530 if (err < 0) {
531 ClearPageUptodate(page);
532 goto out;
533 }
534
535 if (off > inode->i_size)
536 inode->i_size = off;
537
538 if (PageError(page))
539 ClearPageError(page);
540 err = 0;
541out:
542 kunmap(page);
543
544 unlock_page(page);
545 return err;
546}
547
548# if LINUX_VERSION_CODE >= KERNEL_VERSION (2, 6, 24)
549int
550sf_write_begin(struct file *file, struct address_space *mapping, loff_t pos,
551 unsigned len, unsigned flags, struct page **pagep, void **fsdata)
552{
553 pgoff_t index = pos >> PAGE_SHIFT;
554
555 TRACE ();
556
557 *pagep = grab_cache_page_write_begin(mapping, index, flags);
558 if (!*pagep)
559 return -ENOMEM;
560 return 0;
561}
562
563int
564sf_write_end(struct file *file, struct address_space *mapping, loff_t pos,
565 unsigned len, unsigned copied, struct page *page, void *fsdata)
566{
567 struct inode *inode = mapping->host;
568 struct sf_glob_info *sf_g = GET_GLOB_INFO (inode->i_sb);
569 struct sf_reg_info *sf_r = file->private_data;
570 void *buf;
571 unsigned from = pos & (PAGE_SIZE - 1);
572 uint32_t nwritten = len;
573 int err;
574
575 TRACE ();
576
577 buf = kmap(page);
578 err = sf_reg_write_aux (__func__, sf_g, sf_r, buf+from, &nwritten, pos);
579 kunmap(page);
580
581 if (!PageUptodate(page) && err == PAGE_SIZE)
582 SetPageUptodate(page);
583
584 if (err >= 0) {
585 pos += nwritten;
586 if (pos > inode->i_size)
587 inode->i_size = pos;
588 }
589
590 unlock_page(page);
591 page_cache_release(page);
592
593 return nwritten;
594}
595
596# endif /* KERNEL_VERSION >= 2.6.24 */
597
598struct address_space_operations sf_reg_aops = {
599 .readpage = sf_readpage,
600 .writepage = sf_writepage,
601# if LINUX_VERSION_CODE >= KERNEL_VERSION (2, 6, 24)
602 .write_begin = sf_write_begin,
603 .write_end = sf_write_end,
604# else
605 .prepare_write = simple_prepare_write,
606 .commit_write = simple_commit_write,
607# endif
608};
609#endif
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