2000-02-18 03:38:33 +08:00
|
|
|
/* mount.cc
|
|
|
|
|
2001-10-25 05:56:54 +08:00
|
|
|
Copyright 1996, 1997, 1998, 1999, 2000, 2001 Red Hat, Inc.
|
2000-02-18 03:38:33 +08:00
|
|
|
|
|
|
|
This file is part of Cygwin.
|
|
|
|
|
|
|
|
This software is a copyrighted work licensed under the terms of the
|
|
|
|
Cygwin license. Please consult the file "CYGWIN_LICENSE" for
|
|
|
|
details. */
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <sys/mount.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <mntent.h>
|
|
|
|
#include <windows.h>
|
|
|
|
#include <sys/cygwin.h>
|
2000-06-06 02:43:54 +08:00
|
|
|
#include <stdlib.h>
|
2001-04-03 10:41:54 +08:00
|
|
|
#include <getopt.h>
|
2000-02-18 03:38:33 +08:00
|
|
|
|
|
|
|
#ifdef errno
|
|
|
|
#undef errno
|
|
|
|
#endif
|
|
|
|
#include <errno.h>
|
|
|
|
|
2001-09-04 06:02:54 +08:00
|
|
|
static void mount_commands (void);
|
2000-02-18 03:38:33 +08:00
|
|
|
static void show_mounts (void);
|
2000-11-09 04:39:45 +08:00
|
|
|
static void show_cygdrive_info (void);
|
2000-02-18 03:38:33 +08:00
|
|
|
static void change_cygdrive_prefix (const char *new_prefix, int flags);
|
|
|
|
static int mount_already_exists (const char *posix_path, int flags);
|
|
|
|
|
|
|
|
// static short create_missing_dirs = FALSE;
|
|
|
|
static short force = FALSE;
|
|
|
|
|
|
|
|
static const char *progname;
|
|
|
|
|
2000-06-08 20:54:12 +08:00
|
|
|
static void
|
|
|
|
error (const char *path)
|
|
|
|
{
|
|
|
|
fprintf (stderr, "%s: %s: %s\n", progname, path,
|
|
|
|
(errno == EMFILE) ? "Too many mount entries" : strerror (errno));
|
|
|
|
exit (1);
|
|
|
|
}
|
|
|
|
|
2000-02-18 03:38:33 +08:00
|
|
|
/* FIXME: do_mount should also print a warning message if the dev arg
|
|
|
|
is a non-existent Win32 path. */
|
|
|
|
|
|
|
|
static void
|
|
|
|
do_mount (const char *dev, const char *where, int flags)
|
|
|
|
{
|
|
|
|
struct stat statbuf;
|
|
|
|
char win32_path[MAX_PATH];
|
|
|
|
int statres;
|
|
|
|
|
|
|
|
cygwin_conv_to_win32_path (where, win32_path);
|
|
|
|
|
|
|
|
statres = stat (win32_path, &statbuf);
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
if (statres == -1)
|
|
|
|
{
|
|
|
|
/* FIXME: this'll fail if mount dir is missing any parent dirs */
|
|
|
|
if (create_missing_dirs == TRUE)
|
|
|
|
{
|
|
|
|
if (mkdir (where, 0755) == -1)
|
|
|
|
fprintf (stderr, "Warning: unable to create %s!\n", where);
|
|
|
|
else
|
|
|
|
statres = 0; /* Pretend stat succeeded if we could mkdir. */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (mount (dev, where, flags))
|
2000-06-08 20:54:12 +08:00
|
|
|
error (where);
|
2000-02-18 03:38:33 +08:00
|
|
|
|
|
|
|
if (statres == -1)
|
|
|
|
{
|
|
|
|
if (force == FALSE)
|
|
|
|
fprintf (stderr, "%s: warning - %s does not exist.\n", progname, where);
|
|
|
|
}
|
|
|
|
else if (!(statbuf.st_mode & S_IFDIR))
|
|
|
|
{
|
|
|
|
if (force == FALSE)
|
2000-07-14 05:49:36 +08:00
|
|
|
fprintf (stderr, "%s: warning: %s is not a directory.\n", progname, where);
|
2000-10-28 13:00:00 +08:00
|
|
|
}
|
2000-02-18 03:38:33 +08:00
|
|
|
|
|
|
|
exit (0);
|
|
|
|
}
|
|
|
|
|
2001-04-03 10:41:54 +08:00
|
|
|
struct option longopts[] =
|
|
|
|
{
|
2001-04-16 10:51:03 +08:00
|
|
|
{"help", no_argument, NULL, 'h' },
|
2001-04-03 10:41:54 +08:00
|
|
|
{"binary", no_argument, NULL, 'b'},
|
|
|
|
{"force", no_argument, NULL, 'f'},
|
|
|
|
{"system", no_argument, NULL, 's'},
|
|
|
|
{"text", no_argument, NULL, 't'},
|
|
|
|
{"user", no_argument, NULL, 'u'},
|
|
|
|
{"executable", no_argument, NULL, 'x'},
|
|
|
|
{"change-cygdrive-prefix", no_argument, NULL, 'c'},
|
|
|
|
{"cygwin-executable", no_argument, NULL, 'X'},
|
|
|
|
{"show-cygdrive-prefix", no_argument, NULL, 'p'},
|
|
|
|
{"import-old-mounts", no_argument, NULL, 'i'},
|
2001-09-04 06:02:54 +08:00
|
|
|
{"mount-commands", no_argument, NULL, 'm'},
|
2001-04-03 10:41:54 +08:00
|
|
|
{NULL, 0, NULL, 0}
|
|
|
|
};
|
|
|
|
|
2001-09-04 06:02:54 +08:00
|
|
|
char opts[] = "hbfstuxXpicm";
|
2001-04-03 10:41:54 +08:00
|
|
|
|
2000-02-18 03:38:33 +08:00
|
|
|
static void
|
|
|
|
usage (void)
|
|
|
|
{
|
2001-04-03 10:41:54 +08:00
|
|
|
fprintf (stderr, "Usage: %s [OPTION] [<win32path> <posixpath>]\n\
|
|
|
|
-b, --binary text files are equivalent to binary files\n\
|
|
|
|
(newline = \\n)\n\
|
|
|
|
-c, --change-cygdrive-prefix change the cygdrive path prefix to <posixpath>\n\
|
|
|
|
-f, --force force mount, don't warn about missing mount\n\
|
2001-04-16 10:51:03 +08:00
|
|
|
point directories\n\
|
2002-02-16 07:03:14 +08:00
|
|
|
-i, --import-old-mounts copy old registry mount table mounts into the\n\
|
|
|
|
current mount areas\n\
|
2001-04-03 10:41:54 +08:00
|
|
|
-p, --show-cygdrive-prefix show user and/or system cygdrive path prefix\n\
|
|
|
|
-s, --system add mount point to system-wide registry location\n\
|
|
|
|
-t, --text (default) text files get \\r\\n line endings\n\
|
|
|
|
-u, --user (default) add mount point to user registry location\n\
|
|
|
|
-x, --executable treat all files under mount point as executables\n\
|
|
|
|
-X, --cygwin-executable treat all files under mount point as cygwin\n\
|
|
|
|
executables\n\
|
2001-09-04 06:02:54 +08:00
|
|
|
-m, --mount-commands write mount commands to replace user and\n\
|
|
|
|
system mount points and cygdrive prefixes\n\
|
2000-02-18 03:38:33 +08:00
|
|
|
", progname);
|
|
|
|
exit (1);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2001-04-03 10:41:54 +08:00
|
|
|
main (int argc, char **argv)
|
2000-02-18 03:38:33 +08:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
int flags = 0;
|
2001-04-03 10:41:54 +08:00
|
|
|
enum do_what
|
|
|
|
{
|
|
|
|
nada,
|
|
|
|
saw_change_cygdrive_prefix,
|
|
|
|
saw_import_old_mounts,
|
2001-09-04 06:02:54 +08:00
|
|
|
saw_show_cygdrive_prefix,
|
|
|
|
saw_mount_commands
|
2001-04-03 10:41:54 +08:00
|
|
|
} do_what = nada;
|
2000-02-18 03:38:33 +08:00
|
|
|
|
|
|
|
progname = argv[0];
|
|
|
|
|
|
|
|
if (argc == 1)
|
|
|
|
{
|
|
|
|
show_mounts ();
|
|
|
|
exit (0);
|
|
|
|
}
|
|
|
|
|
2001-04-03 10:41:54 +08:00
|
|
|
while ((i = getopt_long (argc, argv, opts, longopts, NULL)) != EOF)
|
|
|
|
switch (i)
|
|
|
|
{
|
|
|
|
case 'b':
|
2000-04-27 11:44:23 +08:00
|
|
|
flags |= MOUNT_BINARY;
|
2001-04-03 10:41:54 +08:00
|
|
|
break;
|
2001-04-16 10:51:03 +08:00
|
|
|
case 'c':
|
2001-04-03 10:41:54 +08:00
|
|
|
if (do_what == nada)
|
|
|
|
do_what = saw_change_cygdrive_prefix;
|
|
|
|
else
|
|
|
|
usage ();
|
|
|
|
break;
|
|
|
|
case 'f':
|
|
|
|
force = TRUE;
|
|
|
|
break;
|
|
|
|
case 'i':
|
|
|
|
if (do_what == nada)
|
|
|
|
do_what = saw_import_old_mounts;
|
|
|
|
else
|
|
|
|
usage ();
|
|
|
|
break;
|
|
|
|
case 'p':
|
|
|
|
if (do_what == nada)
|
|
|
|
do_what = saw_show_cygdrive_prefix;
|
|
|
|
else
|
|
|
|
usage ();
|
|
|
|
break;
|
|
|
|
case 's':
|
2000-04-27 11:44:23 +08:00
|
|
|
flags |= MOUNT_SYSTEM;
|
2001-04-03 10:41:54 +08:00
|
|
|
break;
|
|
|
|
case 't':
|
|
|
|
flags &= ~MOUNT_BINARY;
|
|
|
|
break;
|
|
|
|
case 'u':
|
2000-04-27 11:44:23 +08:00
|
|
|
flags &= ~MOUNT_SYSTEM;
|
2001-04-03 10:41:54 +08:00
|
|
|
break;
|
|
|
|
case 'X':
|
|
|
|
flags |= MOUNT_CYGWIN_EXEC;
|
|
|
|
break;
|
|
|
|
case 'x':
|
2000-04-27 11:44:23 +08:00
|
|
|
flags |= MOUNT_EXEC;
|
2001-04-03 10:41:54 +08:00
|
|
|
break;
|
2001-09-04 06:02:54 +08:00
|
|
|
case 'm':
|
|
|
|
if (do_what == nada)
|
|
|
|
do_what = saw_mount_commands;
|
|
|
|
else
|
|
|
|
usage ();
|
|
|
|
break;
|
2001-04-03 10:41:54 +08:00
|
|
|
default:
|
2000-04-27 11:44:23 +08:00
|
|
|
usage ();
|
2001-04-03 10:41:54 +08:00
|
|
|
}
|
2000-10-28 13:00:00 +08:00
|
|
|
|
2001-04-03 10:41:54 +08:00
|
|
|
argc--;
|
|
|
|
switch (do_what)
|
2000-02-18 03:38:33 +08:00
|
|
|
{
|
2001-04-03 10:41:54 +08:00
|
|
|
case saw_change_cygdrive_prefix:
|
|
|
|
if (optind != argc)
|
|
|
|
usage ();
|
|
|
|
change_cygdrive_prefix (argv[optind], flags);
|
|
|
|
break;
|
|
|
|
case saw_import_old_mounts:
|
|
|
|
if (optind <= argc)
|
|
|
|
usage ();
|
|
|
|
else
|
|
|
|
cygwin_internal (CW_READ_V1_MOUNT_TABLES);
|
|
|
|
break;
|
|
|
|
case saw_show_cygdrive_prefix:
|
|
|
|
if (optind <= argc)
|
|
|
|
usage ();
|
|
|
|
show_cygdrive_info ();
|
|
|
|
break;
|
2001-09-04 06:02:54 +08:00
|
|
|
case saw_mount_commands:
|
|
|
|
if (optind <= argc)
|
|
|
|
usage ();
|
|
|
|
mount_commands ();
|
|
|
|
break;
|
2001-04-03 10:41:54 +08:00
|
|
|
default:
|
|
|
|
if (optind != (argc - 1))
|
|
|
|
{
|
2001-08-29 02:48:38 +08:00
|
|
|
if (optind >= argc)
|
|
|
|
fprintf (stderr, "%s: not enough arguments\n", progname);
|
|
|
|
else
|
2001-09-02 03:58:40 +08:00
|
|
|
fprintf (stderr, "%s: too many arguments\n", progname);
|
2001-04-03 10:41:54 +08:00
|
|
|
usage ();
|
|
|
|
}
|
|
|
|
if (force || !mount_already_exists (argv[optind + 1], flags))
|
|
|
|
do_mount (argv[optind], argv[optind + 1], flags);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
errno = EBUSY;
|
|
|
|
error (argv[optind + 1]);
|
|
|
|
}
|
2000-02-18 03:38:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* NOTREACHED */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2001-09-04 06:02:54 +08:00
|
|
|
static void
|
|
|
|
mount_commands (void)
|
|
|
|
{
|
|
|
|
FILE *m = setmntent ("/-not-used-", "r");
|
|
|
|
struct mntent *p;
|
|
|
|
char *c;
|
|
|
|
const char *format_mnt = "mount%s \"%s\" \"%s\"\n";
|
|
|
|
const char *format_cyg = "mount%s --change-cygdrive-prefix \"%s\"\n";
|
|
|
|
char opts[MAX_PATH];
|
|
|
|
char user[MAX_PATH];
|
|
|
|
char system[MAX_PATH];
|
|
|
|
char user_flags[MAX_PATH];
|
|
|
|
char system_flags[MAX_PATH];
|
|
|
|
|
|
|
|
// write mount commands for user and system mount points
|
|
|
|
while ((p = getmntent (m)) != NULL) {
|
2001-09-04 09:09:39 +08:00
|
|
|
// Only list non-cygdrives
|
|
|
|
if (!strstr (p->mnt_opts, ",noumount")) {
|
|
|
|
strcpy(opts, " -f");
|
|
|
|
if (p->mnt_type[0] == 'u')
|
|
|
|
strcat (opts, " -u");
|
|
|
|
else if (p->mnt_type[0] == 's')
|
|
|
|
strcat (opts, " -s");
|
|
|
|
if (p->mnt_opts[0] == 'b')
|
|
|
|
strcat (opts, " -b");
|
|
|
|
else if (p->mnt_opts[0] == 't')
|
|
|
|
strcat (opts, " -t");
|
|
|
|
if (strstr (p->mnt_opts, ",exec"))
|
|
|
|
strcat (opts, " -x");
|
|
|
|
while ((c = strchr (p->mnt_fsname, '\\')) != NULL)
|
|
|
|
*c = '/';
|
|
|
|
printf (format_mnt, opts, p->mnt_fsname, p->mnt_dir);
|
|
|
|
}
|
2001-09-04 06:02:54 +08:00
|
|
|
}
|
|
|
|
endmntent (m);
|
|
|
|
|
|
|
|
// write mount commands for cygdrive prefixes
|
|
|
|
cygwin_internal (CW_GET_CYGDRIVE_INFO, user, system, user_flags,
|
|
|
|
system_flags);
|
|
|
|
if (strlen (user) > 0) {
|
|
|
|
strcpy (opts, " ");
|
|
|
|
if (user_flags[0] == 'b')
|
|
|
|
strcat (opts, " -b");
|
|
|
|
else if (user_flags[0] == 't')
|
|
|
|
strcat (opts, " -t");
|
|
|
|
printf (format_cyg, opts, user);
|
|
|
|
}
|
|
|
|
if (strlen (system) > 0) {
|
|
|
|
strcpy (opts, " -s");
|
|
|
|
if (system_flags[0] == 'b')
|
|
|
|
strcat (opts, " -b");
|
|
|
|
else if (system_flags[0] == 't')
|
|
|
|
strcat (opts, " -t");
|
|
|
|
printf (format_cyg, opts, system);
|
|
|
|
}
|
|
|
|
|
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
|
2000-02-18 03:38:33 +08:00
|
|
|
static void
|
|
|
|
show_mounts (void)
|
|
|
|
{
|
|
|
|
FILE *m = setmntent ("/-not-used-", "r");
|
|
|
|
struct mntent *p;
|
2001-04-02 22:58:47 +08:00
|
|
|
const char *format = "%s on %s type %s (%s)\n";
|
2000-02-18 03:38:33 +08:00
|
|
|
|
2001-04-02 22:58:47 +08:00
|
|
|
// printf (format, "Device", "Directory", "Type", "Flags");
|
2000-02-18 03:38:33 +08:00
|
|
|
while ((p = getmntent (m)) != NULL)
|
2001-04-02 22:58:47 +08:00
|
|
|
printf (format, p->mnt_fsname, p->mnt_dir, p->mnt_type, p->mnt_opts);
|
2000-02-18 03:38:33 +08:00
|
|
|
endmntent (m);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Return 1 if mountpoint from the same registry area is already in
|
|
|
|
mount table. Otherwise return 0. */
|
|
|
|
static int
|
|
|
|
mount_already_exists (const char *posix_path, int flags)
|
|
|
|
{
|
|
|
|
int found_matching = 0;
|
|
|
|
|
|
|
|
FILE *m = setmntent ("/-not-used-", "r");
|
|
|
|
struct mntent *p;
|
|
|
|
|
|
|
|
while ((p = getmntent (m)) != NULL)
|
|
|
|
{
|
|
|
|
/* if the paths match, and they're both the same type of mount. */
|
|
|
|
if (strcmp (p->mnt_dir, posix_path) == 0)
|
|
|
|
{
|
2000-07-14 05:49:36 +08:00
|
|
|
if (p->mnt_type[0] == 'u')
|
2000-02-18 03:38:33 +08:00
|
|
|
{
|
2001-04-16 10:51:03 +08:00
|
|
|
if (!(flags & MOUNT_SYSTEM)) /* both current_user */
|
|
|
|
found_matching = 1;
|
|
|
|
else
|
|
|
|
fprintf (stderr,
|
|
|
|
"%s: warning: system mount point of '%s' "
|
|
|
|
"will always be masked by user mount.\n",
|
|
|
|
progname, posix_path);
|
2000-02-18 03:38:33 +08:00
|
|
|
break;
|
2001-04-16 10:51:03 +08:00
|
|
|
}
|
2000-07-14 05:49:36 +08:00
|
|
|
else if (p->mnt_type[0] == 's')
|
2000-02-18 03:38:33 +08:00
|
|
|
{
|
2001-04-16 10:51:03 +08:00
|
|
|
if (flags & MOUNT_SYSTEM) /* both system */
|
|
|
|
found_matching = 1;
|
|
|
|
else
|
|
|
|
fprintf (stderr,
|
|
|
|
"%s: warning: user mount point of '%s' "
|
|
|
|
"masks system mount.\n",
|
|
|
|
progname, posix_path);
|
2000-02-18 03:38:33 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2000-07-14 05:49:36 +08:00
|
|
|
fprintf (stderr, "%s: warning: couldn't determine mount type.\n", progname);
|
2000-02-18 03:38:33 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
endmntent (m);
|
|
|
|
|
|
|
|
return found_matching;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* change_cygdrive_prefix: Change the cygdrive prefix */
|
|
|
|
static void
|
|
|
|
change_cygdrive_prefix (const char *new_prefix, int flags)
|
|
|
|
{
|
|
|
|
flags |= MOUNT_AUTO;
|
|
|
|
|
|
|
|
if (mount (NULL, new_prefix, flags))
|
2000-06-08 20:54:12 +08:00
|
|
|
error (new_prefix);
|
2000-10-28 13:00:00 +08:00
|
|
|
|
2000-02-18 03:38:33 +08:00
|
|
|
exit (0);
|
|
|
|
}
|
2000-07-29 06:34:24 +08:00
|
|
|
|
2001-04-03 10:41:54 +08:00
|
|
|
/* show_cygdrive_info: Show the user and/or cygdrive info, i.e., prefix and
|
2000-11-09 04:39:45 +08:00
|
|
|
flags.*/
|
2000-07-29 06:34:24 +08:00
|
|
|
static void
|
2000-11-09 04:39:45 +08:00
|
|
|
show_cygdrive_info ()
|
2000-07-29 06:34:24 +08:00
|
|
|
{
|
2000-11-09 04:39:45 +08:00
|
|
|
/* Get the cygdrive info */
|
2000-07-29 06:34:24 +08:00
|
|
|
char user[MAX_PATH];
|
|
|
|
char system[MAX_PATH];
|
2000-11-09 04:39:45 +08:00
|
|
|
char user_flags[MAX_PATH];
|
|
|
|
char system_flags[MAX_PATH];
|
|
|
|
cygwin_internal (CW_GET_CYGDRIVE_INFO, user, system, user_flags,
|
|
|
|
system_flags);
|
2000-07-29 06:34:24 +08:00
|
|
|
|
2001-04-03 10:41:54 +08:00
|
|
|
/* Display the user and system cygdrive path prefix, if necessary
|
2000-07-29 06:34:24 +08:00
|
|
|
(ie, not empty) */
|
2000-11-09 04:39:45 +08:00
|
|
|
const char *format = "%-18s %-11s %s\n";
|
|
|
|
printf (format, "Prefix", "Type", "Flags");
|
2000-07-29 06:34:24 +08:00
|
|
|
if (strlen (user) > 0)
|
2000-11-09 04:39:45 +08:00
|
|
|
printf (format, user, "user", user_flags);
|
2000-07-29 06:34:24 +08:00
|
|
|
if (strlen (system) > 0)
|
2000-11-09 04:39:45 +08:00
|
|
|
printf (format, system, "system", system_flags);
|
2000-07-29 06:34:24 +08:00
|
|
|
|
|
|
|
exit (0);
|
|
|
|
}
|