17c478bd9Sstevel@tonic-gate /* 27c478bd9Sstevel@tonic-gate * CDDL HEADER START 37c478bd9Sstevel@tonic-gate * 47c478bd9Sstevel@tonic-gate * The contents of this file are subject to the terms of the 519ca6462Scasper * Common Development and Distribution License (the "License"). 619ca6462Scasper * You may not use this file except in compliance with the License. 77c478bd9Sstevel@tonic-gate * 87c478bd9Sstevel@tonic-gate * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 97c478bd9Sstevel@tonic-gate * or http://www.opensolaris.org/os/licensing. 107c478bd9Sstevel@tonic-gate * See the License for the specific language governing permissions 117c478bd9Sstevel@tonic-gate * and limitations under the License. 127c478bd9Sstevel@tonic-gate * 137c478bd9Sstevel@tonic-gate * When distributing Covered Code, include this CDDL HEADER in each 147c478bd9Sstevel@tonic-gate * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 157c478bd9Sstevel@tonic-gate * If applicable, add the following below this CDDL HEADER, with the 167c478bd9Sstevel@tonic-gate * fields enclosed by brackets "[]" replaced with your own identifying 177c478bd9Sstevel@tonic-gate * information: Portions Copyright [yyyy] [name of copyright owner] 187c478bd9Sstevel@tonic-gate * 197c478bd9Sstevel@tonic-gate * CDDL HEADER END 207c478bd9Sstevel@tonic-gate */ 217c478bd9Sstevel@tonic-gate /* 22b83ec4edSjmcp * Copyright (c) 1999, 2010, Oracle and/or its affiliates. All rights reserved. 237c478bd9Sstevel@tonic-gate */ 247c478bd9Sstevel@tonic-gate 257c478bd9Sstevel@tonic-gate 267c478bd9Sstevel@tonic-gate #include <stdio.h> 277c478bd9Sstevel@tonic-gate #include <stdlib.h> 287c478bd9Sstevel@tonic-gate #include <strings.h> 297c478bd9Sstevel@tonic-gate #include <sys/param.h> 307c478bd9Sstevel@tonic-gate #include <fcntl.h> 317c478bd9Sstevel@tonic-gate #include <sys/errno.h> 327c478bd9Sstevel@tonic-gate #include <sys/types.h> 337c478bd9Sstevel@tonic-gate #include <sys/uio.h> 347c478bd9Sstevel@tonic-gate #include <unistd.h> 357c478bd9Sstevel@tonic-gate #include <sys/stat.h> 367c478bd9Sstevel@tonic-gate #include <errno.h> 377c478bd9Sstevel@tonic-gate #include <libgen.h> 380adc1619Smike_s #include "stdusers.h" 397c478bd9Sstevel@tonic-gate 407c478bd9Sstevel@tonic-gate 417c478bd9Sstevel@tonic-gate #define FILE_BUFF 40960 427c478bd9Sstevel@tonic-gate 43b83ec4edSjmcp static int suppress = 0; 44b83ec4edSjmcp 45b83ec4edSjmcp static void usage(void); 46b83ec4edSjmcp static void file_copy(char *src_file, char *dest_file); 47b83ec4edSjmcp static void chown_file(const char *file, const char *group, const char *owner); 48b83ec4edSjmcp static char *find_basename(const char *str); 49b83ec4edSjmcp static int creatdir(char *fn); 507c478bd9Sstevel@tonic-gate 517c478bd9Sstevel@tonic-gate 527c478bd9Sstevel@tonic-gate void 537c478bd9Sstevel@tonic-gate usage(void) 547c478bd9Sstevel@tonic-gate { 557c478bd9Sstevel@tonic-gate (void) fprintf(stderr, 56*c0e7977aSJosef 'Jeff' Sipek "usage: install [-sd][-m mode][-g group][-u owner] " 577c478bd9Sstevel@tonic-gate "-f dir file ...\n"); 587c478bd9Sstevel@tonic-gate } 597c478bd9Sstevel@tonic-gate 607c478bd9Sstevel@tonic-gate void 617c478bd9Sstevel@tonic-gate file_copy(char *src_file, char *dest_file) 627c478bd9Sstevel@tonic-gate { 637c478bd9Sstevel@tonic-gate int src_fd; 647c478bd9Sstevel@tonic-gate int dest_fd; 657c478bd9Sstevel@tonic-gate int count; 667c478bd9Sstevel@tonic-gate static char file_buff[FILE_BUFF]; 677c478bd9Sstevel@tonic-gate 687c478bd9Sstevel@tonic-gate if ((src_fd = open(src_file, O_RDONLY)) == -1) { 69b83ec4edSjmcp (void) fprintf(stderr, "install:file_copy: %s failed " 70b83ec4edSjmcp "(%d): %s\n", src_file, errno, strerror(errno)); 717c478bd9Sstevel@tonic-gate exit(1); 727c478bd9Sstevel@tonic-gate } 737c478bd9Sstevel@tonic-gate 747c478bd9Sstevel@tonic-gate if ((dest_fd = open(dest_file, O_CREAT|O_WRONLY|O_TRUNC, 0755)) == -1) { 75b83ec4edSjmcp (void) fprintf(stderr, "install:file_copy: %s failed " 76b83ec4edSjmcp "(%d): %s\n", dest_file, errno, strerror(errno)); 777c478bd9Sstevel@tonic-gate exit(1); 787c478bd9Sstevel@tonic-gate } 797c478bd9Sstevel@tonic-gate 807c478bd9Sstevel@tonic-gate while ((count = read(src_fd, file_buff, FILE_BUFF)) > 0) { 81b83ec4edSjmcp (void) write(dest_fd, file_buff, count); 827c478bd9Sstevel@tonic-gate } 837c478bd9Sstevel@tonic-gate 847c478bd9Sstevel@tonic-gate if (count == -1) { 85b83ec4edSjmcp (void) fprintf(stderr, "install:file_copy:read failed " 86b83ec4edSjmcp "(%d): %s\n", errno, strerror(errno)); 877c478bd9Sstevel@tonic-gate exit(1); 887c478bd9Sstevel@tonic-gate } 897c478bd9Sstevel@tonic-gate 90b83ec4edSjmcp if (!suppress) 917c478bd9Sstevel@tonic-gate (void) printf("%s installed as %s\n", src_file, dest_file); 927c478bd9Sstevel@tonic-gate 93b83ec4edSjmcp (void) close(src_fd); 94b83ec4edSjmcp (void) close(dest_fd); 957c478bd9Sstevel@tonic-gate } 967c478bd9Sstevel@tonic-gate 977c478bd9Sstevel@tonic-gate 987c478bd9Sstevel@tonic-gate void 997c478bd9Sstevel@tonic-gate chown_file(const char *file, const char *group, const char *owner) 1007c478bd9Sstevel@tonic-gate { 10119ca6462Scasper gid_t grp = (gid_t)-1; 10219ca6462Scasper uid_t own = (uid_t)-1; 1037c478bd9Sstevel@tonic-gate 1047c478bd9Sstevel@tonic-gate if (group) { 1050adc1619Smike_s grp = stdfind(group, groupnames); 1060adc1619Smike_s if (grp < 0) 1077c478bd9Sstevel@tonic-gate (void) fprintf(stderr, "unknown group(%s)\n", group); 1087c478bd9Sstevel@tonic-gate } 1097c478bd9Sstevel@tonic-gate 1107c478bd9Sstevel@tonic-gate if (owner) { 1110adc1619Smike_s own = stdfind(owner, usernames); 1120adc1619Smike_s if (own < 0) { 1137c478bd9Sstevel@tonic-gate (void) fprintf(stderr, "unknown owner(%s)\n", owner); 1147c478bd9Sstevel@tonic-gate exit(1); 1157c478bd9Sstevel@tonic-gate } 1167c478bd9Sstevel@tonic-gate 1177c478bd9Sstevel@tonic-gate } 1187c478bd9Sstevel@tonic-gate 1197c478bd9Sstevel@tonic-gate if (chown(file, own, grp) == -1) { 120b83ec4edSjmcp (void) fprintf(stderr, "install:chown_file: failed " 121b83ec4edSjmcp "(%d): %s\n", errno, strerror(errno)); 1227c478bd9Sstevel@tonic-gate exit(1); 1237c478bd9Sstevel@tonic-gate } 1247c478bd9Sstevel@tonic-gate } 1257c478bd9Sstevel@tonic-gate 126b83ec4edSjmcp 1277c478bd9Sstevel@tonic-gate char * 1287c478bd9Sstevel@tonic-gate find_basename(const char *str) 1297c478bd9Sstevel@tonic-gate { 1307c478bd9Sstevel@tonic-gate int i; 1317c478bd9Sstevel@tonic-gate int len; 1327c478bd9Sstevel@tonic-gate 1337c478bd9Sstevel@tonic-gate len = strlen(str); 1347c478bd9Sstevel@tonic-gate 1357c478bd9Sstevel@tonic-gate for (i = len-1; i >= 0; i--) 1367c478bd9Sstevel@tonic-gate if (str[i] == '/') 1377c478bd9Sstevel@tonic-gate return ((char *)(str + i + 1)); 1387c478bd9Sstevel@tonic-gate return ((char *)str); 1397c478bd9Sstevel@tonic-gate } 1407c478bd9Sstevel@tonic-gate 141b83ec4edSjmcp int 142b83ec4edSjmcp creatdir(char *fn) { 143b83ec4edSjmcp 144b83ec4edSjmcp errno = 0; 145b83ec4edSjmcp 146b83ec4edSjmcp if (mkdirp(fn, 0755) == -1) { 147b83ec4edSjmcp if (errno != EEXIST) 148b83ec4edSjmcp return (errno); 149b83ec4edSjmcp } else if (!suppress) { 150b83ec4edSjmcp (void) printf("directory %s created\n", fn); 151b83ec4edSjmcp } 152b83ec4edSjmcp return (0); 153b83ec4edSjmcp } 154b83ec4edSjmcp 1557c478bd9Sstevel@tonic-gate 1567c478bd9Sstevel@tonic-gate int 1577c478bd9Sstevel@tonic-gate main(int argc, char **argv) 1587c478bd9Sstevel@tonic-gate { 1597c478bd9Sstevel@tonic-gate int c; 1607c478bd9Sstevel@tonic-gate int errflg = 0; 1617c478bd9Sstevel@tonic-gate int dirflg = 0; 1627c478bd9Sstevel@tonic-gate char *group = NULL; 1637c478bd9Sstevel@tonic-gate char *owner = NULL; 1647c478bd9Sstevel@tonic-gate char *dirb = NULL; 1657c478bd9Sstevel@tonic-gate char *ins_file = NULL; 1667c478bd9Sstevel@tonic-gate int mode = -1; 1677c478bd9Sstevel@tonic-gate char dest_file[MAXPATHLEN]; 168b83ec4edSjmcp int rv = 0; 1697c478bd9Sstevel@tonic-gate 170*c0e7977aSJosef 'Jeff' Sipek while ((c = getopt(argc, argv, "f:sm:du:g:")) != EOF) { 1717c478bd9Sstevel@tonic-gate switch (c) { 1727c478bd9Sstevel@tonic-gate case 'f': 1737c478bd9Sstevel@tonic-gate dirb = optarg; 1747c478bd9Sstevel@tonic-gate break; 1757c478bd9Sstevel@tonic-gate case 'g': 1767c478bd9Sstevel@tonic-gate group = optarg; 1777c478bd9Sstevel@tonic-gate break; 1787c478bd9Sstevel@tonic-gate case 'u': 1797c478bd9Sstevel@tonic-gate owner = optarg; 1807c478bd9Sstevel@tonic-gate break; 1817c478bd9Sstevel@tonic-gate case 'd': 1827c478bd9Sstevel@tonic-gate dirflg = 1; 1837c478bd9Sstevel@tonic-gate break; 1847c478bd9Sstevel@tonic-gate case 'm': 1857c478bd9Sstevel@tonic-gate mode = strtol(optarg, NULL, 8); 1867c478bd9Sstevel@tonic-gate break; 1877c478bd9Sstevel@tonic-gate case 's': 188b83ec4edSjmcp suppress = 1; 189b83ec4edSjmcp break; 1907c478bd9Sstevel@tonic-gate case '?': 1917c478bd9Sstevel@tonic-gate errflg++; 1927c478bd9Sstevel@tonic-gate break; 1937c478bd9Sstevel@tonic-gate } 1947c478bd9Sstevel@tonic-gate } 1957c478bd9Sstevel@tonic-gate 1967c478bd9Sstevel@tonic-gate if (errflg) { 1977c478bd9Sstevel@tonic-gate usage(); 1987c478bd9Sstevel@tonic-gate return (1); 1997c478bd9Sstevel@tonic-gate } 2007c478bd9Sstevel@tonic-gate 2017c478bd9Sstevel@tonic-gate if (argc == optind) { 2027c478bd9Sstevel@tonic-gate usage(); 2037c478bd9Sstevel@tonic-gate return (1); 2047c478bd9Sstevel@tonic-gate } 2057c478bd9Sstevel@tonic-gate 2067c478bd9Sstevel@tonic-gate if (!dirflg && (dirb == NULL)) { 2077c478bd9Sstevel@tonic-gate (void) fprintf(stderr, 2087c478bd9Sstevel@tonic-gate "install: no destination directory specified.\n"); 2097c478bd9Sstevel@tonic-gate return (1); 2107c478bd9Sstevel@tonic-gate } 2117c478bd9Sstevel@tonic-gate 2127c478bd9Sstevel@tonic-gate for (c = optind; c < argc; c++) { 2137c478bd9Sstevel@tonic-gate ins_file = argv[c]; 2147c478bd9Sstevel@tonic-gate 2157c478bd9Sstevel@tonic-gate if (dirflg) { 216b83ec4edSjmcp rv = creatdir(ins_file); 217b83ec4edSjmcp if (rv) { 218b83ec4edSjmcp (void) fprintf(stderr, 219b83ec4edSjmcp "install: creatdir %s (%d): %s\n", 220b83ec4edSjmcp ins_file, errno, strerror(errno)); 221b83ec4edSjmcp return (rv); 222b83ec4edSjmcp } 223b83ec4edSjmcp (void) strlcpy(dest_file, ins_file, MAXPATHLEN); 2247c478bd9Sstevel@tonic-gate 2257c478bd9Sstevel@tonic-gate } else { 2267c478bd9Sstevel@tonic-gate (void) strcat(strcat(strcpy(dest_file, dirb), "/"), 2277c478bd9Sstevel@tonic-gate find_basename(ins_file)); 2287c478bd9Sstevel@tonic-gate file_copy(ins_file, dest_file); 2297c478bd9Sstevel@tonic-gate } 2307c478bd9Sstevel@tonic-gate 231*c0e7977aSJosef 'Jeff' Sipek if (group || owner) 2327c478bd9Sstevel@tonic-gate chown_file(dest_file, group, owner); 233*c0e7977aSJosef 'Jeff' Sipek 2347c478bd9Sstevel@tonic-gate if (mode != -1) { 235b83ec4edSjmcp (void) umask(0); 2367c478bd9Sstevel@tonic-gate if (chmod(dest_file, mode) == -1) { 237b83ec4edSjmcp (void) fprintf(stderr, 238b83ec4edSjmcp "install: chmod of %s to mode %o failed " 239b83ec4edSjmcp "(%d): %s\n", 240b83ec4edSjmcp dest_file, mode, errno, strerror(errno)); 2417c478bd9Sstevel@tonic-gate return (1); 2427c478bd9Sstevel@tonic-gate } 2437c478bd9Sstevel@tonic-gate } 2447c478bd9Sstevel@tonic-gate } 2457c478bd9Sstevel@tonic-gate return (0); 2467c478bd9Sstevel@tonic-gate } 247