/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License, Version 1.0 only * (the "License"). You may not use this file except in compliance * with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright 1996 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ /* Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */ /* All Rights Reserved */ /* * University Copyright- Copyright (c) 1982, 1986, 1988 * The Regents of the University of California * All Rights Reserved * * University Acknowledgment- Portions of this document are derived from * software developed by the University of California, Berkeley, and its * contributors. */ #include #include #include #include #include #include #include #include #include #include #include #define DEF_GROUP "staff" /* default group */ #define DEF_OWNER "root" /* default owner */ #define DEF_MODE 0755 /* default mode */ char *group = DEF_GROUP; char *owner = DEF_OWNER; int mode = DEF_MODE; int sflag = 0; struct passwd *pp; struct group *gp; extern int errno; int copy(); void usage(); int main(int argc, char **argv) { extern char *optarg; extern int optind; struct stat stb; char *dirname; int ch; int i; int rc; int dflag = 0; int gflag = 0; int oflag = 0; int mflag = 0; (void) setlocale(LC_ALL, ""); #if !defined(TEXT_DOMAIN) #define TEXT_DOMAIN "SYS_TEST" #endif (void) textdomain(TEXT_DOMAIN); while ((ch = getopt(argc, argv, "dcg:o:m:s")) != EOF) switch((char)ch) { case 'c': break; /* always do "copy" */ case 'd': dflag++; break; case 'g': gflag++; group = optarg; break; case 'm': mflag++; mode = atoo(optarg); break; case 'o': oflag++; owner = optarg; break; case 's': sflag++; break; case '?': default: usage(); } argc -= optind; argv += optind; /* get group and owner id's */ if (!(gp = getgrnam(group))) { fprintf(stderr, gettext("install: unknown group %s.\n"), group); exit(1); } if (!(pp = getpwnam(owner))) { fprintf(stderr, gettext("install: unknown user %s.\n"), owner); exit(1); } if (dflag) { /* install a directory */ int exists = 0; if (argc != 1) usage(); dirname = argv[0]; if (mkdirp(dirname, 0777) < 0) { exists = errno == EEXIST; if (!exists) { fprintf(stderr, gettext("install: mkdir: %s: %s\n"), dirname, strerror(errno)); exit(1); } } if (stat(dirname, &stb) < 0) { fprintf(stderr, gettext("install: stat: %s: %s\n"), dirname, strerror(errno)); exit(1); } if ((stb.st_mode&S_IFMT) != S_IFDIR) { fprintf(stderr, gettext("install: %s is not a directory\n"), dirname); } /* make sure directory setgid bit is inherited */ mode = (mode & ~S_ISGID) | (stb.st_mode & S_ISGID); if (mflag && chmod(dirname, mode)) { fprintf(stderr, gettext("install: chmod: %s: %s\n"), dirname, strerror(errno)); if (!exists) (void) unlink(dirname); exit(1) ; } if (oflag && chown(dirname, pp->pw_uid, -1) && errno != EPERM) { fprintf(stderr, gettext("install: chown: %s: %s\n"), dirname, strerror(errno)); if (!exists) (void) unlink(dirname); exit(1) ; } if (gflag && chown(dirname, -1, gp->gr_gid) && errno != EPERM) { fprintf(stderr, gettext("install: chgrp: %s: %s\n"), dirname, strerror(errno)); if (!exists) (void) unlink(dirname); exit(1) ; } exit(0); } if (argc < 2) usage(); if (argc > 2) { /* last arg must be a directory */ if (stat(argv[argc-1], &stb) < 0) usage(); if ((stb.st_mode&S_IFMT) != S_IFDIR) usage(); } rc = 0; for (i = 0; i < argc-1; i++) rc |= install(argv[i], argv[argc-1]); return (rc); } int install(from, to) char *from, *to; { int to_fd; int devnull; int status = 0; char *path; struct stat from_sb, to_sb; static char pbuf[MAXPATHLEN]; char buf[MAXPATHLEN + 10]; /* check source */ if (stat(from, &from_sb)) { fprintf(stderr, gettext("install: %s: %s\n"), from, strerror(errno)); return (1); } /* special case for removing files */ devnull = !strcmp(from, "/dev/null"); if (!devnull && !((from_sb.st_mode&S_IFMT) == S_IFREG)) { fprintf(stderr, gettext("install: %s isn't a regular file.\n"), from); return (1); } /* build target path, find out if target is same as source */ if (!stat(path = to, &to_sb)) { if ((to_sb.st_mode&S_IFMT) == S_IFDIR) { char *C, *strrchr(); (void) sprintf(path = pbuf, "%s/%s", to, (C = strrchr(from, '/')) ? ++C : from); if (stat(path, &to_sb)) goto nocompare; } if ((to_sb.st_mode&S_IFMT) != S_IFREG) { fprintf(stderr, gettext("install: %s isn't a regular file.\n"), path); return (1); } if (to_sb.st_dev == from_sb.st_dev && to_sb.st_ino == from_sb.st_ino) { fprintf(stderr, gettext("install: %s and %s are the same file.\n"), from, path); return (1); } /* unlink now... avoid ETXTBSY errors later */ (void) unlink(path); } nocompare: /* open target, set mode, owner, group */ if ((to_fd = open(path, O_CREAT|O_WRONLY|O_TRUNC, 0)) < 0) { fprintf(stderr, gettext("install: %s: %s\n"), path, strerror(errno)); return (1); } if (fchmod(to_fd, mode)) { fprintf(stderr, gettext("install: chmod: %s: %s\n"), path, strerror(errno)); status = 1; close(to_fd); goto inst_done; } if (!devnull) { status = copy(from, to_fd, path); /* copy */ close(to_fd); } if (sflag) { sprintf(buf, "strip %s", path); system(buf); } if (chown(path, pp->pw_uid, gp->gr_gid) && errno != EPERM) { fprintf(stderr, gettext("install: chown: %s: %s\n"), path, strerror(errno)); status = 1; } inst_done: if (status) (void) unlink(path); return (status); } /* * copy -- * copy from one file to another */ int copy(from_name, to_fd, to_name) int to_fd; char *from_name, *to_name; { int n, from_fd; int status = 0; char buf[MAXBSIZE]; if ((from_fd = open(from_name, O_RDONLY, 0)) < 0) { fprintf(stderr, gettext("install: open: %s: %s\n"), from_name, strerror(errno)); return (1); } while ((n = read(from_fd, buf, sizeof(buf))) > 0) if (write(to_fd, buf, n) != n) { fprintf(stderr, gettext("install: write: %s: %s\n"), to_name, strerror(errno)); status = 1; goto copy_done; } if (n == -1) { fprintf(stderr, gettext("install: read: %s: %s\n"), from_name, strerror(errno)); status = 1; goto copy_done; } copy_done: (void) close(from_fd); return (status); } /* * atoo -- * octal string to int */ int atoo(str) char *str; { int val; for (val = 0; isdigit(*str); ++str) val = val * 8 + *str - '0'; return(val); } /* * usage -- * print a usage message and die */ void usage() { fputs(gettext("usage: install [-cs] [-g group] [-m mode] [-o owner] file ... destination\n"), stderr); fputs(gettext(" install -d [-g group] [-m mode] [-o owner] dir\n"), stderr); exit(1); } /* * mkdirp -- * make a directory and parents if needed */ int mkdirp(dir, mode) char *dir; int mode; { int err; char *slash; char *strrchr(); extern int errno; if (mkdir(dir, mode) == 0) return (0); if (errno != ENOENT) return (-1); slash = strrchr(dir, '/'); if (slash == NULL) return (-1); *slash = '\0'; err = mkdirp(dir, 0777); *slash = '/'; if (err) return (err); return mkdir(dir, mode); }