1994-03-23 14:12:55 +00:00
|
|
|
|
/* Miscellaneous generic support functions for GNU Make.
|
|
|
|
|
Copyright (C) 1988, 89, 90, 91, 92, 93, 94 Free Software Foundation, Inc.
|
1991-10-07 22:04:20 +00:00
|
|
|
|
This file is part of GNU Make.
|
|
|
|
|
|
|
|
|
|
GNU Make is free software; you can redistribute it and/or modify
|
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
|
the Free Software Foundation; either version 2, or (at your option)
|
|
|
|
|
any later version.
|
|
|
|
|
|
|
|
|
|
GNU Make is distributed in the hope that it will be useful,
|
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
|
along with GNU Make; see the file COPYING. If not, write to
|
|
|
|
|
the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */
|
|
|
|
|
|
|
|
|
|
#include "make.h"
|
|
|
|
|
#include "dep.h"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Compare strings *S1 and *S2.
|
|
|
|
|
Return negative if the first is less, positive if it is greater,
|
|
|
|
|
zero if they are equal. */
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
alpha_compare (s1, s2)
|
|
|
|
|
char **s1, **s2;
|
|
|
|
|
{
|
|
|
|
|
if (**s1 != **s2)
|
|
|
|
|
return **s1 - **s2;
|
|
|
|
|
return strcmp (*s1, *s2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Discard each backslash-newline combination from LINE.
|
|
|
|
|
Backslash-backslash-newline combinations become backslash-newlines.
|
|
|
|
|
This is done by copying the text at LINE into itself. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
collapse_continuations (line)
|
|
|
|
|
char *line;
|
|
|
|
|
{
|
|
|
|
|
register char *in, *out, *p;
|
|
|
|
|
register int backslash;
|
|
|
|
|
register unsigned int bs_write;
|
|
|
|
|
|
|
|
|
|
in = index (line, '\n');
|
|
|
|
|
if (in == 0)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
out = in;
|
|
|
|
|
if (out > line)
|
|
|
|
|
while (out[-1] == '\\')
|
|
|
|
|
--out;
|
|
|
|
|
|
|
|
|
|
while (*in != '\0')
|
|
|
|
|
{
|
|
|
|
|
/* BS_WRITE gets the number of quoted backslashes at
|
|
|
|
|
the end just before IN, and BACKSLASH gets nonzero
|
|
|
|
|
if the next character is quoted. */
|
|
|
|
|
backslash = 0;
|
|
|
|
|
bs_write = 0;
|
|
|
|
|
for (p = in - 1; p >= line && *p == '\\'; --p)
|
|
|
|
|
{
|
|
|
|
|
if (backslash)
|
|
|
|
|
++bs_write;
|
|
|
|
|
backslash = !backslash;
|
|
|
|
|
|
|
|
|
|
/* It should be impossible to go back this far without exiting,
|
|
|
|
|
but if we do, we can't get the right answer. */
|
|
|
|
|
if (in == out - 1)
|
|
|
|
|
abort ();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Output the appropriate number of backslashes. */
|
|
|
|
|
while (bs_write-- > 0)
|
|
|
|
|
*out++ = '\\';
|
|
|
|
|
|
|
|
|
|
/* Skip the newline. */
|
|
|
|
|
++in;
|
|
|
|
|
|
|
|
|
|
/* If the newline is quoted, discard following whitespace
|
|
|
|
|
and any preceding whitespace; leave just one space. */
|
|
|
|
|
if (backslash)
|
|
|
|
|
{
|
|
|
|
|
in = next_token (in);
|
|
|
|
|
while (out > line && isblank (out[-1]))
|
|
|
|
|
--out;
|
|
|
|
|
*out++ = ' ';
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
/* If the newline isn't quoted, put it in the output. */
|
|
|
|
|
*out++ = '\n';
|
|
|
|
|
|
|
|
|
|
/* Now copy the following line to the output.
|
|
|
|
|
Stop when we find backslashes followed by a newline. */
|
|
|
|
|
while (*in != '\0')
|
|
|
|
|
if (*in == '\\')
|
|
|
|
|
{
|
|
|
|
|
p = in + 1;
|
|
|
|
|
while (*p == '\\')
|
|
|
|
|
++p;
|
|
|
|
|
if (*p == '\n')
|
|
|
|
|
{
|
|
|
|
|
in = p;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
while (in < p)
|
|
|
|
|
*out++ = *in++;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
*out++ = *in++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
*out = '\0';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Remove comments from LINE.
|
|
|
|
|
This is done by copying the text at LINE onto itself. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
remove_comments (line)
|
|
|
|
|
char *line;
|
|
|
|
|
{
|
1994-04-21 21:18:04 +00:00
|
|
|
|
char *comment;
|
1991-10-07 22:04:20 +00:00
|
|
|
|
|
1994-04-21 21:18:04 +00:00
|
|
|
|
comment = find_char_unquote (line, '#', 0);
|
1991-10-07 22:04:20 +00:00
|
|
|
|
|
1994-04-21 21:18:04 +00:00
|
|
|
|
if (comment != 0)
|
|
|
|
|
/* Cut off the line at the #. */
|
|
|
|
|
*comment = '\0';
|
1991-10-07 22:04:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Print N spaces (used by DEBUGPR for target-depth). */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
print_spaces (n)
|
|
|
|
|
register unsigned int n;
|
|
|
|
|
{
|
|
|
|
|
while (n-- > 0)
|
|
|
|
|
putchar (' ');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Return a newly-allocated string whose contents
|
|
|
|
|
concatenate those of s1, s2, s3. */
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
concat (s1, s2, s3)
|
|
|
|
|
register char *s1, *s2, *s3;
|
|
|
|
|
{
|
|
|
|
|
register unsigned int len1, len2, len3;
|
|
|
|
|
register char *result;
|
|
|
|
|
|
|
|
|
|
len1 = *s1 != '\0' ? strlen (s1) : 0;
|
|
|
|
|
len2 = *s2 != '\0' ? strlen (s2) : 0;
|
|
|
|
|
len3 = *s3 != '\0' ? strlen (s3) : 0;
|
|
|
|
|
|
|
|
|
|
result = (char *) xmalloc (len1 + len2 + len3 + 1);
|
|
|
|
|
|
|
|
|
|
if (*s1 != '\0')
|
|
|
|
|
bcopy (s1, result, len1);
|
|
|
|
|
if (*s2 != '\0')
|
|
|
|
|
bcopy (s2, result + len1, len2);
|
|
|
|
|
if (*s3 != '\0')
|
|
|
|
|
bcopy (s3, result + len1 + len2, len3);
|
|
|
|
|
*(result + len1 + len2 + len3) = '\0';
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Print a message on stdout. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
message (s1, s2, s3, s4, s5, s6)
|
|
|
|
|
char *s1, *s2, *s3, *s4, *s5, *s6;
|
|
|
|
|
{
|
|
|
|
|
if (makelevel == 0)
|
|
|
|
|
printf ("%s: ", program);
|
|
|
|
|
else
|
|
|
|
|
printf ("%s[%u]: ", program, makelevel);
|
|
|
|
|
printf (s1, s2, s3, s4, s5, s6);
|
|
|
|
|
putchar ('\n');
|
|
|
|
|
fflush (stdout);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Print an error message and exit. */
|
|
|
|
|
|
|
|
|
|
/* VARARGS1 */
|
|
|
|
|
void
|
|
|
|
|
fatal (s1, s2, s3, s4, s5, s6)
|
|
|
|
|
char *s1, *s2, *s3, *s4, *s5, *s6;
|
|
|
|
|
{
|
|
|
|
|
if (makelevel == 0)
|
1992-04-29 21:23:10 +00:00
|
|
|
|
fprintf (stderr, "%s: *** ", program);
|
1991-10-07 22:04:20 +00:00
|
|
|
|
else
|
1992-04-29 21:23:10 +00:00
|
|
|
|
fprintf (stderr, "%s[%u]: *** ", program, makelevel);
|
1991-10-07 22:04:20 +00:00
|
|
|
|
fprintf (stderr, s1, s2, s3, s4, s5, s6);
|
|
|
|
|
fputs (". Stop.\n", stderr);
|
|
|
|
|
|
1994-04-21 02:01:24 +00:00
|
|
|
|
die (2);
|
1991-10-07 22:04:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Print error message. `s1' is printf control string, `s2' is arg for it. */
|
|
|
|
|
|
|
|
|
|
/* VARARGS1 */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
error (s1, s2, s3, s4, s5, s6)
|
|
|
|
|
char *s1, *s2, *s3, *s4, *s5, *s6;
|
|
|
|
|
{
|
|
|
|
|
if (makelevel == 0)
|
|
|
|
|
fprintf (stderr, "%s: ", program);
|
|
|
|
|
else
|
|
|
|
|
fprintf (stderr, "%s[%u]: ", program, makelevel);
|
|
|
|
|
fprintf (stderr, s1, s2, s3, s4, s5, s6);
|
|
|
|
|
putc ('\n', stderr);
|
|
|
|
|
fflush (stderr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
makefile_error (file, lineno, s1, s2, s3, s4, s5, s6)
|
|
|
|
|
char *file;
|
|
|
|
|
unsigned int lineno;
|
|
|
|
|
char *s1, *s2, *s3, *s4, *s5, *s6;
|
|
|
|
|
{
|
|
|
|
|
fprintf (stderr, "%s:%u: ", file, lineno);
|
|
|
|
|
fprintf (stderr, s1, s2, s3, s4, s5, s6);
|
|
|
|
|
putc ('\n', stderr);
|
|
|
|
|
fflush (stderr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
makefile_fatal (file, lineno, s1, s2, s3, s4, s5, s6)
|
|
|
|
|
char *file;
|
|
|
|
|
unsigned int lineno;
|
|
|
|
|
char *s1, *s2, *s3, *s4, *s5, *s6;
|
|
|
|
|
{
|
1992-04-29 21:23:10 +00:00
|
|
|
|
fprintf (stderr, "%s:%u: *** ", file, lineno);
|
1991-10-07 22:04:20 +00:00
|
|
|
|
fprintf (stderr, s1, s2, s3, s4, s5, s6);
|
|
|
|
|
fputs (". Stop.\n", stderr);
|
|
|
|
|
|
1994-04-21 02:01:24 +00:00
|
|
|
|
die (2);
|
1991-10-07 22:04:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
1993-12-02 20:37:46 +00:00
|
|
|
|
#ifndef HAVE_STRERROR
|
1994-02-03 07:21:18 +00:00
|
|
|
|
|
|
|
|
|
#undef strerror
|
|
|
|
|
|
1993-12-02 20:37:46 +00:00
|
|
|
|
char *
|
|
|
|
|
strerror (errnum)
|
|
|
|
|
int errnum;
|
|
|
|
|
{
|
|
|
|
|
extern int errno, sys_nerr;
|
|
|
|
|
extern char *sys_errlist[];
|
|
|
|
|
static char buf[] = "Unknown error 12345678901234567890";
|
|
|
|
|
|
|
|
|
|
if (errno < sys_nerr)
|
|
|
|
|
return sys_errlist[errnum];
|
|
|
|
|
|
|
|
|
|
sprintf ("Unknown error %d", buf, errnum);
|
|
|
|
|
return buf;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
1991-10-07 22:04:20 +00:00
|
|
|
|
/* Print an error message from errno. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
perror_with_name (str, name)
|
|
|
|
|
char *str, *name;
|
|
|
|
|
{
|
1993-12-02 20:37:46 +00:00
|
|
|
|
error ("%s%s: %s", str, name, strerror (errno));
|
1991-10-07 22:04:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Print an error message from errno and exit. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
pfatal_with_name (name)
|
|
|
|
|
char *name;
|
|
|
|
|
{
|
1993-12-02 20:37:46 +00:00
|
|
|
|
fatal ("%s: %s", name, strerror (errno));
|
1991-10-07 22:04:20 +00:00
|
|
|
|
|
|
|
|
|
/* NOTREACHED */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Like malloc but get fatal error if memory is exhausted. */
|
|
|
|
|
|
|
|
|
|
#undef xmalloc
|
|
|
|
|
#undef xrealloc
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
xmalloc (size)
|
|
|
|
|
unsigned int size;
|
|
|
|
|
{
|
1993-01-13 21:23:01 +00:00
|
|
|
|
char *result = (char *) malloc (size);
|
1991-10-07 22:04:20 +00:00
|
|
|
|
if (result == 0)
|
|
|
|
|
fatal ("virtual memory exhausted");
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
xrealloc (ptr, size)
|
|
|
|
|
char *ptr;
|
|
|
|
|
unsigned int size;
|
|
|
|
|
{
|
1993-01-13 21:23:01 +00:00
|
|
|
|
char *result = (char *) realloc (ptr, size);
|
1991-10-07 22:04:20 +00:00
|
|
|
|
if (result == 0)
|
|
|
|
|
fatal ("virtual memory exhausted");
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
savestring (str, length)
|
|
|
|
|
char *str;
|
|
|
|
|
unsigned int length;
|
|
|
|
|
{
|
|
|
|
|
register char *out = (char *) xmalloc (length + 1);
|
|
|
|
|
if (length > 0)
|
|
|
|
|
bcopy (str, out, length);
|
|
|
|
|
out[length] = '\0';
|
|
|
|
|
return out;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Search string BIG (length BLEN) for an occurrence of
|
|
|
|
|
string SMALL (length SLEN). Return a pointer to the
|
|
|
|
|
beginning of the first occurrence, or return nil if none found. */
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
sindex (big, blen, small, slen)
|
|
|
|
|
char *big;
|
|
|
|
|
unsigned int blen;
|
|
|
|
|
char *small;
|
|
|
|
|
unsigned int slen;
|
|
|
|
|
{
|
|
|
|
|
register unsigned int b;
|
|
|
|
|
|
|
|
|
|
if (blen < 1)
|
|
|
|
|
blen = strlen (big);
|
|
|
|
|
if (slen < 1)
|
|
|
|
|
slen = strlen (small);
|
|
|
|
|
|
|
|
|
|
for (b = 0; b < blen; ++b)
|
|
|
|
|
if (big[b] == *small && !strncmp (&big[b + 1], small + 1, slen - 1))
|
|
|
|
|
return (&big[b]);
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Limited INDEX:
|
|
|
|
|
Search through the string STRING, which ends at LIMIT, for the character C.
|
|
|
|
|
Returns a pointer to the first occurrence, or nil if none is found.
|
|
|
|
|
Like INDEX except that the string searched ends where specified
|
|
|
|
|
instead of at the first null. */
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
lindex (s, limit, c)
|
|
|
|
|
register char *s, *limit;
|
|
|
|
|
int c;
|
|
|
|
|
{
|
|
|
|
|
while (s < limit)
|
|
|
|
|
if (*s++ == c)
|
|
|
|
|
return s - 1;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Return the address of the first whitespace or null in the string S. */
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
end_of_token (s)
|
|
|
|
|
char *s;
|
|
|
|
|
{
|
1994-03-04 05:10:20 +00:00
|
|
|
|
while (*s != '\0' && !isblank (*s))
|
|
|
|
|
++s;
|
|
|
|
|
return s;
|
1991-10-07 22:04:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Return the address of the first nonwhitespace or null in the string S. */
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
next_token (s)
|
|
|
|
|
char *s;
|
|
|
|
|
{
|
|
|
|
|
register char *p = s;
|
|
|
|
|
|
|
|
|
|
while (isblank (*p))
|
|
|
|
|
++p;
|
|
|
|
|
return p;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Find the next token in PTR; return the address of it, and store the
|
|
|
|
|
length of the token into *LENGTHPTR if LENGTHPTR is not nil. */
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
find_next_token (ptr, lengthptr)
|
|
|
|
|
char **ptr;
|
|
|
|
|
unsigned int *lengthptr;
|
|
|
|
|
{
|
|
|
|
|
char *p = next_token (*ptr);
|
|
|
|
|
char *end;
|
|
|
|
|
|
|
|
|
|
if (*p == '\0')
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
*ptr = end = end_of_token (p);
|
|
|
|
|
if (lengthptr != 0)
|
|
|
|
|
*lengthptr = end - p;
|
|
|
|
|
return p;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Copy a chain of `struct dep', making a new chain
|
|
|
|
|
with the same contents as the old one. */
|
|
|
|
|
|
|
|
|
|
struct dep *
|
|
|
|
|
copy_dep_chain (d)
|
|
|
|
|
register struct dep *d;
|
|
|
|
|
{
|
|
|
|
|
register struct dep *c;
|
|
|
|
|
struct dep *firstnew = 0;
|
|
|
|
|
struct dep *lastnew;
|
|
|
|
|
|
|
|
|
|
while (d != 0)
|
|
|
|
|
{
|
|
|
|
|
c = (struct dep *) xmalloc (sizeof (struct dep));
|
|
|
|
|
bcopy ((char *) d, (char *) c, sizeof (struct dep));
|
|
|
|
|
if (c->name != 0)
|
|
|
|
|
c->name = savestring (c->name, strlen (c->name));
|
|
|
|
|
c->next = 0;
|
|
|
|
|
if (firstnew == 0)
|
|
|
|
|
firstnew = lastnew = c;
|
|
|
|
|
else
|
|
|
|
|
lastnew = lastnew->next = c;
|
|
|
|
|
|
|
|
|
|
d = d->next;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return firstnew;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef iAPX286
|
|
|
|
|
/* The losing compiler on this machine can't handle this macro. */
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
dep_name (dep)
|
|
|
|
|
struct dep *dep;
|
|
|
|
|
{
|
|
|
|
|
return dep->name == 0 ? dep->file->name : dep->name;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
1993-01-11 19:45:54 +00:00
|
|
|
|
#ifdef GETLOADAVG_PRIVILEGED
|
|
|
|
|
|
1994-02-16 21:25:32 +00:00
|
|
|
|
#ifdef POSIX
|
1994-05-17 03:09:02 +00:00
|
|
|
|
|
1994-02-16 21:25:32 +00:00
|
|
|
|
/* Hopefully if a system says it's POSIX.1 and has the setuid and setgid
|
|
|
|
|
functions, they work as POSIX.1 says. Some systems (Alpha OSF/1 1.2,
|
|
|
|
|
for example) which claim to be POSIX.1 also have the BSD setreuid and
|
|
|
|
|
setregid functions, but they don't work as in BSD and only the POSIX.1
|
|
|
|
|
way works. */
|
|
|
|
|
|
|
|
|
|
#undef HAVE_SETREUID
|
|
|
|
|
#undef HAVE_SETREGID
|
1994-05-17 03:09:02 +00:00
|
|
|
|
|
|
|
|
|
#else /* Not POSIX. */
|
|
|
|
|
|
|
|
|
|
/* Some POSIX.1 systems have the seteuid and setegid functions. In a
|
|
|
|
|
POSIX-like system, they are the best thing to use. However, some
|
|
|
|
|
non-POSIX systems have them too but they do not work in the POSIX style
|
|
|
|
|
and we must use setreuid and setregid instead. */
|
|
|
|
|
|
|
|
|
|
#undef HAVE_SETEUID
|
|
|
|
|
#undef HAVE_SETEGID
|
|
|
|
|
|
|
|
|
|
#endif /* POSIX. */
|
1994-02-16 21:25:32 +00:00
|
|
|
|
|
1992-06-11 04:58:15 +00:00
|
|
|
|
#ifndef HAVE_UNISTD_H
|
1991-10-07 22:04:20 +00:00
|
|
|
|
extern int getuid (), getgid (), geteuid (), getegid ();
|
|
|
|
|
extern int setuid (), setgid ();
|
1994-04-21 02:41:36 +00:00
|
|
|
|
#ifdef HAVE_SETEUID
|
|
|
|
|
extern int seteuid ();
|
|
|
|
|
#else
|
1993-01-22 21:32:00 +00:00
|
|
|
|
#ifdef HAVE_SETREUID
|
|
|
|
|
extern int setreuid ();
|
|
|
|
|
#endif /* Have setreuid. */
|
1994-04-21 02:41:36 +00:00
|
|
|
|
#endif /* Have seteuid. */
|
|
|
|
|
#ifdef HAVE_SETEGID
|
|
|
|
|
extern int setegid ();
|
|
|
|
|
#else
|
1993-01-22 21:32:00 +00:00
|
|
|
|
#ifdef HAVE_SETREGID
|
|
|
|
|
extern int setregid ();
|
|
|
|
|
#endif /* Have setregid. */
|
1994-04-21 02:41:36 +00:00
|
|
|
|
#endif /* Have setegid. */
|
1992-06-11 04:58:15 +00:00
|
|
|
|
#endif /* No <unistd.h>. */
|
1991-10-07 22:04:20 +00:00
|
|
|
|
|
|
|
|
|
/* Keep track of the user and group IDs for user- and make- access. */
|
|
|
|
|
static int user_uid = -1, user_gid = -1, make_uid = -1, make_gid = -1;
|
|
|
|
|
#define access_inited (user_uid != -1)
|
|
|
|
|
static enum { make, user } current_access;
|
|
|
|
|
|
1993-04-14 20:43:53 +00:00
|
|
|
|
|
|
|
|
|
/* Under -d, write a message describing the current IDs. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
log_access (flavor)
|
|
|
|
|
char *flavor;
|
|
|
|
|
{
|
|
|
|
|
if (! debug_flag)
|
|
|
|
|
return;
|
|
|
|
|
|
1993-08-01 20:03:45 +00:00
|
|
|
|
/* All the other debugging messages go to stdout,
|
|
|
|
|
but we write this one to stderr because it might be
|
|
|
|
|
run in a child fork whose stdout is piped. */
|
|
|
|
|
|
|
|
|
|
fprintf (stderr, "%s access: user %d (real %d), group %d (real %d)\n",
|
|
|
|
|
flavor, geteuid (), getuid (), getegid (), getgid ());
|
|
|
|
|
fflush (stderr);
|
1993-04-14 20:43:53 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1991-10-07 22:04:20 +00:00
|
|
|
|
static void
|
|
|
|
|
init_access ()
|
|
|
|
|
{
|
|
|
|
|
user_uid = getuid ();
|
|
|
|
|
user_gid = getgid ();
|
|
|
|
|
|
|
|
|
|
make_uid = geteuid ();
|
|
|
|
|
make_gid = getegid ();
|
|
|
|
|
|
|
|
|
|
/* Do these ever fail? */
|
|
|
|
|
if (user_uid == -1 || user_gid == -1 || make_uid == -1 || make_gid == -1)
|
|
|
|
|
pfatal_with_name ("get{e}[gu]id");
|
|
|
|
|
|
1993-04-14 20:43:53 +00:00
|
|
|
|
log_access ("Initialized");
|
|
|
|
|
|
1991-10-07 22:04:20 +00:00
|
|
|
|
current_access = make;
|
|
|
|
|
}
|
|
|
|
|
|
1993-01-11 19:45:54 +00:00
|
|
|
|
#endif /* GETLOADAVG_PRIVILEGED */
|
|
|
|
|
|
1991-10-07 22:04:20 +00:00
|
|
|
|
/* Give the process appropriate permissions for access to
|
|
|
|
|
user data (i.e., to stat files, or to spawn a child process). */
|
|
|
|
|
void
|
|
|
|
|
user_access ()
|
|
|
|
|
{
|
1993-01-11 19:45:54 +00:00
|
|
|
|
#ifdef GETLOADAVG_PRIVILEGED
|
|
|
|
|
|
1991-10-07 22:04:20 +00:00
|
|
|
|
if (!access_inited)
|
|
|
|
|
init_access ();
|
|
|
|
|
|
|
|
|
|
if (current_access == user)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
/* We are in "make access" mode. This means that the effective user and
|
|
|
|
|
group IDs are those of make (if it was installed setuid or setgid).
|
|
|
|
|
We now want to set the effective user and group IDs to the real IDs,
|
|
|
|
|
which are the IDs of the process that exec'd make. */
|
|
|
|
|
|
1994-04-21 02:41:36 +00:00
|
|
|
|
#ifdef HAVE_SETEUID
|
|
|
|
|
|
|
|
|
|
/* Modern systems have the seteuid/setegid calls which set only the
|
|
|
|
|
effective IDs, which is ideal. */
|
|
|
|
|
|
|
|
|
|
if (seteuid (user_uid) < 0)
|
|
|
|
|
pfatal_with_name ("user_access: seteuid");
|
|
|
|
|
|
|
|
|
|
#else /* Not HAVE_SETEUID. */
|
|
|
|
|
|
1993-01-05 23:02:42 +00:00
|
|
|
|
#ifndef HAVE_SETREUID
|
|
|
|
|
|
1991-10-07 22:04:20 +00:00
|
|
|
|
/* System V has only the setuid/setgid calls to set user/group IDs.
|
|
|
|
|
There is an effective ID, which can be set by setuid/setgid.
|
|
|
|
|
It can be set (unless you are root) only to either what it already is
|
|
|
|
|
(returned by geteuid/getegid, now in make_uid/make_gid),
|
|
|
|
|
the real ID (return by getuid/getgid, now in user_uid/user_gid),
|
|
|
|
|
or the saved set ID (what the effective ID was before this set-ID
|
|
|
|
|
executable (make) was exec'd). */
|
1993-01-05 23:02:42 +00:00
|
|
|
|
|
1991-10-07 22:04:20 +00:00
|
|
|
|
if (setuid (user_uid) < 0)
|
1993-01-05 23:02:42 +00:00
|
|
|
|
pfatal_with_name ("user_access: setuid");
|
|
|
|
|
|
1994-04-21 02:41:36 +00:00
|
|
|
|
#else /* HAVE_SETREUID. */
|
1993-01-05 23:02:42 +00:00
|
|
|
|
|
1991-10-07 22:04:20 +00:00
|
|
|
|
/* In 4BSD, the setreuid/setregid calls set both the real and effective IDs.
|
|
|
|
|
They may be set to themselves or each other. So you have two alternatives
|
|
|
|
|
at any one time. If you use setuid/setgid, the effective will be set to
|
|
|
|
|
the real, leaving only one alternative. Using setreuid/setregid, however,
|
|
|
|
|
you can toggle between your two alternatives by swapping the values in a
|
|
|
|
|
single setreuid or setregid call. */
|
1993-01-05 23:02:42 +00:00
|
|
|
|
|
1991-10-07 22:04:20 +00:00
|
|
|
|
if (setreuid (make_uid, user_uid) < 0)
|
1993-01-05 23:02:42 +00:00
|
|
|
|
pfatal_with_name ("user_access: setreuid");
|
|
|
|
|
|
1994-04-21 02:41:36 +00:00
|
|
|
|
#endif /* Not HAVE_SETREUID. */
|
|
|
|
|
#endif /* HAVE_SETEUID. */
|
1993-01-05 23:02:42 +00:00
|
|
|
|
|
1994-04-21 02:41:36 +00:00
|
|
|
|
#ifdef HAVE_SETEGID
|
|
|
|
|
if (setegid (user_gid) < 0)
|
|
|
|
|
pfatal_with_name ("user_access: setegid");
|
|
|
|
|
#else
|
1993-01-05 23:02:42 +00:00
|
|
|
|
#ifndef HAVE_SETREGID
|
|
|
|
|
if (setgid (user_gid) < 0)
|
|
|
|
|
pfatal_with_name ("user_access: setgid");
|
|
|
|
|
#else
|
1991-10-07 22:04:20 +00:00
|
|
|
|
if (setregid (make_gid, user_gid) < 0)
|
1993-01-05 23:02:42 +00:00
|
|
|
|
pfatal_with_name ("user_access: setregid");
|
1994-04-21 02:41:36 +00:00
|
|
|
|
#endif
|
1991-10-07 22:04:20 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
current_access = user;
|
1993-01-11 19:45:54 +00:00
|
|
|
|
|
1993-04-14 20:43:53 +00:00
|
|
|
|
log_access ("User");
|
|
|
|
|
|
1993-01-11 19:45:54 +00:00
|
|
|
|
#endif /* GETLOADAVG_PRIVILEGED */
|
1991-10-07 22:04:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Give the process appropriate permissions for access to
|
|
|
|
|
make data (i.e., the load average). */
|
|
|
|
|
void
|
|
|
|
|
make_access ()
|
|
|
|
|
{
|
1993-01-11 19:45:54 +00:00
|
|
|
|
#ifdef GETLOADAVG_PRIVILEGED
|
|
|
|
|
|
1991-10-07 22:04:20 +00:00
|
|
|
|
if (!access_inited)
|
|
|
|
|
init_access ();
|
|
|
|
|
|
|
|
|
|
if (current_access == make)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
/* See comments in user_access, above. */
|
|
|
|
|
|
1994-04-21 02:41:36 +00:00
|
|
|
|
#ifdef HAVE_SETEUID
|
|
|
|
|
if (seteuid (make_uid) < 0)
|
|
|
|
|
pfatal_with_name ("make_access: seteuid");
|
|
|
|
|
#else
|
1993-01-05 23:02:42 +00:00
|
|
|
|
#ifndef HAVE_SETREUID
|
1991-10-07 22:04:20 +00:00
|
|
|
|
if (setuid (make_uid) < 0)
|
1993-01-05 23:02:42 +00:00
|
|
|
|
pfatal_with_name ("make_access: setuid");
|
1991-10-07 22:04:20 +00:00
|
|
|
|
#else
|
|
|
|
|
if (setreuid (user_uid, make_uid) < 0)
|
1993-01-05 23:02:42 +00:00
|
|
|
|
pfatal_with_name ("make_access: setreuid");
|
|
|
|
|
#endif
|
1994-04-21 02:41:36 +00:00
|
|
|
|
#endif
|
1993-01-05 23:02:42 +00:00
|
|
|
|
|
1994-04-21 02:41:36 +00:00
|
|
|
|
#ifdef HAVE_SETEGID
|
|
|
|
|
if (setegid (make_gid) < 0)
|
|
|
|
|
pfatal_with_name ("make_access: setegid");
|
|
|
|
|
#else
|
1993-01-05 23:02:42 +00:00
|
|
|
|
#ifndef HAVE_SETREGID
|
|
|
|
|
if (setgid (make_gid) < 0)
|
|
|
|
|
pfatal_with_name ("make_access: setgid");
|
|
|
|
|
#else
|
1991-10-07 22:04:20 +00:00
|
|
|
|
if (setregid (user_gid, make_gid) < 0)
|
1993-01-05 23:02:42 +00:00
|
|
|
|
pfatal_with_name ("make_access: setregid");
|
1994-04-21 02:41:36 +00:00
|
|
|
|
#endif
|
1991-10-07 22:04:20 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
current_access = make;
|
1993-01-11 19:45:54 +00:00
|
|
|
|
|
1993-04-14 20:43:53 +00:00
|
|
|
|
log_access ("Make");
|
|
|
|
|
|
1993-01-11 19:45:54 +00:00
|
|
|
|
#endif /* GETLOADAVG_PRIVILEGED */
|
1991-10-07 22:04:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Give the process appropriate permissions for a child process.
|
|
|
|
|
This is like user_access, but you can't get back to make_access. */
|
|
|
|
|
void
|
|
|
|
|
child_access ()
|
|
|
|
|
{
|
1993-04-15 22:42:20 +00:00
|
|
|
|
#ifdef GETLOADAVG_PRIVILEGED
|
|
|
|
|
|
1993-04-14 20:43:53 +00:00
|
|
|
|
if (!access_inited)
|
|
|
|
|
abort ();
|
|
|
|
|
|
1991-10-07 22:04:20 +00:00
|
|
|
|
/* Set both the real and effective UID and GID to the user's.
|
|
|
|
|
They cannot be changed back to make's. */
|
|
|
|
|
|
1993-01-08 20:32:36 +00:00
|
|
|
|
#ifndef HAVE_SETREUID
|
1991-10-07 22:04:20 +00:00
|
|
|
|
if (setuid (user_uid) < 0)
|
1993-01-05 23:02:42 +00:00
|
|
|
|
pfatal_with_name ("child_access: setuid");
|
1993-01-08 20:32:36 +00:00
|
|
|
|
#else
|
|
|
|
|
if (setreuid (user_uid, user_uid) < 0)
|
|
|
|
|
pfatal_with_name ("child_access: setreuid");
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifndef HAVE_SETREGID
|
1991-10-07 22:04:20 +00:00
|
|
|
|
if (setgid (user_gid) < 0)
|
1993-01-05 23:02:42 +00:00
|
|
|
|
pfatal_with_name ("child_access: setgid");
|
1993-01-08 20:32:36 +00:00
|
|
|
|
#else
|
|
|
|
|
if (setregid (user_gid, user_gid) < 0)
|
|
|
|
|
pfatal_with_name ("child_access: setregid");
|
|
|
|
|
#endif
|
1993-01-11 19:45:54 +00:00
|
|
|
|
|
1993-04-14 20:43:53 +00:00
|
|
|
|
log_access ("Child");
|
|
|
|
|
|
1993-01-11 19:45:54 +00:00
|
|
|
|
#endif /* GETLOADAVG_PRIVILEGED */
|
1991-10-07 22:04:20 +00:00
|
|
|
|
}
|
1992-10-15 23:01:34 +00:00
|
|
|
|
|
|
|
|
|
#ifdef NEED_GET_PATH_MAX
|
|
|
|
|
unsigned int
|
|
|
|
|
get_path_max ()
|
|
|
|
|
{
|
|
|
|
|
static unsigned int value;
|
|
|
|
|
|
|
|
|
|
if (value == 0)
|
|
|
|
|
{
|
|
|
|
|
long int x = pathconf ("/", _PC_PATH_MAX);
|
|
|
|
|
if (x > 0)
|
|
|
|
|
value = x;
|
|
|
|
|
else
|
|
|
|
|
return MAXPATHLEN;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
1994-07-04 21:46:58 +00:00
|
|
|
|
|
|
|
|
|
/* On some systems, stat can return EINTR. */
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
safe_stat (name, buf)
|
|
|
|
|
char *name;
|
|
|
|
|
struct stat *buf;
|
|
|
|
|
{
|
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
|
|
#ifdef EINTR
|
|
|
|
|
do
|
|
|
|
|
#endif
|
|
|
|
|
ret = stat (name, buf);
|
|
|
|
|
#ifdef EINTR
|
|
|
|
|
while (ret < 0 && errno == EINTR);
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|