#include <stdint.h>
#include <mach/machine/boolean.h>
#include <sys/cdefs.h>
#include <sys/param.h>
#define NTFS
#include <sys/mount.h>
#include <sys/stat.h>
#include <sys/sysctl.h>
#include <sys/wait.h>
#include "../ntfs.kextproj/ntfsmount.h"
#include <ctype.h>
#include <err.h>
#include <errno.h>
#include <grp.h>
#include <pwd.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sysexits.h>
#include <unistd.h>
#include "mntopts.h"
#ifndef __dead2
#define __dead2
#endif
static struct mntopt mopts[] = {
MOPT_STDOPTS,
{ NULL }
};
static gid_t a_gid(char *);
static uid_t a_uid(char *);
static mode_t a_mask(char *);
static void usage(void) __dead2;
static int checkLoadable();
static int load_kmod();
int
main(argc, argv)
int argc;
char **argv;
{
struct ntfs_args args;
struct stat sb;
int c, mntflags, set_gid, set_uid, set_mask;
char *dev, *dir, mntpath[MAXPATHLEN];
mntflags = set_gid = set_uid = set_mask = 0;
(void)memset(&args, '\0', sizeof(args));
while ((c = getopt(argc, argv, "asu:g:m:o:")) != -1) {
switch (c) {
case 'u':
args.uid = a_uid(optarg);
set_uid = 1;
break;
case 'g':
args.gid = a_gid(optarg);
set_gid = 1;
break;
case 'm':
args.mode = a_mask(optarg);
set_mask = 1;
break;
case 's':
args.flag |= NTFS_MFLAG_CASE_SENSITIVE;
break;
case 'a':
args.flag |= NTFS_MFLAG_ALLNAMES;
break;
case 'o':
getmntopts(optarg, mopts, &mntflags, 0);
break;
case '?':
default:
usage();
break;
}
}
if (optind + 2 != argc)
usage();
dev = argv[optind];
dir = argv[optind + 1];
(void)checkpath(dir, mntpath);
(void)rmslashes(dev, dev);
args.fspec = dev;
if (!set_gid || !set_uid || !set_mask) {
if (stat(mntpath, &sb) == -1)
err(EX_OSERR, "stat %s", mntpath);
if (!set_uid)
args.uid = sb.st_uid;
if (!set_gid)
args.gid = sb.st_gid;
if (!set_mask)
args.mode = sb.st_mode & (S_IRWXU | S_IRWXG | S_IRWXO);
}
if (checkLoadable())
if (load_kmod())
errx(EX_OSERR, "ntfs filesystem is not available");
if (mount("ntfs", mntpath, mntflags, &args) < 0)
err(EX_OSERR, "%s", dev);
exit (0);
}
gid_t
a_gid(s)
char *s;
{
struct group *gr;
char *gname;
gid_t gid;
if ((gr = getgrnam(s)) != NULL)
gid = gr->gr_gid;
else {
for (gname = s; *s && isdigit(*s); ++s);
if (!*s)
gid = atoi(gname);
else
errx(EX_NOUSER, "unknown group id: %s", gname);
}
return (gid);
}
uid_t
a_uid(s)
char *s;
{
struct passwd *pw;
char *uname;
uid_t uid;
if ((pw = getpwnam(s)) != NULL)
uid = pw->pw_uid;
else {
for (uname = s; *s && isdigit(*s); ++s);
if (!*s)
uid = atoi(uname);
else
errx(EX_NOUSER, "unknown user id: %s", uname);
}
return (uid);
}
mode_t
a_mask(s)
char *s;
{
int done, rv=0;
char *ep;
done = 0;
if (*s >= '0' && *s <= '7') {
done = 1;
rv = strtol(optarg, &ep, 8);
}
if (!done || rv < 0 || *ep)
errx(EX_USAGE, "invalid file mode: %s", s);
return (rv);
}
void
usage()
{
fprintf(stderr, "usage: mount_ntfs [-a] [-s] [-u user] [-g group] [-m mask] bdev dir\n");
exit(EX_USAGE);
}
#define FS_TYPE "ntfs"
static int checkLoadable(void)
{
int error;
struct vfsconf vfc;
error = getvfsbyname(FS_TYPE, &vfc);
return error;
}
#define LOAD_COMMAND "/sbin/kextload"
#define MODULE_PATH "/System/Library/Extensions/ntfs.kext"
static int load_kmod(void)
{
int pid;
int result = -1;
union wait status;
pid = fork();
if (pid == 0) {
result = execl(LOAD_COMMAND, LOAD_COMMAND, MODULE_PATH,NULL);
goto Err_Exit;
}
if (pid == -1) {
result = errno;
goto Err_Exit;
}
if ((wait4(pid, (int *)&status, 0, NULL) == pid) && (WIFEXITED(status))) {
result = status.w_retcode;
}
else {
result = -1;
}
Err_Exit:
return (result);
}