mirror of
https://github.com/GnoConsortium/gno.git
synced 2025-01-20 06:30:12 +00:00
469a09b80b
- Ran sources through indent(1). - Separated out library routines.
462 lines
11 KiB
C
462 lines
11 KiB
C
/*
|
|
* Copyright 1996 Devin Reade <gdr@myrias.com>.
|
|
* All rights reserved.
|
|
*
|
|
* For copying and distribution information, see the file "COPYING"
|
|
* accompanying this file.
|
|
*
|
|
* $Id: inst.c,v 1.2 1996/09/03 03:54:58 gdr Exp $
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <getopt.h>
|
|
#include <errno.h>
|
|
#include <ctype.h>
|
|
#include <stdlib.h>
|
|
#include <assert.h>
|
|
#include <string.h>
|
|
#include <types.h>
|
|
#include <unistd.h>
|
|
#include <sys/stat.h>
|
|
#include <orca.h>
|
|
#include <gsos.h>
|
|
|
|
extern int _mapErr(int);
|
|
|
|
#include "contrib.h"
|
|
|
|
/* actions */
|
|
#define NOCHANGE 0
|
|
#define ASSIGN 1
|
|
#define REMOVE 2
|
|
#define ADD 3
|
|
|
|
/* permissions */
|
|
#define S_USER 0700
|
|
#define S_GROUP 0070
|
|
#define S_OTHER 0007
|
|
#define S_ALL 0777
|
|
#define S_READ 0444
|
|
#define S_WRITE 0222
|
|
#define S_EXECUTE 0111
|
|
|
|
#define TYPE_TXT 0x04
|
|
#define TYPE_SRC 0xB0
|
|
#define TYPE_EXEC 0x00000006
|
|
#define TYPE_NONE 0x00000000
|
|
|
|
#define VERSION "1.0"
|
|
#define EMAIL "<gdr@myrias.com>"
|
|
|
|
char *versionMsg = "Version %s by Devin Reade %s\n";
|
|
int dFlag;
|
|
|
|
extern int mkdir(const char *);
|
|
extern int needsgno(void);
|
|
extern void begin_stack_check(void);
|
|
extern int end_stack_check(void);
|
|
|
|
/*
|
|
* usage
|
|
*
|
|
* display usage information and exit
|
|
*/
|
|
|
|
void
|
|
usage(void)
|
|
{
|
|
fputs("Usage: install [-cdhsv] [-o owner] [-g group] [-m mode] ", stderr);
|
|
fputs("source [...] dest\n\n", stderr);
|
|
fputs("Options:\n", stderr);
|
|
fputs("\t-c Ignored. (Backwards Unix compatibility)\n", stderr);
|
|
fputs("\t-d Create the specified directories\n", stderr);
|
|
fputs("\t-g group Specify group id (not implemented)\n", stderr);
|
|
fputs("\t-h Show usage information and exit.\n", stderr);
|
|
fputs("\t-m mode Specify (Unix) access mode\n", stderr);
|
|
fputs("\t-o owner Specify owner id (not implemented)\n", stderr);
|
|
fputs("\t-s Strip binary (not implemented)\n", stderr);
|
|
fputs("\t-v Show version number\n\n", stderr);
|
|
fprintf(stderr, versionMsg, VERSION, EMAIL);
|
|
exit(1);
|
|
}
|
|
|
|
/*
|
|
* getmode
|
|
*
|
|
* set mode to the value corresponding to the permission bit string
|
|
* <str>. If the first char of <str> is a digit, then it is assumed
|
|
* to be an octal number. Otherwise it is assumed to be a string of
|
|
* the form "ug+rx" (in the usual chmod(1) format). Also sets action
|
|
* to be the type of action to take, whether we're removing, adding,
|
|
* or assigning the permission bits.
|
|
*
|
|
* If these assumptions don't hold, then return non-zero. Returns
|
|
* zero and sets mode on success.
|
|
*
|
|
* Since the IIgs currently doesn't have the concept of "group" and
|
|
* "other" permissions, we take everything from the user permissions.
|
|
*/
|
|
|
|
int
|
|
getmode(char *str, unsigned long *mode, int *action)
|
|
{
|
|
unsigned long who = 0L;
|
|
unsigned long perm = 0L;
|
|
char *p, *q;
|
|
|
|
/* octal number? */
|
|
if (isdigit(*str)) {
|
|
*action = ASSIGN;
|
|
errno = 0;
|
|
*mode = strtoul(str, NULL, 8);
|
|
return errno;
|
|
}
|
|
/* it's not an absolute octal; treat as a string */
|
|
if (((p = strchr(str, '+')) == NULL) &&
|
|
((p = strchr(str, '-')) == NULL) &&
|
|
((p = strchr(str, '=')) == NULL)) {
|
|
errno = EINVAL;
|
|
return errno;
|
|
}
|
|
switch (*p) {
|
|
case '+': *action = ADD; break;
|
|
case '-': *action = REMOVE; break;
|
|
case '=': *action = ASSIGN; break;
|
|
default: assert(0);
|
|
}
|
|
|
|
/*
|
|
* this condition should really be deduced from the umask, if it
|
|
* were supported.
|
|
*/
|
|
if (str == p) {
|
|
who |= S_USER;
|
|
}
|
|
|
|
for (q = str; q < p; q++) {
|
|
switch (*q) {
|
|
case 'u': who |= S_USER; break;
|
|
case 'g': who |= S_GROUP; break;
|
|
case 'o': who |= S_OTHER; break;
|
|
case 'a': who |= S_ALL; break;
|
|
default: errno = EINVAL; return errno;
|
|
}
|
|
}
|
|
|
|
for (q = p + 1; *q; q++) {
|
|
switch (*q) {
|
|
case 'r': perm |= S_READ; break;
|
|
case 'w': perm |= S_WRITE; break;
|
|
case 'x': perm |= S_EXECUTE; break;
|
|
case 's': /* ignored */ break;
|
|
default: errno = EINVAL; return errno;
|
|
}
|
|
}
|
|
|
|
/* currently: ignore all but user permissions */
|
|
if (!(who & S_USER)) {
|
|
*action = NOCHANGE;
|
|
}
|
|
*mode = who & perm;
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* mkdirs
|
|
*
|
|
* argv is assumed to be an array of argc pathnames. mkdirs will
|
|
* create all the listed directories, including their parents if
|
|
* necessary.
|
|
*
|
|
* Returns zero on success. Returns non-zero and prints a suitable
|
|
* error message on failure.
|
|
*/
|
|
|
|
int
|
|
mkdirs(int argc, char **argv)
|
|
{
|
|
static struct stat statbuf; /* reduce stack space */
|
|
char *path, *p;
|
|
size_t pathlen;
|
|
int result = 0;
|
|
int makeit; /* do we try a mkdir()? */
|
|
int abortpath; /* we saw an error; don't both with rest of path */
|
|
int i, j;
|
|
int coloncount;
|
|
|
|
/* loop over each of the pathnames in the array */
|
|
for (i = 0; i < argc; i++) {
|
|
|
|
/* expand to a full pathname */
|
|
if ((path = expandpath(argv[i])) == NULL) {
|
|
perror(argv[i]);
|
|
continue;
|
|
}
|
|
pathlen = strlen(path);
|
|
|
|
/* is this pathname legal? */
|
|
/* place a call to JudgeName() [custom] here */
|
|
|
|
/* find out how many path components there are */
|
|
coloncount = 0;
|
|
p = path;
|
|
while (*p) {
|
|
if (*p == ':') {
|
|
coloncount++;
|
|
}
|
|
p++;
|
|
}
|
|
p = path + 1;
|
|
|
|
/* skip the volume name */
|
|
if ((p = strchr(p, ':')) == NULL) {
|
|
/* only the volume name was given; skip it */
|
|
free(path);
|
|
continue;
|
|
}
|
|
p++;
|
|
--coloncount;
|
|
|
|
/* create each component in path */
|
|
abortpath = 0;
|
|
for (j = 0; !abortpath && j < coloncount; j++) {
|
|
if ((p = strchr(p, ':')) == NULL) {
|
|
p = path + pathlen;
|
|
}
|
|
*p = '\0';
|
|
|
|
if (stat(path, &statbuf) != 0) {
|
|
if (errno == ENOENT) {
|
|
makeit = 1;
|
|
} else {
|
|
perror(path);
|
|
makeit = 0;
|
|
abortpath = 1;
|
|
result = 1;
|
|
}
|
|
} else {
|
|
makeit = 0;
|
|
if (statbuf.st_mode & S_IFDIR == 0) {
|
|
fprintf(stderr, "%s exists and is not a directory\n", path);
|
|
abortpath = 1;
|
|
result = 1;
|
|
} /* else it exists and is a directory */
|
|
}
|
|
|
|
/* go ahead and create the directory */
|
|
if (makeit && mkdir(path)) {
|
|
perror(path);
|
|
abortpath = 1;
|
|
result = 1;
|
|
}
|
|
/* reinstate the ':' that we "nulled-out" */
|
|
if (p != path + pathlen) {
|
|
*p++ = ':';
|
|
}
|
|
}
|
|
free(path);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* copyfiles
|
|
*
|
|
* <argv> is assumed to be an array of <argc> filenames.
|
|
*
|
|
* This routine copies all but the last specified file to the directory
|
|
* or filename specified by the last filename. If argc>2, the last element
|
|
* _must_ be a directory.
|
|
*
|
|
* Returns zero on success. On failure, returns the last non-zero errno
|
|
* and prints error conditions to stderr.
|
|
*
|
|
* If action is not NOCHANGE, this routine will also set file permissions
|
|
* as specified in the install(1) man page. This may involve changing
|
|
* the file type.
|
|
*/
|
|
|
|
static int
|
|
copyfiles(int argc, char **argv, int action, unsigned long mode)
|
|
{
|
|
static FileInfoRecGS inforec;
|
|
static GSString255 filenameGS;
|
|
int i, j;
|
|
int result = 0;
|
|
char *destination;
|
|
Word newaccess;
|
|
|
|
if (argc < 2) {
|
|
errno = EINVAL;
|
|
perror("internal error: not enough arguments to copyfiles()");
|
|
return errno;
|
|
}
|
|
if (argc > 2) {
|
|
|
|
/* find out if argv[argc-1] is a directory */
|
|
|
|
if (__C2GS(argv[argc - 1], &filenameGS) == NULL) {
|
|
errno = EINVAL;
|
|
perror("destination path too long");
|
|
return errno;
|
|
}
|
|
inforec.pCount = 5;
|
|
inforec.pathname = &filenameGS;
|
|
GetFileInfoGS(&inforec);
|
|
if ((errnoGS = toolerror()) != 0) {
|
|
perrorGS("%s", argv[argc - 1]);
|
|
errno = _mapErr(errnoGS);
|
|
return -1;
|
|
}
|
|
if ((inforec.storageType != 0x0D) && (inforec.storageType != 0x0F)) {
|
|
errno = ENOTDIR;
|
|
perror(argv[argc - 1]);
|
|
return errno;
|
|
}
|
|
}
|
|
--argc;
|
|
for (i = 0; i < argc; i++) {
|
|
if ((destination = copyfile(argv[i], argv[argc])) == NULL) {
|
|
errnoGS = toolerror();
|
|
perrorGS("install of %s to %s", argv[i], argv[argc]);
|
|
result = errno = _mapErr(errnoGS);
|
|
}
|
|
if (action == NOCHANGE) {
|
|
continue;
|
|
}
|
|
|
|
/* get the file info for the source file */
|
|
if (__C2GS(argv[i], &filenameGS) == NULL) {
|
|
assert(0);
|
|
}
|
|
inforec.pCount = 7;
|
|
inforec.pathname = &filenameGS;
|
|
GetFileInfoGS(&inforec);
|
|
if ((errnoGS = toolerror()) != 0) {
|
|
perrorGS("GetFileInfo for %s failed", argv[i]);
|
|
result = errno = _mapErr(errnoGS);
|
|
}
|
|
/* modify the permissions as necessary */
|
|
switch (action) {
|
|
case ASSIGN:
|
|
newaccess = 0xFFFF;
|
|
if (!(mode & S_READ)) newaccess &= ~readEnable;
|
|
if (!(mode & S_WRITE)) newaccess &= ~writeEnable;
|
|
inforec.access &= newaccess;
|
|
|
|
if ((mode & S_EXECUTE) &&
|
|
(inforec.fileType == TYPE_TXT) || (inforec.fileType == TYPE_SRC)) {
|
|
inforec.fileType = TYPE_SRC;
|
|
inforec.auxType = TYPE_EXEC;
|
|
}
|
|
break;
|
|
|
|
case ADD:
|
|
if (mode & S_READ) inforec.access |= readEnable;
|
|
if (mode & S_WRITE) inforec.access |= writeEnable;
|
|
|
|
if ((mode & S_EXECUTE) &&
|
|
(inforec.fileType == TYPE_TXT) || (inforec.fileType == TYPE_SRC)) {
|
|
inforec.fileType = TYPE_SRC;
|
|
inforec.auxType = TYPE_EXEC;
|
|
}
|
|
break;
|
|
|
|
case REMOVE:
|
|
if (mode & S_READ) inforec.access &= ~readEnable;
|
|
if (mode & S_WRITE) inforec.access &= ~writeEnable;
|
|
|
|
if ((mode & S_EXECUTE) &&
|
|
(inforec.fileType == TYPE_TXT) || (inforec.fileType == TYPE_SRC)) {
|
|
inforec.fileType = TYPE_TXT;
|
|
inforec.auxType = TYPE_NONE;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
assert(0);
|
|
}
|
|
|
|
/* set the modified file info for the destination file */
|
|
if (__C2GS(destination, &filenameGS) == NULL) {
|
|
assert(0);
|
|
}
|
|
SetFileInfoGS(&inforec);
|
|
if ((errnoGS = toolerror()) != 0) {
|
|
perrorGS("SetFileInfo for %s failed", destination);
|
|
result = errno = _mapErr(errnoGS);
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* obvious ...
|
|
*/
|
|
|
|
int
|
|
main(int argc, char **argv)
|
|
{
|
|
unsigned long mode;
|
|
int c, nfiles;
|
|
int action = NOCHANGE;
|
|
|
|
#ifdef CHECK_STACK
|
|
begin_stack_check();
|
|
#endif
|
|
|
|
if (needsgno() == 0) {
|
|
fprintf(stderr, "Requires GNO/ME\n");
|
|
exit(1);
|
|
}
|
|
/* initialize */
|
|
dFlag = 0;
|
|
mode = 0L;
|
|
|
|
/* parse command line */
|
|
while ((c = getopt(argc, argv, "cdg:hm:o:sv")) != EOF) {
|
|
switch (c) {
|
|
case 'v':
|
|
fprintf(stderr, versionMsg, VERSION, EMAIL);
|
|
exit(1);
|
|
break;
|
|
|
|
case 'm':
|
|
if (getmode(optarg, &mode, &action)) {
|
|
usage();
|
|
}
|
|
break;
|
|
|
|
case 'd': dFlag++;
|
|
case 'c': /* not implemented */
|
|
case 'g': /* not implemented */
|
|
case 'o': /* not implemented */
|
|
case 's': /* not implemented */
|
|
break;
|
|
|
|
case 'h':
|
|
default: usage();
|
|
}
|
|
}
|
|
|
|
nfiles = argc - optind;
|
|
|
|
if (dFlag) {
|
|
if (nfiles < 1) {
|
|
usage();
|
|
}
|
|
c = mkdirs(nfiles, &argv[optind]);
|
|
} else {
|
|
if (nfiles < 2) {
|
|
usage();
|
|
}
|
|
c = copyfiles(nfiles, &argv[optind], action, mode);
|
|
}
|
|
|
|
#ifdef CHECK_STACK
|
|
fprintf(stderr, "stack usage: %d bytes\n", end_stack_check());
|
|
#endif
|
|
|
|
return c;
|
|
}
|