/* $NetBSD: target.c,v 1.44 2003/10/19 20:17:32 dsl Exp $ */ /* * Copyright 1997 Jonathan Stone * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. All advertising materials mentioning features or use of this software * must display the following acknowledgement: * This product includes software developed for the NetBSD Project by * Jonathan Stone. * 4. The name of Jonathan Stone may not be used to endorse * or promote products derived from this software without specific prior * written permission. * * THIS SOFTWARE IS PROVIDED BY JONATHAN STONE ``AS IS'' * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL PIERMONT INFORMATION SYSTEMS INC. BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF * THE POSSIBILITY OF SUCH DAMAGE. * */ /* Copyright below applies to the realpath() code */ /* * Copyright (c) 1989, 1991, 1993, 1995 * The Regents of the University of California. All rights reserved. * * This code is derived from software contributed to Berkeley by * Jan-Simon Pendry. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. Neither the name of the University nor the names of its contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ #include #if defined(LIBC_SCCS) && !defined(lint) __RCSID("$NetBSD: target.c,v 1.44 2003/10/19 20:17:32 dsl Exp $"); #endif /* * target.c -- path-prefixing routines to access the target installation * filesystems. Makes the install tools more independent of whether * we're installing into a separate filesystem hierarchy mounted under * /targetroot, or into the currently active root mounted on /. */ #include /* XXX vm_param.h always defines TRUE*/ #include #include #include /* stat() */ #include /* statfs() */ #include #include #include #include #include /* defines TRUE, but checks */ #include #include "defs.h" #include "md.h" #include "msg_defs.h" #include "menu_defs.h" /* * local prototypes */ static void make_prefixed_dir (const char *prefix, const char *path); static int do_target_chdir (const char *dir, int flag); int target_test(unsigned int mode, const char *path); int target_test_dir (const char *path); /* deprecated */ int target_test_file (const char *path); /* deprecated */ int target_test_symlink (const char *path); /* deprecated */ void backtowin(void); void unwind_mounts (void); int mount_with_unwind (const char *fstype, const char *from, const char *on); /* Record a mount for later unwinding of target mounts. */ struct unwind_mount { struct unwind_mount *um_prev; char um_mountpoint[STRSIZE]; }; /* Unwind-mount stack */ struct unwind_mount *unwind_mountlist = NULL; /* * Debugging options */ /*#define DEBUG_ROOT*/ /* turn on what-is-root? debugging. */ /*#define DEBUG_UNWIND*/ /* turn on unwind-target-mount debugging. */ /* * debugging helper. curses... */ #if defined(DEBUG) || defined(DEBUG_ROOT) void backtowin(void) { fflush(stdout); /* curses does not leave stdout linebuffered. */ getchar(); /* wait for user to press return */ wrefresh(stdscr); } #endif /* * Is the root partition we're running from the same as the root * which the user has selected to install/upgrade? * Uses global variable "diskdev" to find the selected device for * install/upgrade. */ int target_already_root(void) { if (strcmp(diskdev, "") == 0) /* No root partition was ever selected. * Assume that the currently mounted one should be used */ return 1; return is_active_rootpart(diskdev, rootpart); } /* * Is this device partition (e.g., "sd0a") mounted as root? */ int is_active_rootpart(const char *dev, int ptn) { int mib[2]; char rootdev[SSTRSIZE]; int rootptn; size_t varlen; mib[0] = CTL_KERN; mib[1] = KERN_ROOT_DEVICE; varlen = sizeof(rootdev); if (sysctl(mib, 2, rootdev, &varlen, NULL, 0) < 0) return 1; if (strcmp(dev, rootdev) != 0) return 0; mib[1] = KERN_ROOT_PARTITION; varlen = sizeof rootptn; if (sysctl(mib, 2, &rootptn, &varlen, NULL, 0) < 0) return 1; return ptn == rootptn; } /* * Pathname prefixing glue to support installation either * from in-ramdisk miniroots or on-disk diskimages. * If our root is on the target disk, the install target is mounted * on /targetroot and we need to prefix installed pathnames with /targetroot. * otherwise we are installing to the currently-active root and * no prefix is needed. */ const char * target_prefix(void) { /* * XXX fetch sysctl variable for current root, and compare * to the devicename of the install target disk. */ return(target_already_root() ? "" : targetroot_mnt); } /* * concatenate two pathnames. * XXX returns either input args or result in a static buffer. * The caller must copy if it wants to use the pathname past the * next call to a target-prefixing function, or to modify the inputs.. * Used only internally so this is probably safe. */ const char * concat_paths(const char *prefix, const char *suffix) { static char real_path[MAXPATHLEN]; /* absolute prefix and null suffix? */ if (prefix[0] == '/' && suffix[0] == 0) return prefix; /* null prefix and absolute suffix? */ if (prefix[0] == 0 && suffix[0] == '/') return suffix; /* avoid "//" */ if (suffix[0] == '/' || suffix[0] == 0) snprintf(real_path, sizeof(real_path), "%s%s", prefix, suffix); else snprintf(real_path, sizeof(real_path), "%s/%s", prefix, suffix); return (real_path); } /* * Do target prefix expansion on a pathname. * XXX uses concat_paths and so returns result in a static buffer. * The caller must copy if it wants to use the pathname past the * next call to a target-prefixing function, or to modify the inputs.. * Used only internally so this is probably safe. * * Not static so other functions can generate target related file names. */ const char * target_expand(const char *tgtpath) { return concat_paths(target_prefix(), tgtpath); } /* Make a directory, with a prefix like "/targetroot" or possibly just "". */ static void make_prefixed_dir(const char *prefix, const char *path) { run_prog(0, NULL, "/bin/mkdir -p %s", concat_paths(prefix, path)); } /* Make a directory with a pathname relative to the installation target. */ void make_target_dir(const char *path) { make_prefixed_dir(target_prefix(), path); } #ifdef notdef /* Make a directory with a pathname in the currently-mounted root. */ void make_ramdisk_dir(const char *path) { make_prefixed_dir(path, ""); } /* unused, will not work with new run.c */ /* * * Append |string| to the filename |path|, where |path| is * relative to the root of the install target. * for example, * echo_to_target_file( "Newbie.NetBSD.ORG", "/etc/myname"); * would set the default hostname at the next reboot of the installed-on disk. */ void append_to_target_file(const char *path, const char *string) { run_prog(RUN_FATAL, NULL, "echo %s >> %s", string, target_expand(path)); } /* * As append_to_target_file, but with ftrunc semantics. */ void echo_to_target_file(const char *path, const char *string) { trunc_target_file(path); append_to_target_file(path, string); } void sprintf_to_target_file(const char *path, const char *format, ...) { char lines[STRSIZE]; va_list ap; trunc_target_file(path); va_start(ap, format); vsnprintf(lines, STRSIZE, format, ap); va_end(ap); append_to_target_file(path, lines); } void trunc_target_file(const char *path) { run_prog(RUN_FATAL, NULL, "cat < /dev/null > %s", target_expand(path)); } #endif /* if 0 */ static int do_target_chdir(const char *dir, int must_succeed) { const char *tgt_dir; int error; error = 0; tgt_dir = target_expand(dir); #ifndef DEBUG /* chdir returns -1 on error and sets errno. */ if (chdir(tgt_dir) < 0) error = errno; if (logging) { fprintf(logfp, "cd to %s\n", tgt_dir); fflush(logfp); } if (scripting) { scripting_fprintf(NULL, "cd %s\n", tgt_dir); fflush(script); } if (error && must_succeed) { fprintf(stderr, msg_string(MSG_realdir), target_prefix(), strerror(error)); if (logging) fprintf(logfp, msg_string(MSG_realdir), target_prefix(), strerror(error)); exit(1); } return (error); #else printf("target_chdir (%s)\n", tgt_dir); return (0); #endif } void target_chdir_or_die(const char *dir) { (void)do_target_chdir(dir, 1); } #ifdef notdef int target_chdir(const char *dir) { return do_target_chdir(dir, 0); } #endif /* * Copy a file from the current root into the target system, * where the destination pathname is relative to the target root. * Does not check for copy-to-self when target is current root. */ int cp_to_target(const char *srcpath, const char *tgt_path) { const char *real_path = target_expand(tgt_path); return run_prog(0, NULL, "/bin/cp %s %s", srcpath, real_path); } /* * Duplicate a file from the current root to the same pathname * in the target system. Pathname must be an absolute pathname. * If we're running in the target, do nothing. */ void dup_file_into_target(const char *filename) { if (!target_already_root()) cp_to_target(filename, filename); } /* * Do a mv where both pathnames are within the target filesystem. */ void mv_within_target_or_die(const char *frompath, const char *topath) { char realfrom[STRSIZE]; char realto[STRSIZE]; strncpy(realfrom, target_expand(frompath), STRSIZE); strncpy(realto, target_expand(topath), STRSIZE); run_prog(RUN_FATAL, NULL, "mv %s %s", realfrom, realto); } /* Do a cp where both pathnames are within the target filesystem. */ int cp_within_target(const char *frompath, const char *topath) { char realfrom[STRSIZE]; char realto[STRSIZE]; strncpy(realfrom, target_expand(frompath), STRSIZE); strncpy(realto, target_expand(topath), STRSIZE); return (run_prog(0, NULL, "cp -p %s %s", realfrom, realto)); } /* fopen a pathname in the target. */ FILE * target_fopen(const char *filename, const char *type) { return fopen(target_expand(filename), type); } /* * Do a mount and record the mountpoint in a list of mounts to * unwind after completing or aborting a mount. */ int mount_with_unwind(const char *opts, const char *from, const char *on) { int error; struct unwind_mount * m; m = malloc(sizeof(*m)); if (m == 0) return (ENOMEM); /* XXX */ strncpy(m->um_mountpoint, on, STRSIZE); m->um_prev = unwind_mountlist; unwind_mountlist = m; #ifdef DEBUG_UNWIND endwin(); fprintf(stderr, "mounting %s with unwind\n", on); backtowin(); #endif error = run_prog(RUN_PROGRESS, NULL, "/sbin/mount %s %s %s", opts, from, on); return (error); } /* * unwind the mount stack, unmounting mounted filesystems. * For now, ignore any errors in unmount. * (Why would we be unable to unmount? The user has suspended * us and forked shell sitting somewhere in the target root?) */ void unwind_mounts(void) { struct unwind_mount *m; volatile static int unwind_in_progress = 0; /* signal safety */ if (unwind_in_progress) return; unwind_in_progress = 1; for (m = unwind_mountlist; m; ) { struct unwind_mount *prev; #ifdef DEBUG_UNWIND endwin(); fprintf(stderr, "unmounting %s\n", m->um_mountpoint); backtowin(); #endif run_prog(0, NULL, "/sbin/umount %s", m->um_mountpoint); prev = m->um_prev; free(m); m = prev; } unwind_mountlist = NULL; unwind_in_progress = 0; } /* * Do a mount onto a mountpoint in the install target. * NB: does not prefix mount-from, which probably breaks nullfs mounts. */ int target_mount(const char *fstype, const char *from, const char *on) { int error; const char *realmount = target_expand(on); /* mount and record for unmounting when done. */ error = mount_with_unwind(fstype, from, realmount); return (error); } int target_collect_file(int kind, char **buffer, const char *name) { const char *realname = target_expand(name); #ifdef DEBUG printf("collect real name %s\n", realname); #endif return collect(kind, buffer, realname); } /* * Verify a pathname already exists in the target root filesystem, * by running test "testflag" on the expanded target pathname. */ int target_test(unsigned int mode, const char *path) { const char *real_path = target_expand(path); register int result; result = !file_mode_match(real_path, mode); scripting_fprintf(NULL, "if [ $? != 0 ]; then echo \"%s does not exist!\"; fi\n", real_path); #if defined(DEBUG) printf("target_test(%s %s) returning %d\n", test, real_path, result); #endif return (result); } /* * Verify a directory already exists in the target root * filesystem. Do not create the directory if it doesn't exist. * Assumes that sysinst has already mounted the target root. */ int target_test_dir(const char *path) { return target_test(S_IFDIR, path); } /* * Verify an ordinary file already exists in the target root * filesystem. Do not create the directory if it doesn't exist. * Assumes that sysinst has already mounted the target root. */ int target_test_file(const char *path) { return target_test(S_IFREG, path); } int target_test_symlink(const char *path) { return target_test(S_IFLNK, path); } int target_file_exists_p(const char *path) { return (target_test_file(path) == 0); } int target_dir_exists_p(const char *path) { return (target_test_dir(path) == 0); } int target_symlink_exists_p(const char *path) { return (target_test_symlink(path) == 0); } /* * XXXX had to include this to deal with symlinks in some places. * When the target * disk is mounted under /targetroot, absolute symlinks * on it don't work right. * This function will resolve them using the mountpoint as prefix. * Copied verbatim from libc, with added prefix handling. * * char *realpath(const char *path, char resolved_path[MAXPATHLEN]); * * Find the real name of path, by removing all ".", ".." and symlink * components. Returns (resolved) on success, or (NULL) on failure, * in which case the path which caused trouble is left in (resolved). */ char * target_realpath(const char *path, char *resolved) { struct stat sb; int fd, n, rootd, serrno, nlnk = 0; char *p, *q, wbuf[MAXPATHLEN]; char solidus[] = "/", empty[] = ""; /* Save the starting point. */ if ((fd = open(".", O_RDONLY)) < 0) { (void)strlcpy(resolved, ".", MAXPATHLEN); return (NULL); } /* * Find the dirname and basename from the path to be resolved. * Change directory to the dirname component. * lstat the basename part. * if it is a symlink, read in the value and loop. * if it is a directory, then change to that directory. * get the current directory name and append the basename. */ if (target_prefix() != NULL && strcmp(target_prefix(), "") != 0) snprintf(resolved, MAXPATHLEN, "%s/%s", target_prefix(), path); else if (strlcpy(resolved, path, MAXPATHLEN) >= MAXPATHLEN) { errno = ENAMETOOLONG; goto err1; } loop: q = strrchr(resolved, '/'); if (q != NULL) { p = q + 1; if (q == resolved) q = solidus; else { do { --q; } while (q > resolved && *q == '/'); q[1] = '\0'; q = resolved; } if (chdir(q) < 0) goto err1; } else p = resolved; /* Deal with the last component. */ if (lstat(p, &sb) == 0) { if (S_ISLNK(sb.st_mode)) { if (nlnk++ >= MAXSYMLINKS) { errno = ELOOP; goto err1; } n = readlink(p, wbuf, MAXPATHLEN - 1); if (n < 0) goto err1; wbuf[n] = '\0'; if (wbuf[0] == '/') snprintf(resolved, MAXPATHLEN, "%s%s", target_prefix(), wbuf); else strlcpy(resolved, wbuf, MAXPATHLEN); goto loop; } if (S_ISDIR(sb.st_mode)) { if (chdir(p) < 0) goto err1; p = empty; } } /* * Save the last component name and get the full pathname of * the current directory. */ if (strlcpy(wbuf, p, sizeof(wbuf)) >= sizeof(wbuf)) { errno = ENAMETOOLONG; goto err1; } /* * Call the internal internal version of getcwd which * does a physical search rather than using the $PWD short-cut */ if (getcwd(resolved, MAXPATHLEN) == 0) goto err1; /* * Join the two strings together, ensuring that the right thing * happens if the last component is empty, or the dirname is root. */ if (resolved[0] == '/' && resolved[1] == '\0') rootd = 1; else rootd = 0; if (*wbuf) { if (strlen(resolved) + strlen(wbuf) + (rootd ? 0 : 1) + 1 > MAXPATHLEN) { errno = ENAMETOOLONG; goto err1; } if (rootd == 0) if (strlcat(resolved, "/", MAXPATHLEN) >= MAXPATHLEN) { errno = ENAMETOOLONG; goto err1; } if (strlcat(resolved, wbuf, MAXPATHLEN) >= MAXPATHLEN) { errno = ENAMETOOLONG; goto err1; } } /* Go back to where we came from. */ if (fchdir(fd) < 0) { serrno = errno; goto err2; } /* It's okay if the close fails, what's an fd more or less? */ (void)close(fd); return (resolved); err1: serrno = errno; (void)fchdir(fd); err2: (void)close(fd); errno = serrno; return (NULL); }