VirtualBox

source: vbox/trunk/src/VBox/Additions/solaris/SharedFolders/vboxfs_prov.c@ 39561

Last change on this file since 39561 was 39297, checked in by vboxsync, 13 years ago

Additions/solaris/SharedFolders: Pass file/dir mode directly to create/mkdir calls instead of using set_attr explicitly after each call.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 22.5 KB
Line 
1/** @file
2 * VirtualBox File System for Solaris Guests, provider implementation.
3 * Portions contributed by: Ronald.
4 */
5
6/*
7 * Copyright (C) 2008-2010 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27/*
28 * Provider interfaces for shared folder file system.
29 */
30
31#include <sys/types.h>
32#include <sys/stat.h>
33#include <sys/mntent.h>
34#include <sys/param.h>
35#include <sys/modctl.h>
36#include <sys/mount.h>
37#include <sys/policy.h>
38#include <sys/atomic.h>
39#include <sys/sysmacros.h>
40#include <sys/ddi.h>
41#include <sys/sunddi.h>
42#include <sys/dirent.h>
43#include "vboxfs_prov.h"
44#ifdef u
45#undef u
46#endif
47#include "../../common/VBoxGuestLib/VBoxGuestR0LibSharedFolders.h"
48
49#define SFPROV_VERSION 1
50
51static VBSFCLIENT vbox_client;
52
53static int sfprov_vbox2errno(int rc)
54{
55 if (rc == VERR_ACCESS_DENIED)
56 return (EACCES);
57 return (RTErrConvertToErrno(rc));
58}
59
60/*
61 * utility to create strings
62 */
63static SHFLSTRING *
64sfprov_string(char *path, int *sz)
65{
66 SHFLSTRING *str;
67 int len = strlen(path);
68
69 *sz = len + 1 + sizeof (*str) - sizeof (str->String);
70 str = kmem_zalloc(*sz, KM_SLEEP);
71 str->u16Size = len + 1;
72 str->u16Length = len;
73 strcpy(str->String.utf8, path);
74 return (str);
75}
76
77sfp_connection_t *
78sfprov_connect(int version)
79{
80 /*
81 * only one version for now, so must match
82 */
83 int rc = -1;
84 if (version != SFPROV_VERSION)
85 {
86 cmn_err(CE_WARN, "sfprov_connect: wrong version. version=%d expected=%d\n", version, SFPROV_VERSION);
87 return NULL;
88 }
89 rc = vboxInit();
90 if (RT_SUCCESS(rc))
91 {
92 rc = vboxConnect(&vbox_client);
93 if (RT_SUCCESS(rc))
94 {
95 rc = vboxCallSetUtf8(&vbox_client);
96 if (RT_SUCCESS(rc))
97 {
98 return ((sfp_connection_t *)&vbox_client);
99 }
100 else
101 cmn_err(CE_WARN, "sfprov_connect: vboxCallSetUtf8() failed\n");
102
103 vboxDisconnect(&vbox_client);
104 }
105 else
106 cmn_err(CE_WARN, "sfprov_connect: vboxConnect() failed rc=%d\n", rc);
107 vboxUninit();
108 }
109 else
110 cmn_err(CE_WARN, "sfprov_connect: vboxInit() failed rc=%d\n", rc);
111 return (NULL);
112}
113
114void
115sfprov_disconnect(sfp_connection_t *conn)
116{
117 if (conn != (sfp_connection_t *)&vbox_client)
118 cmn_err(CE_WARN, "sfprov_disconnect: bad argument\n");
119 vboxDisconnect(&vbox_client);
120 vboxUninit();
121}
122
123
124/*
125 * representation of an active mount point
126 */
127struct sfp_mount {
128 VBSFMAP map;
129};
130
131int
132sfprov_mount(sfp_connection_t *conn, char *path, sfp_mount_t **mnt)
133{
134 sfp_mount_t *m;
135 SHFLSTRING *str;
136 int size;
137 int rc;
138
139 m = kmem_zalloc(sizeof (*m), KM_SLEEP);
140 str = sfprov_string(path, &size);
141 rc = vboxCallMapFolder(&vbox_client, str, &m->map);
142 if (RT_FAILURE(rc)) {
143 cmn_err(CE_WARN, "sfprov_mount: vboxCallMapFolder() failed rc=%d\n", rc);
144 kmem_free(m, sizeof (*m));
145 *mnt = NULL;
146 rc = EINVAL;
147 } else {
148 *mnt = m;
149 rc = 0;
150 }
151 kmem_free(str, size);
152 return (rc);
153}
154
155int
156sfprov_unmount(sfp_mount_t *mnt)
157{
158 int rc;
159
160 rc = vboxCallUnmapFolder(&vbox_client, &mnt->map);
161 if (RT_FAILURE(rc)) {
162 cmn_err(CE_WARN, "sfprov_mount: vboxCallUnmapFolder() failed rc=%d\n", rc);
163 rc = EINVAL;
164 } else {
165 rc = 0;
166 }
167 kmem_free(mnt, sizeof (*mnt));
168 return (rc);
169}
170
171/*
172 * query information about a mounted file system
173 */
174int
175sfprov_get_fsinfo(sfp_mount_t *mnt, sffs_fsinfo_t *fsinfo)
176{
177 int rc;
178 SHFLVOLINFO info;
179 uint32_t bytes = sizeof(SHFLVOLINFO);
180
181 rc = vboxCallFSInfo(&vbox_client, &mnt->map, 0,
182 (SHFL_INFO_GET | SHFL_INFO_VOLUME), &bytes, (SHFLDIRINFO *)&info);
183 if (RT_FAILURE(rc))
184 return (EINVAL);
185
186 fsinfo->blksize = info.ulBytesPerAllocationUnit;
187 fsinfo->blksused = (info.ullTotalAllocationBytes - info.ullAvailableAllocationBytes) / info.ulBytesPerAllocationUnit;
188 fsinfo->blksavail = info.ullAvailableAllocationBytes / info.ulBytesPerAllocationUnit;
189 fsinfo->maxnamesize = info.fsProperties.cbMaxComponent;
190 fsinfo->readonly = info.fsProperties.fReadOnly;
191 return (0);
192}
193
194/*
195 * file/directory information conversions.
196 */
197static void
198sfprov_fmode_from_mode(RTFMODE *fMode, mode_t mode)
199{
200 RTFMODE m = 0;
201
202#define mode_set(r) ((mode & (S_##r)) ? RTFS_UNIX_##r : 0)
203 m = mode_set (ISUID);
204 m |= mode_set (ISGID);
205 m |= (mode & S_ISVTX) ? RTFS_UNIX_ISTXT : 0;
206 m |= mode_set (IRUSR);
207 m |= mode_set (IWUSR);
208 m |= mode_set (IXUSR);
209 m |= mode_set (IRGRP);
210 m |= mode_set (IWGRP);
211 m |= mode_set (IXGRP);
212 m |= mode_set (IROTH);
213 m |= mode_set (IWOTH);
214 m |= mode_set (IXOTH);
215
216 if (S_ISDIR(mode))
217 m |= RTFS_TYPE_DIRECTORY;
218 else if (S_ISREG(mode))
219 m |= RTFS_TYPE_FILE;
220 else if (S_ISFIFO(mode))
221 m |= RTFS_TYPE_FIFO;
222 else if (S_ISCHR(mode))
223 m |= RTFS_TYPE_DEV_CHAR;
224 else if (S_ISBLK(mode))
225 m |= RTFS_TYPE_DEV_BLOCK;
226 else if (S_ISLNK(mode))
227 m |= RTFS_TYPE_SYMLINK;
228 else if (S_ISSOCK(mode))
229 m |= RTFS_TYPE_SOCKET;
230 else
231 m |= RTFS_TYPE_FILE;
232
233 *fMode = m;
234}
235
236static void
237sfprov_mode_from_fmode(mode_t *mode, RTFMODE fMode)
238{
239 mode_t m = 0;
240
241 if (RTFS_IS_DIRECTORY(fMode))
242 m |= S_IFDIR;
243 else if (RTFS_IS_FILE(fMode))
244 m |= S_IFREG;
245 else if (RTFS_IS_FIFO(fMode))
246 m |= S_IFIFO;
247 else if (RTFS_IS_DEV_CHAR(fMode))
248 m |= S_IFCHR;
249 else if (RTFS_IS_DEV_BLOCK(fMode))
250 m |= S_IFBLK;
251 else if (RTFS_IS_SYMLINK(fMode))
252 m |= S_IFLNK;
253 else if (RTFS_IS_SOCKET(fMode))
254 m |= S_IFSOCK;
255
256 if (fMode & RTFS_UNIX_IRUSR)
257 m |= S_IRUSR;
258 if (fMode & RTFS_UNIX_IWUSR)
259 m |= S_IWUSR;
260 if (fMode & RTFS_UNIX_IXUSR)
261 m |= S_IXUSR;
262 if (fMode & RTFS_UNIX_IRGRP)
263 m |= S_IRGRP;
264 if (fMode & RTFS_UNIX_IWGRP)
265 m |= S_IWGRP;
266 if (fMode & RTFS_UNIX_IXGRP)
267 m |= S_IXGRP;
268 if (fMode & RTFS_UNIX_IROTH)
269 m |= S_IROTH;
270 if (fMode & RTFS_UNIX_IWOTH)
271 m |= S_IWOTH;
272 if (fMode & RTFS_UNIX_IXOTH)
273 m |= S_IXOTH;
274 if (fMode & RTFS_UNIX_ISUID)
275 m |= S_ISUID;
276 if (fMode & RTFS_UNIX_ISGID)
277 m |= S_ISGID;
278 if (fMode & RTFS_UNIX_ISTXT)
279 m |= S_ISVTX;
280 *mode = m;
281}
282
283static void
284sfprov_ftime_from_timespec(timestruc_t *time, RTTIMESPEC *ts)
285{
286 uint64_t nanosec = RTTimeSpecGetNano(ts);
287 time->tv_sec = nanosec / UINT64_C(1000000000);
288 time->tv_nsec = nanosec % UINT64_C(1000000000);
289}
290
291static void
292sfprov_stat_from_info(sffs_stat_t *stat, SHFLFSOBJINFO *info)
293{
294 sfprov_mode_from_fmode(&stat->sf_mode, info->Attr.fMode);
295 stat->sf_size = info->cbObject;
296 stat->sf_alloc = info->cbAllocated;
297 sfprov_ftime_from_timespec(&stat->sf_atime, &info->AccessTime);
298 sfprov_ftime_from_timespec(&stat->sf_mtime, &info->ModificationTime);
299 sfprov_ftime_from_timespec(&stat->sf_ctime, &info->ChangeTime);
300}
301
302/*
303 * File operations: open/close/read/write/etc.
304 *
305 * open/create can return any relevant errno, however ENOENT
306 * generally means that the host file didn't exist.
307 */
308struct sfp_file {
309 SHFLHANDLE handle;
310 VBSFMAP map; /* need this again for the close operation */
311};
312
313int
314sfprov_create(
315 sfp_mount_t *mnt,
316 char *path,
317 mode_t mode,
318 sfp_file_t **fp,
319 sffs_stat_t *stat)
320{
321
322 int rc;
323 SHFLCREATEPARMS parms;
324 SHFLSTRING *str;
325 int size;
326 sfp_file_t *newfp;
327
328 str = sfprov_string(path, &size);
329 parms.Handle = SHFL_HANDLE_NIL;
330 parms.Info.cbObject = 0;
331 sfprov_fmode_from_mode(&parms.Info.Attr.fMode, mode);
332 parms.CreateFlags = SHFL_CF_ACT_CREATE_IF_NEW |
333 SHFL_CF_ACT_REPLACE_IF_EXISTS | SHFL_CF_ACCESS_READWRITE;
334 rc = vboxCallCreate(&vbox_client, &mnt->map, str, &parms);
335 kmem_free(str, size);
336
337 if (RT_FAILURE(rc))
338 {
339 if (rc != VERR_ACCESS_DENIED && rc != VERR_WRITE_PROTECT)
340 cmn_err(CE_WARN, "sfprov_create: vboxCallCreate failed! path=%s rc=%d\n", path, rc);
341 return (sfprov_vbox2errno(rc));
342 }
343 if (parms.Handle == SHFL_HANDLE_NIL) {
344 if (parms.Result == SHFL_FILE_EXISTS)
345 return (EEXIST);
346 return (ENOENT);
347 }
348 newfp = kmem_alloc(sizeof(sfp_file_t), KM_SLEEP);
349 newfp->handle = parms.Handle;
350 newfp->map = mnt->map;
351 *fp = newfp;
352 sfprov_stat_from_info(stat, &parms.Info);
353 return (0);
354}
355
356int
357sfprov_open(sfp_mount_t *mnt, char *path, sfp_file_t **fp)
358{
359 int rc;
360 SHFLCREATEPARMS parms;
361 SHFLSTRING *str;
362 int size;
363 sfp_file_t *newfp;
364
365 /*
366 * First we attempt to open it read/write. If that fails we
367 * try read only.
368 */
369 bzero(&parms, sizeof(parms));
370 str = sfprov_string(path, &size);
371 parms.Handle = SHFL_HANDLE_NIL;
372 parms.Info.cbObject = 0;
373 parms.CreateFlags = SHFL_CF_ACT_FAIL_IF_NEW | SHFL_CF_ACCESS_READWRITE;
374 rc = vboxCallCreate(&vbox_client, &mnt->map, str, &parms);
375 if (RT_FAILURE(rc) && rc != VERR_ACCESS_DENIED) {
376 kmem_free(str, size);
377 return (sfprov_vbox2errno(rc));
378 }
379 if (parms.Handle == SHFL_HANDLE_NIL) {
380 if (parms.Result == SHFL_PATH_NOT_FOUND ||
381 parms.Result == SHFL_FILE_NOT_FOUND) {
382 kmem_free(str, size);
383 return (ENOENT);
384 }
385 parms.CreateFlags =
386 SHFL_CF_ACT_FAIL_IF_NEW | SHFL_CF_ACCESS_READ;
387 rc = vboxCallCreate(&vbox_client, &mnt->map, str, &parms);
388 if (RT_FAILURE(rc)) {
389 kmem_free(str, size);
390 return (sfprov_vbox2errno(rc));
391 }
392 if (parms.Handle == SHFL_HANDLE_NIL) {
393 kmem_free(str, size);
394 return (ENOENT);
395 }
396 }
397 else
398 kmem_free(str, size);
399 newfp = kmem_alloc(sizeof(sfp_file_t), KM_SLEEP);
400 newfp->handle = parms.Handle;
401 newfp->map = mnt->map;
402 *fp = newfp;
403 return (0);
404}
405
406int
407sfprov_trunc(sfp_mount_t *mnt, char *path)
408{
409 int rc;
410 SHFLCREATEPARMS parms;
411 SHFLSTRING *str;
412 int size;
413
414 /*
415 * open it read/write.
416 */
417 str = sfprov_string(path, &size);
418 parms.Handle = 0;
419 parms.Info.cbObject = 0;
420 parms.CreateFlags = SHFL_CF_ACT_FAIL_IF_NEW | SHFL_CF_ACCESS_READWRITE |
421 SHFL_CF_ACT_OVERWRITE_IF_EXISTS;
422 rc = vboxCallCreate(&vbox_client, &mnt->map, str, &parms);
423 kmem_free(str, size);
424
425 if (RT_FAILURE(rc)) {
426 return (EINVAL);
427 }
428 (void)vboxCallClose(&vbox_client, &mnt->map, parms.Handle);
429 return (0);
430}
431
432int
433sfprov_close(sfp_file_t *fp)
434{
435 int rc;
436
437 rc = vboxCallClose(&vbox_client, &fp->map, fp->handle);
438 kmem_free(fp, sizeof(sfp_file_t));
439 return (0);
440}
441
442int
443sfprov_read(sfp_file_t *fp, char *buffer, uint64_t offset, uint32_t *numbytes)
444{
445 int rc;
446
447 rc = vboxCallRead(&vbox_client, &fp->map, fp->handle, offset,
448 numbytes, (uint8_t *)buffer, 0); /* what is that last arg? */
449 if (RT_FAILURE(rc))
450 return (EINVAL);
451 return (0);
452}
453
454int
455sfprov_write(sfp_file_t *fp, char *buffer, uint64_t offset, uint32_t *numbytes)
456{
457 int rc;
458
459 rc = vboxCallWrite(&vbox_client, &fp->map, fp->handle, offset,
460 numbytes, (uint8_t *)buffer, 0); /* what is that last arg? */
461 if (RT_FAILURE(rc))
462 return (EINVAL);
463 return (0);
464}
465
466int
467sfprov_fsync(sfp_file_t *fp)
468{
469 int rc;
470
471 rc = vboxCallFlush(&vbox_client, &fp->map, fp->handle);
472 if (RT_FAILURE(rc))
473 return (EIO);
474 return (0);
475}
476
477
478static int
479sfprov_getinfo(sfp_mount_t *mnt, char *path, PSHFLFSOBJINFO info)
480{
481 int rc;
482 SHFLCREATEPARMS parms;
483 SHFLSTRING *str;
484 int size;
485
486 str = sfprov_string(path, &size);
487 parms.Handle = 0;
488 parms.Info.cbObject = 0;
489 parms.CreateFlags = SHFL_CF_LOOKUP | SHFL_CF_ACT_FAIL_IF_NEW;
490 rc = vboxCallCreate(&vbox_client, &mnt->map, str, &parms);
491 kmem_free(str, size);
492
493 if (RT_FAILURE(rc))
494 return (EINVAL);
495 if (parms.Result != SHFL_FILE_EXISTS)
496 return (ENOENT);
497 *info = parms.Info;
498 return (0);
499}
500
501/*
502 * get information about a file (or directory)
503 */
504int
505sfprov_get_mode(sfp_mount_t *mnt, char *path, mode_t *mode)
506{
507 int rc;
508 SHFLFSOBJINFO info;
509
510 rc = sfprov_getinfo(mnt, path, &info);
511 if (rc)
512 return (rc);
513 sfprov_mode_from_fmode(mode, info.Attr.fMode);
514 return (0);
515}
516
517int
518sfprov_get_size(sfp_mount_t *mnt, char *path, uint64_t *size)
519{
520 int rc;
521 SHFLFSOBJINFO info;
522
523 rc = sfprov_getinfo(mnt, path, &info);
524 if (rc)
525 return (rc);
526 *size = info.cbObject;
527 return (0);
528}
529
530
531int
532sfprov_get_atime(sfp_mount_t *mnt, char *path, timestruc_t *time)
533{
534 int rc;
535 SHFLFSOBJINFO info;
536
537 rc = sfprov_getinfo(mnt, path, &info);
538 if (rc)
539 return (rc);
540 sfprov_ftime_from_timespec(time, &info.AccessTime);
541 return (0);
542}
543
544int
545sfprov_get_mtime(sfp_mount_t *mnt, char *path, timestruc_t *time)
546{
547 int rc;
548 SHFLFSOBJINFO info;
549
550 rc = sfprov_getinfo(mnt, path, &info);
551 if (rc)
552 return (rc);
553 sfprov_ftime_from_timespec(time, &info.ModificationTime);
554 return (0);
555}
556
557int
558sfprov_get_ctime(sfp_mount_t *mnt, char *path, timestruc_t *time)
559{
560 int rc;
561 SHFLFSOBJINFO info;
562
563 rc = sfprov_getinfo(mnt, path, &info);
564 if (rc)
565 return (rc);
566 sfprov_ftime_from_timespec(time, &info.ChangeTime);
567 return (0);
568}
569
570int
571sfprov_get_attr(sfp_mount_t *mnt, char *path, sffs_stat_t *attr)
572{
573 int rc;
574 SHFLFSOBJINFO info;
575
576 rc = sfprov_getinfo(mnt, path, &info);
577 if (rc)
578 return (rc);
579 sfprov_stat_from_info(attr, &info);
580 return (0);
581}
582
583static void
584sfprov_timespec_from_ftime(RTTIMESPEC *ts, timestruc_t time)
585{
586 uint64_t nanosec = UINT64_C(1000000000) * time.tv_sec + time.tv_nsec;
587 RTTimeSpecSetNano(ts, nanosec);
588}
589
590int
591sfprov_set_attr(
592 sfp_mount_t *mnt,
593 char *path,
594 uint_t mask,
595 mode_t mode,
596 timestruc_t atime,
597 timestruc_t mtime,
598 timestruc_t ctime)
599{
600 int rc, err;
601 SHFLCREATEPARMS parms;
602 SHFLSTRING *str;
603 SHFLFSOBJINFO info;
604 uint32_t bytes;
605 int str_size;
606
607 str = sfprov_string(path, &str_size);
608 parms.Handle = 0;
609 parms.Info.cbObject = 0;
610 parms.CreateFlags = SHFL_CF_ACT_OPEN_IF_EXISTS
611 | SHFL_CF_ACT_FAIL_IF_NEW
612 | SHFL_CF_ACCESS_ATTR_WRITE;
613
614 rc = vboxCallCreate(&vbox_client, &mnt->map, str, &parms);
615
616 if (RT_FAILURE(rc)) {
617 cmn_err(CE_WARN, "sfprov_set_attr: vboxCallCreate(%s) failed rc=%d\n",
618 path, rc);
619 err = EINVAL;
620 goto fail2;
621 }
622 if (parms.Result != SHFL_FILE_EXISTS) {
623 err = ENOENT;
624 goto fail1;
625 }
626
627 RT_ZERO(info);
628 if (mask & AT_MODE)
629 sfprov_fmode_from_mode(&info.Attr.fMode, mode);
630 if (mask & AT_ATIME)
631 sfprov_timespec_from_ftime(&info.AccessTime, atime);
632 if (mask & AT_MTIME)
633 sfprov_timespec_from_ftime(&info.ModificationTime, mtime);
634 if (mask & AT_CTIME)
635 sfprov_timespec_from_ftime(&info.ChangeTime, ctime);
636
637 bytes = sizeof(info);
638 rc = vboxCallFSInfo(&vbox_client, &mnt->map, parms.Handle,
639 (SHFL_INFO_SET | SHFL_INFO_FILE), &bytes, (SHFLDIRINFO *)&info);
640 if (RT_FAILURE(rc)) {
641 if (rc != VERR_ACCESS_DENIED && rc != VERR_WRITE_PROTECT)
642 {
643 cmn_err(CE_WARN, "sfprov_set_attr: vboxCallFSInfo(%s, FILE) failed rc=%d\n",
644 path, rc);
645 }
646 err = sfprov_vbox2errno(rc);
647 goto fail1;
648 }
649
650 err = 0;
651
652fail1:
653 rc = vboxCallClose(&vbox_client, &mnt->map, parms.Handle);
654 if (RT_FAILURE(rc)) {
655 cmn_err(CE_WARN, "sfprov_set_attr: vboxCallClose(%s) failed rc=%d\n",
656 path, rc);
657 }
658fail2:
659 kmem_free(str, str_size);
660 return err;
661}
662
663int
664sfprov_set_size(sfp_mount_t *mnt, char *path, uint64_t size)
665{
666 int rc, err;
667 SHFLCREATEPARMS parms;
668 SHFLSTRING *str;
669 SHFLFSOBJINFO info;
670 uint32_t bytes;
671 int str_size;
672
673 str = sfprov_string(path, &str_size);
674 parms.Handle = 0;
675 parms.Info.cbObject = 0;
676 parms.CreateFlags = SHFL_CF_ACT_OPEN_IF_EXISTS
677 | SHFL_CF_ACT_FAIL_IF_NEW
678 | SHFL_CF_ACCESS_WRITE;
679
680 rc = vboxCallCreate(&vbox_client, &mnt->map, str, &parms);
681
682 if (RT_FAILURE(rc)) {
683 cmn_err(CE_WARN, "sfprov_set_size: vboxCallCreate(%s) failed rc=%d\n",
684 path, rc);
685 err = EINVAL;
686 goto fail2;
687 }
688 if (parms.Result != SHFL_FILE_EXISTS) {
689 err = ENOENT;
690 goto fail1;
691 }
692
693 RT_ZERO(info);
694 info.cbObject = size;
695 bytes = sizeof(info);
696 rc = vboxCallFSInfo(&vbox_client, &mnt->map, parms.Handle,
697 (SHFL_INFO_SET | SHFL_INFO_SIZE), &bytes, (SHFLDIRINFO *)&info);
698 if (RT_FAILURE(rc)) {
699 cmn_err(CE_WARN, "sfprov_set_size: vboxCallFSInfo(%s, SIZE) failed rc=%d\n",
700 path, rc);
701 err = sfprov_vbox2errno(rc);
702 goto fail1;
703 }
704
705 err = 0;
706
707fail1:
708 rc = vboxCallClose(&vbox_client, &mnt->map, parms.Handle);
709 if (RT_FAILURE(rc)) {
710 cmn_err(CE_WARN, "sfprov_set_size: vboxCallClose(%s) failed rc=%d\n",
711 path, rc);
712 }
713fail2:
714 kmem_free(str, str_size);
715 return err;
716}
717
718/*
719 * Directory operations
720 */
721int
722sfprov_mkdir(
723 sfp_mount_t *mnt,
724 char *path,
725 mode_t mode,
726 sfp_file_t **fp,
727 sffs_stat_t *stat)
728{
729 int rc;
730 SHFLCREATEPARMS parms;
731 SHFLSTRING *str;
732 int size;
733 sfp_file_t *newfp;
734
735 str = sfprov_string(path, &size);
736 parms.Handle = SHFL_HANDLE_NIL;
737 parms.Info.cbObject = 0;
738 sfprov_fmode_from_mode(&parms.Info.Attr.fMode, mode);
739 parms.CreateFlags = SHFL_CF_DIRECTORY | SHFL_CF_ACT_CREATE_IF_NEW |
740 SHFL_CF_ACT_FAIL_IF_EXISTS | SHFL_CF_ACCESS_READ;
741 rc = vboxCallCreate(&vbox_client, &mnt->map, str, &parms);
742 kmem_free(str, size);
743
744 if (RT_FAILURE(rc))
745 return (sfprov_vbox2errno(rc));
746 if (parms.Handle == SHFL_HANDLE_NIL) {
747 if (parms.Result == SHFL_FILE_EXISTS)
748 return (EEXIST);
749 return (ENOENT);
750 }
751 newfp = kmem_alloc(sizeof(sfp_file_t), KM_SLEEP);
752 newfp->handle = parms.Handle;
753 newfp->map = mnt->map;
754 *fp = newfp;
755 sfprov_stat_from_info(stat, &parms.Info);
756 return (0);
757}
758
759int
760sfprov_set_show_symlinks(void)
761{
762 int rc;
763
764 rc = vboxCallSetSymlinks(&vbox_client);
765 if (RT_FAILURE(rc))
766 return (sfprov_vbox2errno(rc));
767
768 return (0);
769}
770
771int
772sfprov_remove(sfp_mount_t *mnt, char *path, uint_t is_link)
773{
774 int rc;
775 SHFLSTRING *str;
776 int size;
777
778 str = sfprov_string(path, &size);
779 rc = vboxCallRemove(&vbox_client, &mnt->map, str,
780 SHFL_REMOVE_FILE | (is_link ? SHFL_REMOVE_SYMLINK : 0));
781 kmem_free(str, size);
782 if (RT_FAILURE(rc))
783 return (sfprov_vbox2errno(rc));
784 return (0);
785}
786
787int
788sfprov_readlink(
789 sfp_mount_t *mnt,
790 char *path,
791 char *target,
792 size_t tgt_size)
793{
794 int rc;
795 SHFLSTRING *str;
796 int size;
797
798 str = sfprov_string(path, &size);
799
800 rc = vboxReadLink(&vbox_client, &mnt->map, str, (uint32_t) tgt_size,
801 target);
802 if (RT_FAILURE(rc))
803 rc = sfprov_vbox2errno(rc);
804
805 kmem_free(str, size);
806 return (rc);
807}
808
809int
810sfprov_symlink(
811 sfp_mount_t *mnt,
812 char *linkname,
813 char *target,
814 sffs_stat_t *stat)
815{
816 int rc;
817 SHFLSTRING *lnk, *tgt;
818 int lnk_size, tgt_size;
819 SHFLFSOBJINFO info;
820
821 lnk = sfprov_string(linkname, &lnk_size);
822 tgt = sfprov_string(target, &tgt_size);
823
824 rc = vboxCallSymlink(&vbox_client, &mnt->map, lnk, tgt, &info);
825 if (RT_FAILURE(rc)) {
826 rc = sfprov_vbox2errno(rc);
827 goto done;
828 }
829
830 if (stat != NULL)
831 sfprov_stat_from_info(stat, &info);
832
833done:
834 kmem_free(lnk, lnk_size);
835 kmem_free(tgt, tgt_size);
836
837 return (rc);
838}
839
840int
841sfprov_rmdir(sfp_mount_t *mnt, char *path)
842{
843 int rc;
844 SHFLSTRING *str;
845 int size;
846
847 str = sfprov_string(path, &size);
848 rc = vboxCallRemove(&vbox_client, &mnt->map, str, SHFL_REMOVE_DIR);
849 kmem_free(str, size);
850 if (RT_FAILURE(rc))
851 return (sfprov_vbox2errno(rc));
852 return (0);
853}
854
855int
856sfprov_rename(sfp_mount_t *mnt, char *from, char *to, uint_t is_dir)
857{
858 int rc;
859 SHFLSTRING *old, *new;
860 int old_size, new_size;
861
862 old = sfprov_string(from, &old_size);
863 new = sfprov_string(to, &new_size);
864 rc = vboxCallRename(&vbox_client, &mnt->map, old, new,
865 (is_dir ? SHFL_RENAME_DIR : SHFL_RENAME_FILE) |
866 SHFL_RENAME_REPLACE_IF_EXISTS);
867 kmem_free(old, old_size);
868 kmem_free(new, new_size);
869 if (RT_FAILURE(rc))
870 return (sfprov_vbox2errno(rc));
871 return (0);
872}
873
874
875/*
876 * Read all filenames in a directory.
877 *
878 * - success - all entries read and returned
879 * - ENOENT - Couldn't open the directory for reading
880 * - EINVAL - Internal error of some kind
881 *
882 * On successful return, *dirents points to a list of sffs_dirents_t;
883 * for each dirent, all fields except the d_ino will be set appropriately.
884 * The caller is responsible for freeing the dirents buffer.
885 */
886int
887sfprov_readdir(
888 sfp_mount_t *mnt,
889 char *path,
890 sffs_dirents_t **dirents)
891{
892 int error;
893 char *cp;
894 int len;
895 SHFLSTRING *mask_str = NULL; /* must be path with "/*" appended */
896 int mask_size;
897 sfp_file_t *fp;
898 uint32_t infobuff_alloc = 16384;
899 SHFLDIRINFO *infobuff = NULL, *info;
900 uint32_t numbytes;
901 uint32_t nents;
902 uint32_t size;
903 off_t offset;
904 sffs_dirents_t *cur_buf;
905 struct sffs_dirent *dirent;
906 unsigned short reclen;
907 unsigned short entlen;
908
909 *dirents = NULL;
910
911 error = sfprov_open(mnt, path, &fp);
912 if (error != 0)
913 return (ENOENT);
914
915 /*
916 * Allocate the first dirents buffers.
917 */
918 *dirents = kmem_alloc(SFFS_DIRENTS_SIZE, KM_SLEEP);
919 if (*dirents == NULL) {
920 error = (ENOSPC);
921 goto done;
922 }
923 cur_buf = *dirents;
924 cur_buf->sf_next = NULL;
925 cur_buf->sf_len = 0;
926
927 /*
928 * Create mask that VBox expects. This needs to be the directory path,
929 * plus a "*" wildcard to get all files.
930 */
931 len = strlen(path) + 3;
932 cp = kmem_alloc(len, KM_SLEEP);
933 if (cp == NULL) {
934 error = (ENOSPC);
935 goto done;
936 }
937 strcpy(cp, path);
938 strcat(cp, "/*");
939 mask_str = sfprov_string(cp, &mask_size);
940 kmem_free(cp, len);
941
942 /*
943 * Now loop using vboxCallDirInfo
944 */
945 infobuff = kmem_alloc(infobuff_alloc, KM_SLEEP);
946 if (infobuff == NULL) {
947 error = (ENOSPC);
948 goto done;
949 }
950
951 offset = 0;
952 for (;;) {
953 numbytes = infobuff_alloc;
954 error = vboxCallDirInfo(&vbox_client, &fp->map, fp->handle,
955 mask_str, 0, 0, &numbytes, infobuff, &nents);
956 switch (error) {
957
958 case VINF_SUCCESS:
959 /* fallthrough */
960 case VERR_NO_MORE_FILES:
961 break;
962
963 case VERR_NO_TRANSLATION:
964 /* XXX ??? */
965 break;
966
967 default:
968 error = sfprov_vbox2errno(error);
969 goto done;
970 }
971
972 /*
973 * Create the dirent_t's and save the stats for each name
974 */
975 for (info = infobuff; (char *) info < (char *) infobuff + numbytes; nents--) {
976 /* expand buffers if we need more space */
977 reclen = DIRENT64_RECLEN(strlen(info->name.String.utf8));
978 entlen = sizeof(sffs_stat_t) + reclen;
979 if (SFFS_DIRENTS_OFF + cur_buf->sf_len + entlen > SFFS_DIRENTS_SIZE) {
980 cur_buf->sf_next = kmem_alloc(SFFS_DIRENTS_SIZE, KM_SLEEP);
981 if (cur_buf->sf_next == NULL) {
982 error = ENOSPC;
983 goto done;
984 }
985 cur_buf = cur_buf->sf_next;
986 cur_buf->sf_next = NULL;
987 cur_buf->sf_len = 0;
988 }
989
990 /* create the dirent with the name, offset, and len */
991 dirent = (struct sffs_dirent *)
992 (((char *) &cur_buf->sf_entries[0]) + cur_buf->sf_len);
993 strncpy(&dirent->sf_entry.d_name[0], info->name.String.utf8, DIRENT64_NAMELEN(reclen));
994 dirent->sf_entry.d_reclen = reclen;
995 offset += entlen;
996 dirent->sf_entry.d_off = offset;
997
998 /* save the stats */
999 sfprov_stat_from_info(&dirent->sf_stat, &info->Info);
1000
1001 /* next info */
1002 cur_buf->sf_len += entlen;
1003 size = offsetof (SHFLDIRINFO, name.String) + info->name.u16Size;
1004 info = (SHFLDIRINFO *) ((uintptr_t) info + size);
1005 }
1006 ASSERT(nents == 0);
1007 ASSERT((char *) info == (char *) infobuff + numbytes);
1008
1009 if (error == VERR_NO_MORE_FILES)
1010 break;
1011 }
1012 error = 0;
1013
1014done:
1015 if (error != 0) {
1016 while (*dirents) {
1017 cur_buf = (*dirents)->sf_next;
1018 kmem_free(*dirents, SFFS_DIRENTS_SIZE);
1019 *dirents = cur_buf;
1020 }
1021 }
1022 if (infobuff != NULL)
1023 kmem_free(infobuff, infobuff_alloc);
1024 if (mask_str != NULL)
1025 kmem_free(mask_str, mask_size);
1026 sfprov_close(fp);
1027 return (error);
1028}
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