dmake: do not set MAKEFLAGS=k
[unleashed/tickless.git] / usr / src / cmd / fs.d / autofs / automount.c
blob0a2ebcc9c5c0dc4f0443c7c4289dfb659775021d
1 /*
2 * CDDL HEADER START
4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License (the "License").
6 * You may not use this file except in compliance with the License.
8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 * or http://www.opensolaris.org/os/licensing.
10 * See the License for the specific language governing permissions
11 * and limitations under the License.
13 * When distributing Covered Code, include this CDDL HEADER in each
14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 * If applicable, add the following below this CDDL HEADER, with the
16 * fields enclosed by brackets "[]" replaced with your own identifying
17 * information: Portions Copyright [yyyy] [name of copyright owner]
19 * CDDL HEADER END
22 * automount.c
24 * Copyright (c) 2010, Oracle and/or its affiliates. All rights reserved.
28 #include <ctype.h>
29 #include <stdio.h>
30 #include <unistd.h>
31 #include <stdlib.h>
32 #include <locale.h>
33 #include <stdarg.h>
34 #include <errno.h>
35 #include <string.h>
36 #include <dirent.h>
37 #include <signal.h>
38 #include <syslog.h>
39 #include <libshare.h>
40 #include <libscf.h>
41 #include <sys/param.h>
42 #include <sys/time.h>
43 #include <sys/vfs.h>
44 #include <sys/types.h>
45 #include <sys/stat.h>
46 #include <sys/mnttab.h>
47 #include <sys/mntent.h>
48 #include <sys/mount.h>
49 #include <sys/utsname.h>
50 #include <sys/tiuser.h>
51 #include <rpc/rpc.h>
52 #include <rpcsvc/nfs_prot.h>
53 #include <nsswitch.h>
54 #include <deflt.h>
55 #include <rpcsvc/daemon_utils.h>
56 #include "automount.h"
57 #include "smfcfg.h"
59 static int mkdir_r(char *);
60 struct autodir *dir_head;
61 struct autodir *dir_tail;
62 static struct extmnttab *find_mount();
63 int verbose = 0;
64 int trace = 0;
66 static void usage();
67 static int compare_opts(char *, char *);
68 static void do_unmounts();
70 static int mount_timeout = AUTOFS_MOUNT_TIMEOUT;
72 static char *service_list[] = { AUTOMOUNTD, NULL };
75 * XXX
76 * The following are needed because they're used in auto_subr.c and
77 * we link with it. Should avoid this.
79 mutex_t cleanup_lock;
80 cond_t cleanup_start_cv;
81 cond_t cleanup_done_cv;
83 int
84 main(int argc, char *argv[])
86 int c;
87 struct autofs_args ai;
88 struct utsname utsname;
89 char autofs_addr[MAXADDRLEN];
90 struct autodir *dir, *d;
91 struct stat stbuf;
92 char *master_map = "auto_master";
93 int null;
94 struct extmnttab mnt, *mntp;
95 struct mnttab *omntp;
96 char mntopts[MAX_MNTOPT_STR];
97 int mntflgs;
98 int count = 0;
99 char *stack[STACKSIZ];
100 char **stkptr;
101 char *defval;
102 struct sigaction sigintact;
103 int ret = 0, bufsz = 0;
104 char valbuf[6];
107 * protect this command from session termination when run in background
108 * we test background by whether SIGINT is ignored
110 (void) sigaction(SIGINT, NULL, &sigintact);
111 if (sigintact.sa_sigaction == SIG_IGN) {
112 (void) signal(SIGHUP, SIG_IGN);
113 (void) setsid();
117 * Read in the values from SMF first before we check
118 * commandline options so the options override the SMF values.
120 bufsz = 6;
121 ret = autofs_smf_get_prop("timeout", valbuf, DEFAULT_INSTANCE,
122 SCF_TYPE_INTEGER, AUTOMOUNTD, &bufsz);
123 if (ret == SA_OK)
125 * Ignore errno. In event of failure, mount_timeout is
126 * already initialized to the correct value.
128 mount_timeout = strtol(valbuf, (char **)NULL, 10);
130 bufsz = 6;
131 ret = autofs_smf_get_prop("automount_verbose", valbuf, DEFAULT_INSTANCE,
132 SCF_TYPE_BOOLEAN, AUTOMOUNTD, &bufsz);
133 if (ret == SA_OK) {
134 if (strncasecmp("true", valbuf, 4) == 0)
135 verbose = TRUE;
138 put_automountd_env();
140 while ((c = getopt(argc, argv, "mM:D:f:t:v?")) != EOF) {
141 switch (c) {
142 case 'm':
143 pr_msg("Warning: -m option not supported");
144 break;
145 case 'M':
146 pr_msg("Warning: -M option not supported");
147 break;
148 case 'D':
149 pr_msg("Warning: -D option not supported");
150 break;
151 case 'f':
152 pr_msg("Error: -f option no longer supported");
153 usage();
154 break;
155 case 't':
156 if (strchr(optarg, '=')) {
157 pr_msg("Error: invalid value for -t");
158 usage();
160 mount_timeout = atoi(optarg);
161 break;
162 case 'v':
163 verbose++;
164 break;
165 default:
166 usage();
167 break;
171 if (optind < argc) {
172 pr_msg("%s: command line mountpoints/maps "
173 "no longer supported", argv[optind]);
174 usage();
177 current_mounts = getmntlist();
178 if (current_mounts == NULL) {
179 pr_msg("Couldn't establish current mounts");
180 exit(1);
183 (void) umask(0);
184 ns_setup(stack, &stkptr);
186 openlog("automount", LOG_PID, LOG_DAEMON);
187 (void) loadmaster_map(master_map, "", stack, &stkptr);
188 if (dir_head != NULL) {
190 * automount maps found. enable services as needed.
192 _check_services(service_list);
195 closelog();
197 if (uname(&utsname) < 0) {
198 pr_msg("uname: %m");
199 exit(1);
201 (void) strcpy(autofs_addr, utsname.nodename);
202 (void) strcat(autofs_addr, ".autofs");
203 ai.addr.buf = autofs_addr;
204 ai.addr.len = strlen(ai.addr.buf);
205 ai.addr.maxlen = ai.addr.len;
207 ai.mount_to = mount_timeout;
208 ai.rpc_to = AUTOFS_RPC_TIMEOUT;
211 * Mount the daemon at its mount points.
213 for (dir = dir_head; dir; dir = dir->dir_next) {
216 * Skip null entries
218 if (strcmp(dir->dir_map, "-null") == 0)
219 continue;
222 * Skip null'ed entries
224 null = 0;
225 for (d = dir->dir_prev; d; d = d->dir_prev) {
226 if (strcmp(dir->dir_name, d->dir_name) == 0)
227 null = 1;
229 if (null)
230 continue;
233 * Check whether there's already an entry
234 * in the mnttab for this mountpoint.
236 if (mntp = find_mount(dir->dir_name, 1)) {
238 * If it's not an autofs mount - don't
239 * mount over it.
241 if (strcmp(mntp->mnt_fstype, MNTTYPE_AUTOFS) != 0) {
242 pr_msg("%s: already mounted",
243 mntp->mnt_mountp);
244 continue;
248 * Compare the mnttab entry with the master map
249 * entry. If the map or mount options are
250 * different, then update this information
251 * with a remount.
253 if (strcmp(mntp->mnt_special, dir->dir_map) == 0 &&
254 compare_opts(dir->dir_opts,
255 mntp->mnt_mntopts) == 0) {
256 continue; /* no change */
260 * Check for an overlaid direct autofs mount.
261 * Cannot remount since it's inaccessible.
263 omntp = (struct mnttab *)mntp;
264 if (hasmntopt(omntp, "direct") != NULL) {
265 mntp = find_mount(dir->dir_name, 0);
266 omntp = (struct mnttab *)mntp;
267 if (hasmntopt(omntp, "direct") == NULL) {
268 if (verbose)
269 pr_msg("%s: cannot remount",
270 dir->dir_name);
271 continue;
275 dir->dir_remount = 1;
279 * Create a mount point if necessary
280 * If the path refers to an existing symbolic
281 * link, refuse to mount on it. This avoids
282 * future problems.
284 if (lstat(dir->dir_name, &stbuf) == 0) {
285 if ((stbuf.st_mode & S_IFMT) != S_IFDIR) {
286 pr_msg("%s: Not a directory", dir->dir_name);
287 continue;
289 } else {
290 if (mkdir_r(dir->dir_name)) {
291 pr_msg("%s: %m", dir->dir_name);
292 continue;
296 ai.path = dir->dir_name;
297 ai.opts = dir->dir_opts;
298 ai.map = dir->dir_map;
299 ai.subdir = "";
300 ai.direct = dir->dir_direct;
301 if (dir->dir_direct)
302 ai.key = dir->dir_name;
303 else
304 ai.key = "";
306 (void) sprintf(mntopts, "ignore,%s",
307 dir->dir_direct ? "direct" : "indirect");
308 if (dir->dir_opts && *dir->dir_opts) {
309 (void) strcat(mntopts, ",");
310 (void) strcat(mntopts, dir->dir_opts);
312 mntflgs = MS_OPTIONSTR | (dir->dir_remount ? MS_REMOUNT : 0);
313 if (mount(dir->dir_map, dir->dir_name, MS_DATA | mntflgs,
314 MNTTYPE_AUTOFS, &ai, sizeof (ai), mntopts,
315 MAX_MNTOPT_STR) < 0) {
316 pr_msg("mount %s: %m", dir->dir_name);
317 continue;
320 count++;
322 if (verbose) {
323 if (dir->dir_remount)
324 pr_msg("%s remounted", dir->dir_name);
325 else
326 pr_msg("%s mounted", dir->dir_name);
330 if (verbose && count == 0)
331 pr_msg("no mounts");
334 * Now compare the /etc/mnttab with the master
335 * map. Any autofs mounts in the /etc/mnttab
336 * that are not in the master map must be
337 * unmounted
339 do_unmounts();
341 return (0);
345 * Find a mount entry given
346 * the mountpoint path.
347 * Optionally return the first
348 * or last entry.
350 static struct extmnttab *
351 find_mount(mntpnt, first)
352 char *mntpnt;
353 int first;
355 struct mntlist *mntl;
356 struct extmnttab *found = NULL;
358 for (mntl = current_mounts; mntl; mntl = mntl->mntl_next) {
360 if (strcmp(mntpnt, mntl->mntl_mnt->mnt_mountp) == 0) {
361 found = mntl->mntl_mnt;
362 if (first)
363 break;
367 return (found);
370 static char *ignore_opts[] = {"ignore", "direct", "indirect", "dev", NULL};
373 * Compare mount options
374 * ignoring "ignore", "direct", "indirect"
375 * and "dev=".
377 static int
378 compare_opts(opts, mntopts)
379 char *opts, *mntopts;
381 char optbuf1[MAX_MNTOPT_STR], *s = optbuf1;
382 char optbuf2[MAX_MNTOPT_STR];
383 char **opttbl1, **opttbl2;
384 int nopts1, nopts2;
385 char *ostart, *optr, *valp;
386 int j, i, notsame;
388 opttbl1 = opttbl2 = NULL;
390 * Parse the two option strings to split them both into
391 * lists of individual options.
393 if (mntopts != NULL)
394 (void) strcpy(s, mntopts);
395 else
396 *s = '\0';
397 if (*s != '\0')
398 nopts1 = 1;
399 else
400 nopts1 = 0;
401 for (s = strchr(s, ','); s != NULL; s = strchr(s, ',')) {
402 nopts1++;
403 s++;
405 if (nopts1)
406 if ((opttbl1 = memalign(sizeof (char *),
407 nopts1 * sizeof (char *))) == NULL)
408 return (1);
409 nopts1 = 0;
410 s = optbuf1;
411 for (ostart = optr = s; *optr != '\0'; ostart = optr) {
412 if (getsubopt(&optr, ignore_opts, &valp) == -1) {
413 opttbl1[nopts1++] = ostart;
416 s = optbuf2;
417 if (opts != NULL)
418 (void) strcpy(s, opts);
419 else
420 *s = '\0';
421 if (*s != '\0')
422 nopts2 = 1;
423 else
424 nopts2 = 0;
425 for (s = strchr(s, ','); s != NULL; s = strchr(s, ',')) {
426 nopts2++;
427 s++;
429 if (nopts2)
430 if ((opttbl2 = memalign(sizeof (char *),
431 nopts2 * sizeof (char *))) == NULL) {
432 notsame = 1;
433 goto done;
435 nopts2 = 0;
436 s = optbuf2;
437 for (ostart = optr = s; *optr != '\0'; ostart = optr) {
438 if (getsubopt(&optr, ignore_opts, &valp) == -1) {
439 opttbl2[nopts2++] = ostart;
442 if (nopts2 != nopts1) {
443 notsame = 1;
444 goto done;
446 notsame = 0;
447 for (i = 0; i < nopts1; i++) {
448 notsame = 1;
449 for (j = 0; j < nopts2; j++) {
450 if (strcmp(opttbl1[i], opttbl2[j]) == 0) {
451 notsame = 0;
452 break;
455 if (notsame)
456 break;
459 done:
460 free(opttbl1);
461 free(opttbl2);
462 return (notsame);
465 static void
466 usage()
468 pr_msg("Usage: automount [ -v ] [ -t duration ]");
469 exit(1);
470 /* NOTREACHED */
474 * Unmount any autofs mounts that
475 * aren't in the master map
477 static void
478 do_unmounts()
480 struct mntlist *mntl;
481 struct extmnttab *mnt;
482 struct mnttab *omnt;
483 struct autodir *dir;
484 int current;
485 int count = 0;
486 struct zone_summary *zsp;
488 zsp = fs_get_zone_summaries();
489 if (zsp == NULL) {
490 pr_msg("Couldn't establish active zones");
491 exit(1);
493 for (mntl = current_mounts; mntl; mntl = mntl->mntl_next) {
494 mnt = mntl->mntl_mnt;
495 omnt = (struct mnttab *)mnt;
496 if (strcmp(mnt->mnt_fstype, MNTTYPE_AUTOFS) != 0)
497 continue;
498 if (fs_mount_in_other_zone(zsp, mnt->mnt_mountp))
499 continue;
501 * Don't unmount autofs mounts done
502 * from the autofs mount command.
503 * How do we tell them apart ?
504 * Autofs mounts not eligible for auto-unmount
505 * have the "nest" pseudo-option.
507 if (hasmntopt(omnt, "nest") != NULL)
508 continue;
510 current = 0;
511 for (dir = dir_head; dir; dir = dir->dir_next) {
512 if (strcmp(dir->dir_name, mnt->mnt_mountp) == 0) {
513 current = strcmp(dir->dir_map, "-null");
514 break;
517 if (current)
518 continue;
521 if (umount(mnt->mnt_mountp) == 0) {
522 if (verbose) {
523 pr_msg("%s unmounted",
524 mnt->mnt_mountp);
526 count++;
529 if (verbose && count == 0)
530 pr_msg("no unmounts");
533 static int
534 mkdir_r(dir)
535 char *dir;
537 int err;
538 char *slash;
540 if (mkdir(dir, 0555) == 0 || errno == EEXIST)
541 return (0);
542 if (errno != ENOENT)
543 return (-1);
544 slash = strrchr(dir, '/');
545 if (slash == NULL)
546 return (-1);
547 *slash = '\0';
548 err = mkdir_r(dir);
549 *slash++ = '/';
550 if (err || !*slash)
551 return (err);
552 return (mkdir(dir, 0555));
556 * Print an error.
557 * Works like printf (fmt string and variable args)
558 * except that it will subsititute an error message
559 * for a "%m" string (like syslog).
561 /* VARARGS1 */
562 void
563 pr_msg(const char *fmt, ...)
565 va_list ap;
566 char buf[BUFSIZ], *p2;
567 char *p1;
568 char *nfmt;
570 (void) strcpy(buf, "automount: ");
571 p2 = buf + strlen(buf);
573 nfmt = gettext(fmt);
575 for (p1 = nfmt; *p1; p1++) {
576 if (*p1 == '%' && *(p1+1) == 'm') {
577 (void) strcpy(p2, strerror(errno));
578 p2 += strlen(p2);
579 p1++;
580 } else {
581 *p2++ = *p1;
584 if (p2 > buf && *(p2-1) != '\n')
585 *p2++ = '\n';
586 *p2 = '\0';
588 va_start(ap, fmt);
589 (void) vfprintf(stderr, buf, ap);
590 va_end(ap);