VirtualBox

source: vbox/trunk/src/VBox/Additions/linux/sharedfolders/mount.vboxsf.c@ 8207

Last change on this file since 8207 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.3 KB
Line 
1/** @file
2 * vboxvfs -- VirtualBox Guest Additions for Linux: mount(8) helper
3 *
4 * Parses options provided by mount (or user directly)
5 * Packs them into struct vbsfmount and passes to mount(2)
6 * Optionally adds entries to mtab
7 */
8
9/*
10 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
11 *
12 * This file is part of VirtualBox Open Source Edition (OSE), as
13 * available from http://www.virtualbox.org. This file is free software;
14 * you can redistribute it and/or modify it under the terms of the GNU
15 * General Public License (GPL) as published by the Free Software
16 * Foundation, in version 2 as it comes in the "COPYING" file of the
17 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
18 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
19 *
20 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
21 * Clara, CA 95054 USA or visit http://www.sun.com if you need
22 * additional information or have any questions.
23 */
24
25
26#ifndef _GNU_SOURCE
27#define _GNU_SOURCE
28#endif
29
30/* #define DEBUG */
31#define DBG if (0)
32#include <errno.h>
33#include <fcntl.h>
34#include <ctype.h>
35#include <getopt.h>
36#include <mntent.h>
37#include <pwd.h>
38#include <stdarg.h>
39#include <stdio.h>
40#include <stdlib.h>
41#include <string.h>
42#include <sys/mount.h>
43#include <sys/stat.h>
44#include <sys/types.h>
45#include <unistd.h>
46#include <mntent.h>
47#include <limits.h>
48#include <iconv.h>
49
50#include "vbsfmount.h"
51
52/* Compile-time assertion. If a == 0, we get two identical switch cases, which is not
53 allowed. */
54#define CT_ASSERT(a) \
55 do { \
56 switch(0) { case 0: case (a): ; } \
57 } while (0)
58
59struct opts {
60 int uid;
61 int gid;
62 int ttl;
63 int ronly;
64 int noexec;
65 int nodev;
66 int nosuid;
67 int remount;
68 char nls_name[MAX_NLS_NAME];
69 char *convertcp;
70};
71
72#define PANIC_ATTR __attribute ((noreturn, __format__ (__printf__, 1, 2)))
73
74static void PANIC_ATTR
75panic (const char *fmt, ...)
76{
77 va_list ap;
78
79 va_start (ap, fmt);
80 vfprintf (stderr, fmt, ap);
81 va_end (ap);
82 exit (EXIT_FAILURE);
83}
84
85static void PANIC_ATTR
86panic_err (const char *fmt, ...)
87{
88 va_list ap;
89 int errno_code = errno;
90
91 va_start (ap, fmt);
92 vfprintf (stderr, fmt, ap);
93 va_end (ap);
94 fprintf (stderr, ": %s\n", strerror (errno_code));
95 exit (EXIT_FAILURE);
96}
97
98static int
99safe_atoi (const char *s, size_t size)
100{
101 char *endptr;
102 long long int val = strtoll (s, &endptr, 10);
103
104 if (val < INT_MIN || val > INT_MAX || endptr < s + size) {
105 errno = ERANGE;
106 panic_err ("could not convert %.*s to integer, result = %d",
107 size, s, (int) val);
108 }
109 return (int) val;
110}
111
112static void
113process_mount_opts (const char *s, struct opts *opts)
114{
115 const char *next = s;
116 size_t len;
117 typedef enum handler_opt
118 {
119 HORW,
120 HORO,
121 HOUID,
122 HOGID,
123 HOTTL,
124 HOIOCHARSET,
125 HOCONVERTCP,
126 HONOEXEC,
127 HOEXEC,
128 HONODEV,
129 HODEV,
130 HONOSUID,
131 HOSUID,
132 HOREMOUNT
133 } handler_opt;
134 struct {
135 const char *name;
136 handler_opt opt;
137 int has_arg;
138 const char *desc;
139 } handlers[] = {
140 {"rw", HORW, 0, "mount read write (default)"},
141 {"ro", HORO, 0, "mount read only"},
142 {"uid", HOUID, 1, "default file owner user id"},
143 {"gid", HOGID, 1, "default file owner group id"},
144 {"ttl", HOTTL, 1, "time to live for dentry"},
145 {"iocharset", HOIOCHARSET, 1, "i/o charset (default utf8)"},
146 {"convertcp", HOCONVERTCP, 1, "convert share name from given charset to utf8"},
147 {"noexec", HONOEXEC, 0, 0 }, /* don't document these options directly here */
148 {"exec", HOEXEC, 0, 0 }, /* as they are well known and described in the */
149 {"nodev", HONODEV, 0, 0 }, /* usual manpages */
150 {"dev", HODEV, 0, 0 },
151 {"nosuid", HONOSUID, 0, 0 },
152 {"suid", HOSUID, 0, 0 },
153 {"remount", HOREMOUNT, 0, 0 },
154 {NULL, 0, 0, NULL}
155 }, *handler;
156
157 while (next) {
158 const char *val;
159 size_t key_len, val_len;
160
161 s = next;
162 next = strchr (s, ',');
163 if (!next) {
164 len = strlen (s);
165 }
166 else {
167 len = next - s;
168 next += 1;
169 if (!*next) {
170 next = 0;
171 }
172 }
173
174 val = NULL;
175 val_len = 0;
176 for (key_len = 0; key_len < len; ++key_len) {
177 if (s[key_len] == '=') {
178 if (key_len + 1 < len) {
179 val = s + key_len + 1;
180 val_len = len - key_len - 1;
181 }
182 break;
183 }
184 }
185
186 for (handler = handlers; handler->name; ++handler) {
187 size_t j;
188 for (j = 0; j < key_len && handler->name[j] == s[j]; ++j)
189 ;
190
191 if (j == key_len && !handler->name[j]) {
192 if (handler->has_arg) {
193 if (!(val && *val)) {
194 panic ("%.*s requires an argument (i.e. %.*s=<arg>)\n",
195 len, s, len, s);
196 }
197 }
198
199 switch(handler->opt)
200 {
201 case HORW:
202 opts->ronly = 0;
203 break;
204 case HORO:
205 opts->ronly = 1;
206 break;
207 case HONOEXEC:
208 opts->noexec = 1;
209 break;
210 case HOEXEC:
211 opts->noexec = 0;
212 break;
213 case HONODEV:
214 opts->nodev = 1;
215 break;
216 case HODEV:
217 opts->nodev = 0;
218 break;
219 case HONOSUID:
220 opts->nosuid = 1;
221 break;
222 case HOSUID:
223 opts->nosuid = 0;
224 break;
225 case HOREMOUNT:
226 opts->remount = 1;
227 break;
228 case HOUID:
229 opts->uid = safe_atoi (val, val_len);
230 break;
231 case HOGID:
232 opts->gid = safe_atoi (val, val_len);
233 break;
234 case HOTTL:
235 opts->ttl = safe_atoi (val, val_len);
236 break;
237 case HOIOCHARSET:
238 if (val_len + 1 > sizeof (opts->nls_name)) {
239 panic ("iocharset name too long\n");
240 }
241 memcpy (opts->nls_name, val, val_len);
242 opts->nls_name[val_len] = 0;
243 break;
244 case HOCONVERTCP:
245 opts->convertcp = malloc (val_len + 1);
246 if (!opts->convertcp) {
247 panic_err ("could not allocate memory");
248 }
249 memcpy (opts->convertcp, val, val_len);
250 opts->convertcp[val_len] = 0;
251 break;
252 }
253 break;
254 }
255 continue;
256 }
257
258 if (!handler->name) {
259 fprintf (stderr, "unknown mount option `%.*s'\n", len, s);
260 fprintf (stderr, "valid options:\n");
261
262 for (handler = handlers; handler->name; ++handler) {
263 if (handler->desc)
264 fprintf (stderr, " %-10s%s %s\n", handler->name,
265 handler->has_arg ? "=<arg>" : "", handler->desc);
266 }
267 exit (EXIT_FAILURE);
268 }
269 }
270}
271
272static void
273complete (char *host_name, char *mount_point,
274 unsigned long flags, struct opts *opts)
275{
276 FILE *f, *m;
277 char *buf;
278 size_t size;
279 struct mntent e;
280
281 m = open_memstream (&buf, &size);
282 if (!m)
283 panic_err ("could not update mount table (failed to create memstream)");
284
285 if (opts->uid)
286 fprintf (m, "uid=%d,", opts->uid);
287 if (opts->gid)
288 fprintf (m, "gid=%d,", opts->gid);
289 if (opts->ttl)
290 fprintf (m, "ttl=%d,", opts->ttl);
291 if (*opts->nls_name)
292 fprintf (m, "iocharset=%s,", opts->nls_name);
293 if (flags & MS_NOSUID)
294 fprintf (m, "%s,", MNTOPT_NOSUID);
295 if (flags & MS_RDONLY)
296 fprintf (m, "%s,", MNTOPT_RO);
297 else
298 fprintf (m, "%s,", MNTOPT_RW);
299
300 fclose (m);
301
302 if (size > 0)
303 buf[size - 1] = 0;
304 else
305 buf = "defaults";
306
307 f = setmntent (MOUNTED, "a+");
308 if (!f)
309 panic_err ("could not open mount table for update");
310
311 e.mnt_fsname = host_name;
312 e.mnt_dir = mount_point;
313 e.mnt_type = "vboxsf";
314 e.mnt_opts = buf;
315 e.mnt_freq = 0;
316 e.mnt_passno = 0;
317
318 if (addmntent (f, &e))
319 {
320 if (size > 0)
321 {
322 memset (buf, size, 0);
323 free (buf);
324 }
325 panic_err ("could not add an entry to the mount table");
326 }
327
328 endmntent (f);
329
330 if (size > 0)
331 {
332 memset (buf, size, 0);
333 free (buf);
334 }
335}
336
337static void
338convertcp (char *in_codeset, char *host_name, struct vbsf_mount_info *info)
339{
340 char *i = host_name;
341 char *o = info->name;
342 size_t ib = strlen (host_name);
343 size_t ob = sizeof (info->name) - 1;
344 iconv_t cd;
345
346 cd = iconv_open ("UTF-8", in_codeset);
347 if (cd == (iconv_t) -1) {
348 panic_err ("could not convert share name, iconv_open `%s' failed",
349 in_codeset);
350 }
351
352 while (ib) {
353 size_t c = iconv (cd, &i, &ib, &o, &ob);
354 if (c == (size_t) -1) {
355 panic_err ("could not convert share name(%s) at %d",
356 host_name, strlen (host_name) - ib);
357 }
358 }
359 *o = 0;
360}
361
362
363/**
364 * Print out a usage message and exit.
365 *
366 * @param name The name of the application
367 */
368static void __attribute ((noreturn)) usage(char *name)
369{
370 printf("Usage: %s [OPTIONS] NAME MOUNTPOINT\n"
371 "Mount the VirtualBox shared folder NAME from the host system to MOUNTPOINT.\n"
372 "\n"
373 " -w mount the shared folder writably (the default)\n"
374 " -r mount the shared folder read-only\n"
375 " -n do not add information about the folder to the mtab file\n"
376 " -o OPTION[,OPTION...] use the mount options specified\n"
377 "\n", name);
378 printf("Available mount options are:\n"
379 "\n"
380 " rw mount writably (the default)\n"
381 " ro mount read only\n"
382 " uid=UID set the default file owner user id to UID\n"
383 " gid=GID set the default file owner group id to GID\n"
384 " ttl=TTL set the \"time to live\" to TID for the dentry\n"
385 " iocharset CHARSET use the character set CHARSET for i/o operations (default utf8)\n"
386 " convertcp CHARSET convert the shared folder name from the character set CHARSET to utf8\n");
387 printf("Less common used options:\n"
388 " noexec,exec,nodev,dev,nosuid,suid\n");
389 exit(1);
390}
391
392int
393main (int argc, char **argv)
394{
395 int c;
396 int err;
397 int nomtab = 0;
398 unsigned long flags = MS_NODEV;
399 char *host_name;
400 char *mount_point;
401 struct vbsf_mount_info mntinf;
402 struct opts opts = {
403 0, /* uid */
404 0, /* gid */
405 0, /* ttl */
406 0, /* ronly */
407 0, /* noexec */
408 0, /* nodev */
409 0, /* nosuid */
410 0, /* remount */
411 "\0", /* nls_name */
412 NULL, /* convertcp */
413 };
414
415 if (getuid ())
416 panic ("Only root can mount shared folders from the host.\n");
417
418 if (!argv[0])
419 argv[0] = "mount.vboxsf";
420
421 /* Compile-time assertions */
422 CT_ASSERT(sizeof(uid_t) == sizeof(int));
423 CT_ASSERT(sizeof(gid_t) == sizeof(int));
424
425 while ((c = getopt (argc, argv, "rwno:h")) != -1) {
426 switch (c) {
427 default:
428 fprintf (stderr, "unknown option `%c:%#x'\n", c, c);
429 case '?':
430 case 'h':
431 usage(argv[0]);
432
433 case 'r':
434 opts.ronly = 1;
435 break;
436
437 case 'w':
438 opts.ronly = 0;
439
440 case 'o':
441 process_mount_opts (optarg, &opts);
442 break;
443
444 case 'n':
445 nomtab = 1;
446 break;
447 }
448 }
449
450 if (argc - optind < 2)
451 usage(argv[0]);
452
453 host_name = argv[optind];
454 mount_point = argv[optind + 1];
455
456 if (opts.convertcp)
457 convertcp (opts.convertcp, host_name, &mntinf);
458 else
459 {
460 if (strlen (host_name) > MAX_HOST_NAME - 1)
461 panic ("host name is too big\n");
462
463 strcpy (mntinf.name, host_name);
464 }
465
466 if (strlen (opts.nls_name) > MAX_NLS_NAME - 1)
467 panic ("%s: the character set name for I/O is too long.\n", argv[0]);
468
469 strcpy (mntinf.nls_name, opts.nls_name);
470
471 if (opts.ronly)
472 flags |= MS_RDONLY;
473 if (opts.noexec)
474 flags |= MS_NOEXEC;
475 if (opts.nodev)
476 flags |= MS_NODEV;
477
478 mntinf.uid = opts.uid;
479 mntinf.gid = opts.gid;
480 mntinf.ttl = opts.ttl;
481
482 err = mount (NULL, mount_point, "vboxsf", flags, &mntinf);
483 if (err)
484 panic_err ("%s: mounting failed with the error", argv[0]);
485
486 if (!nomtab)
487 complete (host_name, mount_point, flags, &opts);
488
489 exit (EXIT_SUCCESS);
490}
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