1999-03-26 12:04:42 +08:00
|
|
|
|
/* Builtin function expansion for GNU Make.
|
1997-08-28 04:30:54 +08:00
|
|
|
|
Copyright (C) 1988,89,91,92,93,94,95,96,97 Free Software Foundation, Inc.
|
1992-01-11 19:37:36 +08: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
|
1999-03-05 13:55:11 +08:00
|
|
|
|
the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
|
|
|
Boston, MA 02111-1307, USA. */
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
|
|
|
|
#include "make.h"
|
1996-03-20 22:57:41 +08:00
|
|
|
|
#include "filedef.h"
|
1992-01-11 19:37:36 +08:00
|
|
|
|
#include "variable.h"
|
|
|
|
|
#include "dep.h"
|
|
|
|
|
#include "job.h"
|
1996-03-20 22:57:41 +08:00
|
|
|
|
#include "commands.h"
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1996-05-14 02:39:52 +08:00
|
|
|
|
#ifdef _AMIGA
|
|
|
|
|
#include "amiga.h"
|
|
|
|
|
#endif
|
1994-07-26 07:06:00 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
|
|
|
|
|
struct function_table_entry
|
|
|
|
|
{
|
|
|
|
|
const char *name;
|
|
|
|
|
int len;
|
|
|
|
|
int required_arguments;
|
|
|
|
|
int expand_all_arguments;
|
|
|
|
|
char *(*func_ptr) PARAMS((char *output, char **argv, const char*funcname));
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static struct function_table_entry function_table[];
|
|
|
|
|
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
|
|
|
|
/* Store into VARIABLE_BUFFER at O the result of scanning TEXT and replacing
|
|
|
|
|
each occurrence of SUBST with REPLACE. TEXT is null-terminated. SLEN is
|
|
|
|
|
the length of SUBST and RLEN is the length of REPLACE. If BY_WORD is
|
|
|
|
|
nonzero, substitutions are done only on matches which are complete
|
|
|
|
|
whitespace-delimited words. If SUFFIX_ONLY is nonzero, substitutions are
|
|
|
|
|
done only at the ends of whitespace-delimited words. */
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1992-01-11 19:37:36 +08:00
|
|
|
|
char *
|
|
|
|
|
subst_expand (o, text, subst, replace, slen, rlen, by_word, suffix_only)
|
|
|
|
|
char *o;
|
|
|
|
|
char *text;
|
|
|
|
|
char *subst, *replace;
|
|
|
|
|
unsigned int slen, rlen;
|
|
|
|
|
int by_word, suffix_only;
|
|
|
|
|
{
|
|
|
|
|
register char *t = text;
|
|
|
|
|
register char *p;
|
|
|
|
|
|
|
|
|
|
if (slen == 0 && !by_word && !suffix_only)
|
|
|
|
|
{
|
|
|
|
|
/* The first occurrence of "" in any string is its end. */
|
|
|
|
|
o = variable_buffer_output (o, t, strlen (t));
|
|
|
|
|
if (rlen > 0)
|
|
|
|
|
o = variable_buffer_output (o, replace, rlen);
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
|
1994-03-04 13:36:26 +08:00
|
|
|
|
do
|
1992-01-11 19:37:36 +08:00
|
|
|
|
{
|
1994-03-04 13:36:26 +08:00
|
|
|
|
if ((by_word | suffix_only) && slen == 0)
|
|
|
|
|
/* When matching by words, the empty string should match
|
|
|
|
|
the end of each word, rather than the end of the whole text. */
|
|
|
|
|
p = end_of_token (next_token (t));
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
p = sindex (t, 0, subst, slen);
|
|
|
|
|
if (p == 0)
|
|
|
|
|
{
|
|
|
|
|
/* No more matches. Output everything left on the end. */
|
|
|
|
|
o = variable_buffer_output (o, t, strlen (t));
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1992-01-11 19:37:36 +08:00
|
|
|
|
/* Output everything before this occurrence of the string to replace. */
|
|
|
|
|
if (p > t)
|
|
|
|
|
o = variable_buffer_output (o, t, p - t);
|
|
|
|
|
|
|
|
|
|
/* If we're substituting only by fully matched words,
|
|
|
|
|
or only at the ends of words, check that this case qualifies. */
|
|
|
|
|
if ((by_word
|
|
|
|
|
&& ((p > t && !isblank (p[-1]))
|
|
|
|
|
|| (p[slen] != '\0' && !isblank (p[slen]))))
|
|
|
|
|
|| (suffix_only
|
|
|
|
|
&& (p[slen] != '\0' && !isblank (p[slen]))))
|
|
|
|
|
/* Struck out. Output the rest of the string that is
|
|
|
|
|
no longer to be replaced. */
|
|
|
|
|
o = variable_buffer_output (o, subst, slen);
|
|
|
|
|
else if (rlen > 0)
|
|
|
|
|
/* Output the replacement string. */
|
|
|
|
|
o = variable_buffer_output (o, replace, rlen);
|
|
|
|
|
|
|
|
|
|
/* Advance T past the string to be replaced. */
|
|
|
|
|
t = p + slen;
|
1994-03-04 13:36:26 +08:00
|
|
|
|
} while (*t != '\0');
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Store into VARIABLE_BUFFER at O the result of scanning TEXT
|
|
|
|
|
and replacing strings matching PATTERN with REPLACE.
|
|
|
|
|
If PATTERN_PERCENT is not nil, PATTERN has already been
|
|
|
|
|
run through find_percent, and PATTERN_PERCENT is the result.
|
|
|
|
|
If REPLACE_PERCENT is not nil, REPLACE has already been
|
|
|
|
|
run through find_percent, and REPLACE_PERCENT is the result. */
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
patsubst_expand (o, text, pattern, replace, pattern_percent, replace_percent)
|
|
|
|
|
char *o;
|
|
|
|
|
char *text;
|
|
|
|
|
register char *pattern, *replace;
|
|
|
|
|
register char *pattern_percent, *replace_percent;
|
|
|
|
|
{
|
1996-05-10 02:28:15 +08:00
|
|
|
|
unsigned int pattern_prepercent_len, pattern_postpercent_len;
|
1998-07-31 04:54:47 +08:00
|
|
|
|
unsigned int replace_prepercent_len, replace_postpercent_len = 0;
|
1996-05-10 02:28:15 +08:00
|
|
|
|
char *t;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
unsigned int len;
|
|
|
|
|
int doneany = 0;
|
|
|
|
|
|
|
|
|
|
/* We call find_percent on REPLACE before checking PATTERN so that REPLACE
|
|
|
|
|
will be collapsed before we call subst_expand if PATTERN has no %. */
|
|
|
|
|
if (replace_percent == 0)
|
|
|
|
|
replace_percent = find_percent (replace);
|
|
|
|
|
if (replace_percent != 0)
|
|
|
|
|
{
|
|
|
|
|
/* Record the length of REPLACE before and after the % so
|
|
|
|
|
we don't have to compute these lengths more than once. */
|
|
|
|
|
replace_prepercent_len = replace_percent - replace;
|
|
|
|
|
replace_postpercent_len = strlen (replace_percent + 1);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
/* We store the length of the replacement
|
|
|
|
|
so we only need to compute it once. */
|
|
|
|
|
replace_prepercent_len = strlen (replace);
|
|
|
|
|
|
|
|
|
|
if (pattern_percent == 0)
|
|
|
|
|
pattern_percent = find_percent (pattern);
|
|
|
|
|
if (pattern_percent == 0)
|
|
|
|
|
/* With no % in the pattern, this is just a simple substitution. */
|
|
|
|
|
return subst_expand (o, text, pattern, replace,
|
|
|
|
|
strlen (pattern), strlen (replace), 1, 0);
|
|
|
|
|
|
|
|
|
|
/* Record the length of PATTERN before and after the %
|
|
|
|
|
so we don't have to compute it more than once. */
|
|
|
|
|
pattern_prepercent_len = pattern_percent - pattern;
|
|
|
|
|
pattern_postpercent_len = strlen (pattern_percent + 1);
|
|
|
|
|
|
1993-01-07 06:57:35 +08:00
|
|
|
|
while ((t = find_next_token (&text, &len)) != 0)
|
1992-01-11 19:37:36 +08:00
|
|
|
|
{
|
|
|
|
|
int fail = 0;
|
|
|
|
|
|
|
|
|
|
/* Is it big enough to match? */
|
|
|
|
|
if (len < pattern_prepercent_len + pattern_postpercent_len)
|
|
|
|
|
fail = 1;
|
|
|
|
|
|
1997-09-16 22:17:23 +08:00
|
|
|
|
/* Does the prefix match? */
|
1992-01-11 19:37:36 +08:00
|
|
|
|
if (!fail && pattern_prepercent_len > 0
|
|
|
|
|
&& (*t != *pattern
|
|
|
|
|
|| t[pattern_prepercent_len - 1] != pattern_percent[-1]
|
|
|
|
|
|| strncmp (t + 1, pattern + 1, pattern_prepercent_len - 1)))
|
|
|
|
|
fail = 1;
|
|
|
|
|
|
1997-09-16 22:17:23 +08:00
|
|
|
|
/* Does the suffix match? */
|
1992-01-11 19:37:36 +08:00
|
|
|
|
if (!fail && pattern_postpercent_len > 0
|
|
|
|
|
&& (t[len - 1] != pattern_percent[pattern_postpercent_len]
|
|
|
|
|
|| t[len - pattern_postpercent_len] != pattern_percent[1]
|
|
|
|
|
|| strncmp (&t[len - pattern_postpercent_len],
|
|
|
|
|
&pattern_percent[1], pattern_postpercent_len - 1)))
|
|
|
|
|
fail = 1;
|
|
|
|
|
|
|
|
|
|
if (fail)
|
|
|
|
|
/* It didn't match. Output the string. */
|
|
|
|
|
o = variable_buffer_output (o, t, len);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* It matched. Output the replacement. */
|
|
|
|
|
|
|
|
|
|
/* Output the part of the replacement before the %. */
|
|
|
|
|
o = variable_buffer_output (o, replace, replace_prepercent_len);
|
|
|
|
|
|
|
|
|
|
if (replace_percent != 0)
|
|
|
|
|
{
|
|
|
|
|
/* Output the part of the matched string that
|
|
|
|
|
matched the % in the pattern. */
|
|
|
|
|
o = variable_buffer_output (o, t + pattern_prepercent_len,
|
|
|
|
|
len - (pattern_prepercent_len
|
|
|
|
|
+ pattern_postpercent_len));
|
|
|
|
|
/* Output the part of the replacement after the %. */
|
|
|
|
|
o = variable_buffer_output (o, replace_percent + 1,
|
|
|
|
|
replace_postpercent_len);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Output a space, but not if the replacement is "". */
|
|
|
|
|
if (fail || replace_prepercent_len > 0
|
|
|
|
|
|| (replace_percent != 0 && len + replace_postpercent_len > 0))
|
|
|
|
|
{
|
|
|
|
|
o = variable_buffer_output (o, " ", 1);
|
|
|
|
|
doneany = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (doneany)
|
|
|
|
|
/* Kill the last space. */
|
|
|
|
|
--o;
|
|
|
|
|
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Look up a function by name.
|
|
|
|
|
The table is currently small enough that it's not really worthwhile to use
|
|
|
|
|
a fancier lookup algorithm. If it gets larger, maybe...
|
|
|
|
|
*/
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
static const struct function_table_entry *
|
|
|
|
|
lookup_function (table, s)
|
|
|
|
|
const struct function_table_entry *table;
|
|
|
|
|
const char *s;
|
|
|
|
|
{
|
|
|
|
|
int len = strlen(s);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
for (; table->name != NULL; ++table)
|
|
|
|
|
if (table->len <= len
|
|
|
|
|
&& (isblank (s[table->len]) || s[table->len] == '\0')
|
|
|
|
|
&& strneq (s, table->name, table->len))
|
|
|
|
|
return table;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
return NULL;
|
|
|
|
|
}
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
|
1998-07-31 04:54:47 +08:00
|
|
|
|
/* Return 1 if PATTERN matches STR, 0 if not. */
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
|
|
|
|
int
|
1998-07-31 04:54:47 +08:00
|
|
|
|
pattern_matches (pattern, percent, str)
|
|
|
|
|
register char *pattern, *percent, *str;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
{
|
1998-07-31 04:54:47 +08:00
|
|
|
|
unsigned int sfxlen, strlength;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
|
|
|
|
if (percent == 0)
|
|
|
|
|
{
|
|
|
|
|
unsigned int len = strlen (pattern) + 1;
|
1999-03-26 12:04:42 +08:00
|
|
|
|
char *new_chars = (char *) alloca (len);
|
|
|
|
|
bcopy (pattern, new_chars, len);
|
|
|
|
|
pattern = new_chars;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
percent = find_percent (pattern);
|
|
|
|
|
if (percent == 0)
|
1998-07-31 04:54:47 +08:00
|
|
|
|
return streq (pattern, str);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
}
|
|
|
|
|
|
1992-08-20 06:16:15 +08:00
|
|
|
|
sfxlen = strlen (percent + 1);
|
1998-07-31 04:54:47 +08:00
|
|
|
|
strlength = strlen (str);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1998-07-31 04:54:47 +08:00
|
|
|
|
if (strlength < (percent - pattern) + sfxlen
|
|
|
|
|
|| strncmp (pattern, str, percent - pattern))
|
1992-01-11 19:37:36 +08:00
|
|
|
|
return 0;
|
|
|
|
|
|
1998-07-31 04:54:47 +08:00
|
|
|
|
return !strcmp (percent + 1, str + (strlength - sfxlen));
|
1992-01-11 19:37:36 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Find the next comma or ENDPAREN (counting nested STARTPAREN and
|
|
|
|
|
ENDPARENtheses), starting at PTR before END. Return a pointer to
|
|
|
|
|
next character.
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
If no next argument is found, return NULL.
|
|
|
|
|
*/
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
|
|
|
|
static char *
|
1999-03-26 12:04:42 +08:00
|
|
|
|
find_next_argument (startparen, endparen, ptr, end)
|
|
|
|
|
char startparen;
|
|
|
|
|
char endparen;
|
|
|
|
|
const char *ptr;
|
|
|
|
|
const char *end;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
{
|
1999-03-26 12:04:42 +08:00
|
|
|
|
int count = 0;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
for (; ptr < end; ++ptr)
|
|
|
|
|
if (*ptr == startparen)
|
|
|
|
|
++count;
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
else if (*ptr == endparen)
|
1992-01-11 19:37:36 +08:00
|
|
|
|
{
|
1999-03-26 12:04:42 +08:00
|
|
|
|
--count;
|
|
|
|
|
if (count < 0)
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
else if (*ptr == ',' && !count)
|
|
|
|
|
return (char *)ptr;
|
1997-09-16 22:17:23 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* We didn't find anything. */
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
1997-09-16 22:17:23 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
static char *
|
|
|
|
|
expand_builtin_function (o, argc, argv, entry_p)
|
|
|
|
|
char *o;
|
|
|
|
|
int argc;
|
|
|
|
|
char **argv;
|
|
|
|
|
struct function_table_entry *entry_p;
|
|
|
|
|
{
|
|
|
|
|
if (argc < entry_p->required_arguments && entry_p->required_arguments >= 0)
|
|
|
|
|
fatal (reading_file,
|
|
|
|
|
"Insufficient number of arguments (%d) to function `%s'",
|
|
|
|
|
argc, entry_p->name);
|
1997-09-16 22:17:23 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
if (!entry_p->func_ptr)
|
|
|
|
|
fatal (reading_file, "Unimplemented on this platform: function `%s'",
|
|
|
|
|
entry_p->name);
|
1997-09-16 22:17:23 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
return entry_p->func_ptr (o, argv, entry_p->name);
|
|
|
|
|
}
|
1997-09-16 22:17:23 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Check for a function invocation in *STRINGP. *STRINGP points at the
|
|
|
|
|
opening ( or { and is not null-terminated. If a function invocation
|
|
|
|
|
is found, expand it into the buffer at *OP, updating *OP, incrementing
|
|
|
|
|
*STRINGP past the reference and returning nonzero. If not, return zero. */
|
1997-09-16 22:17:23 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
int
|
|
|
|
|
handle_function (op, stringp)
|
|
|
|
|
char **op;
|
|
|
|
|
char **stringp;
|
|
|
|
|
{
|
|
|
|
|
const struct function_table_entry *entry_p;
|
|
|
|
|
char openparen = (*stringp)[0];
|
|
|
|
|
char closeparen = openparen == '(' ? ')' : '}';
|
|
|
|
|
char *beg = *stringp + 1;
|
|
|
|
|
char *endref;
|
|
|
|
|
int count = 0;
|
|
|
|
|
char *argbeg;
|
|
|
|
|
register char *p;
|
|
|
|
|
char **argv, **argvp;
|
|
|
|
|
int nargs;
|
1997-09-16 22:17:23 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
entry_p = lookup_function (function_table, beg);
|
1997-09-16 22:17:23 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
if (!entry_p)
|
|
|
|
|
return 0;
|
1997-09-16 22:17:23 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* We have found a call to a builtin function. Find the end of the
|
|
|
|
|
arguments, and do the function. */
|
1997-04-07 15:21:16 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
endref = beg + entry_p->len;
|
1993-07-15 08:22:55 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Space after function name isn't part of the args. */
|
|
|
|
|
p = next_token (endref);
|
|
|
|
|
argbeg = p;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Find the end of the function invocation, counting nested use of
|
|
|
|
|
whichever kind of parens we use. Since we're looking, count commas
|
|
|
|
|
to get a rough estimate of how many arguments we might have. The
|
|
|
|
|
count might be high, but it'll never be low. */
|
1997-04-07 15:21:16 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
for (nargs=1; *p != '\0'; ++p)
|
|
|
|
|
if (*p == ',')
|
|
|
|
|
++nargs;
|
|
|
|
|
else if (*p == openparen)
|
|
|
|
|
++count;
|
|
|
|
|
else if (*p == closeparen && --count < 0)
|
|
|
|
|
break;
|
1993-07-15 08:22:55 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
if (count >= 0)
|
|
|
|
|
fatal (reading_file,
|
|
|
|
|
"unterminated call to function `%s': missing `%c'",
|
|
|
|
|
entry_p->name, closeparen);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Get some memory to store the arg pointers. */
|
|
|
|
|
argvp = argv = (char **) alloca (sizeof(char *) * (nargs + 2));
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Chop the string into arguments, then store the end pointer and a nul. */
|
|
|
|
|
*argvp = argbeg;
|
|
|
|
|
nargs = 1;
|
|
|
|
|
while (1)
|
|
|
|
|
{
|
|
|
|
|
char *next = find_next_argument (openparen, closeparen, *argvp, p);
|
Thu May 9 13:54:49 1996 Roland McGrath <roland@delasyd.gnu.ai.mit.edu>
* GNUmakefile (globfiles): Add AmigaDOS support files.
(distfiles): Add $(amigafiles).
(amigafiles): New variable.
Thu Nov 7 10:18:16 1995 Aaron Digulla <digulla@fh-konstanz.de>
* Added Amiga support in commands.c, dir.c, function.c,
job.c, main.c, make.h, read.c, remake.c
* commands.c: Amiga has neither SIGHUP nor SIGQUIT
* dir.c: Amiga has filenames with Upper- and Lowercase,
but "FileName" is the same as "filename". Added strieq()
which is use to compare filenames. This is like streq()
on all other systems. Also there is no such thing as
"." under AmigaDOS.
* function.c: On Amiga, the environment is not passed as envp,
there are no pipes and Amiga can't fork. Use my own function
to create a new child.
* job.c: default_shell is "" (The system automatically chooses
a shell for me). Have to use the same workaround as MSDOS for
running batch commands. Added HAVE_SYS_PARAM_H. NOFILE isn't
known on Amiga. Cloned code to run children from MSDOS. Own
version of sh_chars[] and sh_cmds[]. No dup2() or dup() on Amiga.
* main.c: Force stack to 20000 bytes. Read environment from ENV:
device. On Amiga, exec_command() does return, so I exit()
afterwards.
* make.h: Added strieq() to compare filenames.
* read.c: Amiga needs special extension to have passwd. Only
one include-dir. "Makefile" and "makefile" are the same.
Added "SMakefile". Added special code to handle device names (xxx:)
and "./" in rules.
* remake.c: Only one lib-dir. Amiga link-libs are named "%s.lib"
instead of "lib%s.a".
* main.c, rule.c, variable.c: Avoid floats at all costs.
* vpath.c: Get rid of as many alloca()s as possible.
1996-05-10 02:02:06 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
if (!next)
|
1992-01-11 19:37:36 +08:00
|
|
|
|
break;
|
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
*(++argvp) = next+1;
|
|
|
|
|
++nargs;
|
|
|
|
|
}
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
*(++argvp) = p+1;
|
|
|
|
|
*(++argvp) = 0;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* If we should expand, do it. */
|
|
|
|
|
if (entry_p->expand_all_arguments)
|
|
|
|
|
{
|
|
|
|
|
for (argvp=argv; argvp[1] != 0; ++argvp)
|
|
|
|
|
*argvp = expand_argument (*argvp, argvp[1]-1);
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* end pointer doesn't make sense for expanded stuff. */
|
|
|
|
|
*argvp = 0;
|
|
|
|
|
}
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Finally! Run the function... */
|
|
|
|
|
*op = expand_builtin_function (*op, nargs, argv, entry_p);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* If we allocated memory for the expanded args, free it again. */
|
|
|
|
|
if (entry_p->expand_all_arguments)
|
|
|
|
|
for (argvp=argv; *argvp != 0; ++argvp)
|
|
|
|
|
free (*argvp);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
*stringp = p;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Glob-expand LINE. The returned pointer is
|
|
|
|
|
only good until the next call to string_glob. */
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
static char *
|
|
|
|
|
string_glob (line)
|
|
|
|
|
char *line;
|
|
|
|
|
{
|
|
|
|
|
static char *result = 0;
|
|
|
|
|
static unsigned int length;
|
|
|
|
|
register struct nameseq *chain;
|
|
|
|
|
register unsigned int idx;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
chain = multi_glob (parse_file_seq
|
|
|
|
|
(&line, '\0', sizeof (struct nameseq),
|
|
|
|
|
/* We do not want parse_file_seq to strip `./'s.
|
|
|
|
|
That would break examples like:
|
|
|
|
|
$(patsubst ./%.c,obj/%.o,$(wildcard ./*.c)). */
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Yep, and the preceding comment triggers a GCC
|
|
|
|
|
warning. Nothing wrong though. */
|
|
|
|
|
0),
|
|
|
|
|
sizeof (struct nameseq));
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
if (result == 0)
|
|
|
|
|
{
|
|
|
|
|
length = 100;
|
|
|
|
|
result = (char *) xmalloc (100);
|
|
|
|
|
}
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
idx = 0;
|
|
|
|
|
while (chain != 0)
|
|
|
|
|
{
|
|
|
|
|
register char *name = chain->name;
|
|
|
|
|
unsigned int len = strlen (name);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
struct nameseq *next = chain->next;
|
|
|
|
|
free ((char *) chain);
|
|
|
|
|
chain = next;
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* multi_glob will pass names without globbing metacharacters
|
|
|
|
|
through as is, but we want only files that actually exist. */
|
|
|
|
|
if (file_exists_p (name))
|
|
|
|
|
{
|
|
|
|
|
if (idx + len + 1 > length)
|
|
|
|
|
{
|
|
|
|
|
length += (len + 1) * 2;
|
|
|
|
|
result = (char *) xrealloc (result, length);
|
|
|
|
|
}
|
|
|
|
|
bcopy (name, &result[idx], len);
|
|
|
|
|
idx += len;
|
|
|
|
|
result[idx++] = ' ';
|
|
|
|
|
}
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
free (name);
|
|
|
|
|
}
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Kill the last space and terminate the string. */
|
|
|
|
|
if (idx == 0)
|
|
|
|
|
result[0] = '\0';
|
|
|
|
|
else
|
|
|
|
|
result[idx - 1] = '\0';
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
Builtin functions
|
|
|
|
|
*/
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
static char *
|
|
|
|
|
func_patsubst (o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
o = patsubst_expand (o, argv[2], argv[0], argv[1], (char *) 0, (char *) 0);
|
|
|
|
|
return o;
|
|
|
|
|
}
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
static char *
|
|
|
|
|
func_join(o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
int doneany = 0;
|
|
|
|
|
|
|
|
|
|
/* Write each word of the first argument directly followed
|
|
|
|
|
by the corresponding word of the second argument.
|
|
|
|
|
If the two arguments have a different number of words,
|
|
|
|
|
the excess words are just output separated by blanks. */
|
|
|
|
|
register char *tp;
|
|
|
|
|
register char *pp;
|
|
|
|
|
char *list1_iterator = argv[0];
|
|
|
|
|
char *list2_iterator = argv[1];
|
|
|
|
|
do
|
|
|
|
|
{
|
|
|
|
|
unsigned int len1, len2;
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
tp = find_next_token (&list1_iterator, &len1);
|
|
|
|
|
if (tp != 0)
|
|
|
|
|
o = variable_buffer_output (o, tp, len1);
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
pp = find_next_token (&list2_iterator, &len2);
|
|
|
|
|
if (pp != 0)
|
|
|
|
|
o = variable_buffer_output (o, pp, len2);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
if (tp != 0 || pp != 0)
|
1992-01-11 19:37:36 +08:00
|
|
|
|
{
|
1999-03-26 12:04:42 +08:00
|
|
|
|
o = variable_buffer_output (o, " ", 1);
|
|
|
|
|
doneany = 1;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
}
|
1999-03-26 12:04:42 +08:00
|
|
|
|
}
|
|
|
|
|
while (tp != 0 || pp != 0);
|
|
|
|
|
if (doneany)
|
|
|
|
|
/* Kill the last blank. */
|
|
|
|
|
--o;
|
|
|
|
|
|
|
|
|
|
return o;
|
|
|
|
|
}
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
static char *
|
|
|
|
|
func_origin(o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
/* Expand the argument. */
|
|
|
|
|
register struct variable *v = lookup_variable (argv[0], strlen (argv[0]));
|
|
|
|
|
if (v == 0)
|
|
|
|
|
o = variable_buffer_output (o, "undefined", 9);
|
|
|
|
|
else
|
|
|
|
|
switch (v->origin)
|
1992-01-11 19:37:36 +08:00
|
|
|
|
{
|
1999-03-26 12:04:42 +08:00
|
|
|
|
default:
|
|
|
|
|
case o_invalid:
|
|
|
|
|
abort ();
|
|
|
|
|
break;
|
|
|
|
|
case o_default:
|
|
|
|
|
o = variable_buffer_output (o, "default", 7);
|
|
|
|
|
break;
|
|
|
|
|
case o_env:
|
|
|
|
|
o = variable_buffer_output (o, "environment", 11);
|
|
|
|
|
break;
|
|
|
|
|
case o_file:
|
|
|
|
|
o = variable_buffer_output (o, "file", 4);
|
|
|
|
|
break;
|
|
|
|
|
case o_env_override:
|
|
|
|
|
o = variable_buffer_output (o, "environment override", 20);
|
|
|
|
|
break;
|
|
|
|
|
case o_command:
|
|
|
|
|
o = variable_buffer_output (o, "command line", 12);
|
|
|
|
|
break;
|
|
|
|
|
case o_override:
|
|
|
|
|
o = variable_buffer_output (o, "override", 8);
|
|
|
|
|
break;
|
|
|
|
|
case o_automatic:
|
|
|
|
|
o = variable_buffer_output (o, "automatic", 9);
|
|
|
|
|
break;
|
|
|
|
|
}
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
return o;
|
|
|
|
|
}
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
#ifdef VMS
|
|
|
|
|
#define IS_PATHSEP(c) ((c) == ']')
|
|
|
|
|
#else
|
|
|
|
|
#ifdef __MSDOS__
|
|
|
|
|
#define IS_PATHSEP(c) ((c) == '/' || (c) == '\\')
|
|
|
|
|
#else
|
|
|
|
|
#define IS_PATHSEP(c) ((c) == '/')
|
|
|
|
|
#endif
|
|
|
|
|
#endif
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
static char *
|
|
|
|
|
func_notdir_suffix(o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
/* Expand the argument. */
|
|
|
|
|
char *list_iterator = argv[0];
|
|
|
|
|
char *p2 =0;
|
|
|
|
|
int doneany =0;
|
|
|
|
|
int len=0;
|
|
|
|
|
|
|
|
|
|
int is_suffix = streq(funcname, "suffix");
|
|
|
|
|
int is_notdir = !is_suffix;
|
|
|
|
|
while ((p2 = find_next_token (&list_iterator, &len)) != 0)
|
|
|
|
|
{
|
|
|
|
|
char *p = p2 + len;
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
while (p >= p2 && (!is_suffix || *p != '.'))
|
1992-01-11 19:37:36 +08:00
|
|
|
|
{
|
1999-03-26 12:04:42 +08:00
|
|
|
|
if (IS_PATHSEP (*p))
|
1997-09-16 22:17:23 +08:00
|
|
|
|
break;
|
1999-03-26 12:04:42 +08:00
|
|
|
|
--p;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (p >= p2)
|
|
|
|
|
{
|
|
|
|
|
if (is_notdir)
|
|
|
|
|
++p;
|
|
|
|
|
else if (*p != '.')
|
|
|
|
|
continue;
|
|
|
|
|
o = variable_buffer_output (o, p, len - (p - p2));
|
|
|
|
|
}
|
|
|
|
|
#if defined(WINDOWS32) || defined(__MSDOS__)
|
|
|
|
|
/* Handle the case of "d:foo/bar". */
|
|
|
|
|
else if (streq(funcname, "notdir") && p2[0] && p2[1] == ':')
|
|
|
|
|
{
|
|
|
|
|
p = p2 + 2;
|
|
|
|
|
o = variable_buffer_output (o, p, len - (p - p2));
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
else if (is_notdir)
|
|
|
|
|
o = variable_buffer_output (o, p2, len);
|
|
|
|
|
|
|
|
|
|
if (is_notdir || p >= p2)
|
|
|
|
|
{
|
1992-01-11 19:37:36 +08:00
|
|
|
|
o = variable_buffer_output (o, " ", 1);
|
|
|
|
|
doneany = 1;
|
|
|
|
|
}
|
1999-03-26 12:04:42 +08:00
|
|
|
|
}
|
|
|
|
|
if (doneany)
|
|
|
|
|
/* Kill last space. */
|
|
|
|
|
--o;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return o;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
|
func_basename_dir(o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
/* Expand the argument. */
|
|
|
|
|
char *p3 = argv[0];
|
|
|
|
|
char *p2=0;
|
|
|
|
|
int doneany=0;
|
|
|
|
|
int len=0;
|
|
|
|
|
char *p=0;
|
|
|
|
|
int is_basename= streq(funcname, "basename");
|
|
|
|
|
int is_dir= !is_basename;
|
|
|
|
|
|
|
|
|
|
while ((p2 = find_next_token (&p3, &len)) != 0)
|
|
|
|
|
{
|
|
|
|
|
p = p2 + len;
|
|
|
|
|
while (p >= p2 && (!is_basename || *p != '.'))
|
|
|
|
|
{
|
|
|
|
|
if (IS_PATHSEP(*p))
|
|
|
|
|
break;
|
|
|
|
|
--p;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (p >= p2 && (is_dir))
|
|
|
|
|
o = variable_buffer_output (o, p2, ++p - p2);
|
|
|
|
|
else if (p >= p2 && (*p == '.'))
|
|
|
|
|
o = variable_buffer_output (o, p2, p - p2);
|
|
|
|
|
#if defined(WINDOWS32) || defined(__MSDOS__)
|
|
|
|
|
/* Handle the "d:foobar" case */
|
|
|
|
|
else if (p2[0] && p2[1] == ':' && is_dir)
|
|
|
|
|
o = variable_buffer_output (o, p2, 2);
|
|
|
|
|
#endif
|
|
|
|
|
else if (is_dir)
|
|
|
|
|
#ifdef VMS
|
|
|
|
|
o = variable_buffer_output (o, "[]", 2);
|
|
|
|
|
#else
|
|
|
|
|
#ifndef _AMIGA
|
|
|
|
|
o = variable_buffer_output (o, "./", 2);
|
|
|
|
|
#else
|
|
|
|
|
; /* Just a nop... */
|
|
|
|
|
#endif /* AMIGA */
|
|
|
|
|
#endif /* !VMS */
|
|
|
|
|
else
|
|
|
|
|
/* The entire name is the basename. */
|
|
|
|
|
o = variable_buffer_output (o, p2, len);
|
|
|
|
|
|
|
|
|
|
o = variable_buffer_output (o, " ", 1);
|
|
|
|
|
doneany = 1;
|
|
|
|
|
}
|
|
|
|
|
if (doneany)
|
|
|
|
|
/* Kill last space. */
|
1992-01-11 19:37:36 +08:00
|
|
|
|
--o;
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
|
func_addsuffix_addprefix(o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
int fixlen = strlen (argv[0]);
|
|
|
|
|
char *list_iterator = argv[1];
|
|
|
|
|
int is_addprefix = streq (funcname, "addprefix");
|
|
|
|
|
int is_addsuffix = !is_addprefix;
|
|
|
|
|
|
|
|
|
|
int doneany =0;
|
|
|
|
|
char *p=0;
|
|
|
|
|
int len =0;
|
|
|
|
|
|
|
|
|
|
while ((p = find_next_token (&list_iterator, &len)) != 0)
|
|
|
|
|
{
|
|
|
|
|
if (is_addprefix)
|
|
|
|
|
o = variable_buffer_output (o, argv[0], fixlen);
|
|
|
|
|
o = variable_buffer_output (o, p, len);
|
|
|
|
|
if (is_addsuffix)
|
|
|
|
|
o = variable_buffer_output (o, argv[0], fixlen);
|
|
|
|
|
o = variable_buffer_output (o, " ", 1);
|
|
|
|
|
doneany = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (doneany)
|
|
|
|
|
/* Kill last space. */
|
|
|
|
|
--o;
|
|
|
|
|
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
|
func_subst(o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
o = subst_expand (o, argv[2], argv[0], argv[1], strlen (argv[0]),
|
|
|
|
|
strlen (argv[1]), 0, 0);
|
|
|
|
|
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
|
func_firstword(o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
int i=0;
|
|
|
|
|
char *words = argv[0];
|
|
|
|
|
char *p = find_next_token (&words, &i);
|
|
|
|
|
|
|
|
|
|
if (p != 0)
|
|
|
|
|
o = variable_buffer_output (o, p, i);
|
|
|
|
|
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
|
func_words(o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
int i = 0;
|
|
|
|
|
char *word_iterator = argv[0];
|
|
|
|
|
char buf[20];
|
|
|
|
|
|
|
|
|
|
while (find_next_token (&word_iterator, (unsigned int *) 0) != 0)
|
|
|
|
|
++i;
|
|
|
|
|
|
|
|
|
|
sprintf (buf, "%d", i);
|
|
|
|
|
o = variable_buffer_output (o, buf, strlen (buf));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
strip_whitespace (begpp, endpp)
|
|
|
|
|
char **begpp;
|
|
|
|
|
char **endpp;
|
|
|
|
|
{
|
|
|
|
|
while (isspace (**begpp) && *begpp <= *endpp)
|
|
|
|
|
(*begpp) ++;
|
|
|
|
|
while (isspace (**endpp) && *endpp >= *begpp)
|
|
|
|
|
(*endpp) --;
|
|
|
|
|
return *begpp;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
is_numeric (p)
|
|
|
|
|
char *p;
|
|
|
|
|
{
|
|
|
|
|
char *end = p + strlen (p) -1;
|
|
|
|
|
char *beg = p;
|
|
|
|
|
strip_whitespace (&p, &end);
|
|
|
|
|
while (p <= end)
|
|
|
|
|
{
|
|
|
|
|
if (!isdigit (*p))
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
p++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return (end - beg >= 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
check_numeric (s, message)
|
|
|
|
|
char *s;
|
|
|
|
|
char *message;
|
|
|
|
|
{
|
|
|
|
|
if (!is_numeric (s))
|
|
|
|
|
fatal (reading_file, message);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
|
func_word(o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
char *end_p=0;
|
|
|
|
|
int i=0;
|
|
|
|
|
char *p=0;
|
|
|
|
|
|
|
|
|
|
/* Check the first argument. */
|
|
|
|
|
check_numeric (argv[0], "non-numeric first argument to `word' function");
|
|
|
|
|
i = atoi (argv[0]);
|
|
|
|
|
|
|
|
|
|
if (i == 0)
|
|
|
|
|
fatal (reading_file, "the `word' function takes a positive index argument");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
end_p = argv[1];
|
|
|
|
|
while ((p = find_next_token (&end_p, 0)) != 0)
|
|
|
|
|
if (--i == 0)
|
1992-01-11 19:37:36 +08:00
|
|
|
|
break;
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
if (i == 0)
|
|
|
|
|
o = variable_buffer_output (o, p, end_p - p);
|
|
|
|
|
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
|
func_wordlist (o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
int i=0;
|
|
|
|
|
int j=0;
|
|
|
|
|
|
|
|
|
|
/* Check the first argument. */
|
|
|
|
|
check_numeric (argv[0],
|
|
|
|
|
"non-numeric first argument to `wordlist' function");
|
|
|
|
|
i =atoi(argv[0]);
|
|
|
|
|
check_numeric (argv[1],
|
|
|
|
|
"non-numeric second argument to `wordlist' function");
|
|
|
|
|
|
|
|
|
|
j = atoi(argv[1]);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
char *p;
|
|
|
|
|
char *end_p = argv[2];
|
|
|
|
|
|
|
|
|
|
int start = (i < j) ? i : j;
|
|
|
|
|
int count = j -i ;
|
|
|
|
|
if (count < 0)
|
|
|
|
|
count = - count;
|
|
|
|
|
count ++;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
while (((p = find_next_token (&end_p, 0)) != 0) && --start)
|
|
|
|
|
{}
|
|
|
|
|
if (p)
|
|
|
|
|
{
|
|
|
|
|
while (--count && (find_next_token (&end_p, 0) != 0))
|
|
|
|
|
{}
|
|
|
|
|
o = variable_buffer_output (o, p, end_p - p);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static char*
|
|
|
|
|
func_findstring(o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
/* Find the first occurrence of the first string in the second. */
|
|
|
|
|
int i = strlen (argv[0]);
|
|
|
|
|
if (sindex (argv[1], 0, argv[0], i) != 0)
|
|
|
|
|
o = variable_buffer_output (o, argv[0], i);
|
|
|
|
|
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
|
func_foreach (o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
/* expand only the first two. */
|
|
|
|
|
char *varname = expand_argument (argv[0], argv[1] -1);
|
|
|
|
|
char *list = expand_argument (argv[1], argv[2] -1);
|
|
|
|
|
char *body = savestring (argv[2], argv[3] - argv[2] -1 );
|
|
|
|
|
|
|
|
|
|
int len =0;
|
|
|
|
|
char *list_iterator = list;
|
|
|
|
|
char *p;
|
|
|
|
|
register struct variable *var=0;
|
|
|
|
|
int doneany =0;
|
|
|
|
|
|
|
|
|
|
push_new_variable_scope ();
|
|
|
|
|
var = define_variable (varname, strlen (varname), "", o_automatic, 0);
|
|
|
|
|
|
|
|
|
|
/* loop through LIST, put the value in VAR and expand BODY */
|
|
|
|
|
while ((p = find_next_token (&list_iterator, &len)) != 0)
|
|
|
|
|
{
|
|
|
|
|
char *result = 0;
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
char save = p[len];
|
|
|
|
|
|
|
|
|
|
p[len] = '\0';
|
|
|
|
|
free (var->value);
|
|
|
|
|
var->value = (char *) xstrdup ((char*) p);
|
|
|
|
|
p[len] = save;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result = allocated_variable_expand (body);
|
|
|
|
|
|
|
|
|
|
o = variable_buffer_output (o, result, strlen (result));
|
|
|
|
|
o = variable_buffer_output (o, " ", 1);
|
|
|
|
|
doneany = 1;
|
|
|
|
|
free (result);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (doneany)
|
|
|
|
|
/* Kill the last space. */
|
|
|
|
|
--o;
|
|
|
|
|
|
|
|
|
|
pop_variable_scope ();
|
|
|
|
|
free (varname);
|
|
|
|
|
free (list);
|
|
|
|
|
free (body);
|
|
|
|
|
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct a_word
|
|
|
|
|
{
|
|
|
|
|
struct a_word *next;
|
|
|
|
|
char *str;
|
|
|
|
|
int matched;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
|
func_filter_filterout (o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
struct a_word *wordhead =0;
|
|
|
|
|
struct a_word *wordtail =0;
|
|
|
|
|
|
|
|
|
|
int is_filter = streq (funcname, "filter");
|
|
|
|
|
char *patterns = argv[0];
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
char *p;
|
|
|
|
|
int len;
|
|
|
|
|
char *word_iterator = argv[1];
|
|
|
|
|
|
|
|
|
|
/* Chop ARGV[1] up into words and then run each pattern through. */
|
|
|
|
|
while ((p = find_next_token (&word_iterator, &len)) != 0)
|
|
|
|
|
{
|
|
|
|
|
struct a_word *w = (struct a_word *)alloca(sizeof(struct a_word));
|
|
|
|
|
if (wordhead == 0)
|
|
|
|
|
wordhead = w;
|
|
|
|
|
else
|
|
|
|
|
wordtail->next = w;
|
|
|
|
|
wordtail = w;
|
|
|
|
|
|
|
|
|
|
if (*word_iterator != '\0')
|
|
|
|
|
++word_iterator;
|
|
|
|
|
p[len] = '\0';
|
|
|
|
|
w->str = p;
|
|
|
|
|
w->matched = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (wordhead != 0)
|
|
|
|
|
{
|
|
|
|
|
struct a_word *wp =0;
|
|
|
|
|
char *pat_iterator = patterns;
|
|
|
|
|
int doneany = 0;
|
|
|
|
|
|
|
|
|
|
wordtail->next = 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Run each pattern through the words, killing words. */
|
|
|
|
|
while ((p = find_next_token (&pat_iterator, &len)) != 0)
|
|
|
|
|
{
|
|
|
|
|
char *percent;
|
|
|
|
|
char save = p[len];
|
|
|
|
|
p[len] = '\0';
|
|
|
|
|
|
|
|
|
|
percent = find_percent (p);
|
|
|
|
|
for (wp = wordhead; wp != 0; wp = wp->next)
|
|
|
|
|
wp->matched |= (percent == 0 ? streq (p, wp->str)
|
|
|
|
|
: pattern_matches (p, percent, wp->str));
|
|
|
|
|
|
|
|
|
|
p[len] = save;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Output the words that matched (or didn't, for filter-out). */
|
|
|
|
|
for (wp = wordhead; wp != 0; wp = wp->next)
|
|
|
|
|
if (is_filter ? wp->matched : !wp->matched)
|
|
|
|
|
{
|
|
|
|
|
o = variable_buffer_output (o, wp->str, strlen (wp->str));
|
|
|
|
|
o = variable_buffer_output (o, " ", 1);
|
|
|
|
|
doneany = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (doneany)
|
|
|
|
|
/* Kill the last space. */
|
|
|
|
|
--o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
|
func_strip(o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
char *p = argv[0];
|
|
|
|
|
int doneany =0;
|
|
|
|
|
|
|
|
|
|
while (*p != '\0')
|
|
|
|
|
{
|
|
|
|
|
int i=0;
|
|
|
|
|
char *word_start=0;
|
|
|
|
|
|
|
|
|
|
while (isspace(*p))
|
|
|
|
|
++p;
|
|
|
|
|
word_start = p;
|
|
|
|
|
for (i=0; *p != '\0' && !isspace(*p); ++p, ++i)
|
|
|
|
|
{}
|
|
|
|
|
if (!i)
|
|
|
|
|
break;
|
|
|
|
|
o = variable_buffer_output (o, word_start, i);
|
|
|
|
|
o = variable_buffer_output (o, " ", 1);
|
|
|
|
|
doneany = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (doneany)
|
|
|
|
|
/* Kill the last space. */
|
|
|
|
|
--o;
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
Print a warning or fatal message.
|
|
|
|
|
*/
|
|
|
|
|
static char *
|
|
|
|
|
func_error (o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
char **argvp;
|
|
|
|
|
char *msg, *p;
|
|
|
|
|
int len;
|
|
|
|
|
|
|
|
|
|
/* The arguments will be broken on commas. Rather than create yet
|
|
|
|
|
another special case where function arguments aren't broken up,
|
|
|
|
|
just create a format string that puts them back together. */
|
|
|
|
|
for (len=0, argvp=argv; *argvp != 0; ++argvp)
|
|
|
|
|
len += strlen(*argvp) + 2;
|
|
|
|
|
|
|
|
|
|
p = msg = alloca (len + 1);
|
|
|
|
|
|
|
|
|
|
for (argvp=argv; argvp[1] != 0; ++argvp)
|
|
|
|
|
{
|
|
|
|
|
strcpy(p, *argvp);
|
|
|
|
|
p += strlen(*argvp);
|
|
|
|
|
*(p++) = ',';
|
|
|
|
|
*(p++) = ' ';
|
|
|
|
|
}
|
|
|
|
|
strcpy(p, *argvp);
|
|
|
|
|
|
|
|
|
|
if (*funcname == 'e')
|
|
|
|
|
fatal (reading_file, "%s", msg);
|
|
|
|
|
|
|
|
|
|
/* The warning function expands to the empty string. */
|
|
|
|
|
error (reading_file, "%s", msg);
|
|
|
|
|
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
chop argv[0] into words, and sort them.
|
|
|
|
|
*/
|
|
|
|
|
static char *
|
|
|
|
|
func_sort (o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
char **words = 0;
|
|
|
|
|
int nwords = 0;
|
|
|
|
|
register int wordi = 0;
|
|
|
|
|
|
|
|
|
|
/* Chop ARGV[0] into words and put them in WORDS. */
|
|
|
|
|
char *t = argv[0];
|
|
|
|
|
char *p=0;
|
|
|
|
|
int len;
|
|
|
|
|
while ((p = find_next_token (&t, &len)) != 0)
|
|
|
|
|
{
|
|
|
|
|
if (wordi >= nwords - 1)
|
|
|
|
|
{
|
|
|
|
|
nwords = 2* nwords + 5;
|
|
|
|
|
words = (char **) xrealloc ((char *) words,
|
|
|
|
|
nwords * sizeof (char *));
|
|
|
|
|
}
|
|
|
|
|
words[wordi++] = savestring (p, len);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (wordi > 0)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
/* Now sort the list of words. */
|
|
|
|
|
qsort ((char *) words, wordi, sizeof (char *), alpha_compare);
|
|
|
|
|
|
|
|
|
|
/* Now write the sorted list. */
|
|
|
|
|
for (i = 0; i < wordi; ++i)
|
|
|
|
|
{
|
|
|
|
|
len = strlen (words[i]);
|
|
|
|
|
if (i == wordi - 1 || strlen (words[i + 1]) != len
|
|
|
|
|
|| strcmp (words[i], words[i + 1]))
|
|
|
|
|
{
|
|
|
|
|
o = variable_buffer_output (o, words[i], len);
|
|
|
|
|
o = variable_buffer_output (o, " ", 1);
|
|
|
|
|
}
|
|
|
|
|
free (words[i]);
|
|
|
|
|
}
|
|
|
|
|
/* Kill the last space. */
|
|
|
|
|
--o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
free ((char *) words);
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
|
func_wildcard(o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1996-05-14 02:39:52 +08:00
|
|
|
|
#ifdef _AMIGA
|
1999-03-26 12:04:42 +08:00
|
|
|
|
o = wildcard_expansion (argv[0], o);
|
1996-05-14 02:39:52 +08:00
|
|
|
|
#else
|
1999-03-26 12:04:42 +08:00
|
|
|
|
char *p = string_glob (argv[0]);
|
|
|
|
|
o = variable_buffer_output (o, p, strlen (p));
|
1996-05-14 02:39:52 +08:00
|
|
|
|
#endif
|
1999-03-26 12:04:42 +08:00
|
|
|
|
return o;
|
|
|
|
|
}
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/*
|
|
|
|
|
\r is replaced on UNIX as well. Is this desirable?
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
fold_newlines (buffer, length)
|
|
|
|
|
char *buffer;
|
|
|
|
|
int *length;
|
|
|
|
|
{
|
|
|
|
|
char *dst = buffer;
|
|
|
|
|
char *src = buffer;
|
|
|
|
|
char *last_nonnl = buffer -1;
|
|
|
|
|
src[*length] = 0;
|
|
|
|
|
for (; *src != '\0'; ++src)
|
|
|
|
|
{
|
|
|
|
|
if (src[0] == '\r' && src[1] == '\n')
|
|
|
|
|
continue;
|
|
|
|
|
if (*src == '\n')
|
|
|
|
|
{
|
|
|
|
|
*dst++ = ' ';
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
last_nonnl = dst;
|
|
|
|
|
*dst++ = *src;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
*(++last_nonnl) = '\0';
|
|
|
|
|
*length = last_nonnl - buffer;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int shell_function_pid = 0, shell_function_completed;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef WINDOWS32
|
|
|
|
|
/*untested*/
|
|
|
|
|
|
|
|
|
|
#include <windows.h>
|
|
|
|
|
#include <io.h>
|
|
|
|
|
#include "sub_proc.h"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
windows32_openpipe (int *pipedes, int *pid_p, char **command_argv, char **envp)
|
|
|
|
|
{
|
|
|
|
|
SECURITY_ATTRIBUTES saAttr;
|
|
|
|
|
HANDLE hIn;
|
|
|
|
|
HANDLE hErr;
|
|
|
|
|
HANDLE hChildOutRd;
|
|
|
|
|
HANDLE hChildOutWr;
|
|
|
|
|
HANDLE hProcess;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
saAttr.nLength = sizeof(SECURITY_ATTRIBUTES);
|
|
|
|
|
saAttr.bInheritHandle = TRUE;
|
|
|
|
|
saAttr.lpSecurityDescriptor = NULL;
|
|
|
|
|
|
|
|
|
|
if (DuplicateHandle(GetCurrentProcess(),
|
|
|
|
|
GetStdHandle(STD_INPUT_HANDLE),
|
|
|
|
|
GetCurrentProcess(),
|
|
|
|
|
&hIn,
|
|
|
|
|
0,
|
|
|
|
|
TRUE,
|
|
|
|
|
DUPLICATE_SAME_ACCESS) == FALSE) {
|
|
|
|
|
fatal (NILF, "create_child_process: DuplicateHandle(In) failed (e=%d)\n",
|
|
|
|
|
GetLastError());
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
if (DuplicateHandle(GetCurrentProcess(),
|
|
|
|
|
GetStdHandle(STD_ERROR_HANDLE),
|
|
|
|
|
GetCurrentProcess(),
|
|
|
|
|
&hErr,
|
|
|
|
|
0,
|
|
|
|
|
TRUE,
|
|
|
|
|
DUPLICATE_SAME_ACCESS) == FALSE) {
|
|
|
|
|
fatal (NILF, "create_child_process: DuplicateHandle(Err) failed (e=%d)\n",
|
|
|
|
|
GetLastError());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!CreatePipe(&hChildOutRd, &hChildOutWr, &saAttr, 0))
|
|
|
|
|
fatal (NILF, "CreatePipe() failed (e=%d)\n", GetLastError());
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
hProcess = process_init_fd(hIn, hChildOutWr, hErr);
|
|
|
|
|
|
|
|
|
|
if (!hProcess)
|
|
|
|
|
fatal (NILF, "windows32_openpipe (): process_init_fd() failed\n");
|
|
|
|
|
|
|
|
|
|
else
|
|
|
|
|
process_register(hProcess);
|
|
|
|
|
|
|
|
|
|
/* make sure that CreateProcess() has Path it needs */
|
|
|
|
|
sync_Path_environment();
|
|
|
|
|
|
|
|
|
|
if (!process_begin(hProcess, command_argv, envp, command_argv[0], NULL))
|
|
|
|
|
*pid_p = (int) hProcess;
|
|
|
|
|
else
|
|
|
|
|
fatal (NILF, "windows32_openpipe (): unable to launch process (e=%d)\n",
|
|
|
|
|
process_last_err(hProcess));
|
|
|
|
|
|
|
|
|
|
/* set up to read data from child */
|
|
|
|
|
pipedes[0] = _open_osfhandle((long) hChildOutRd, O_RDONLY);
|
|
|
|
|
|
|
|
|
|
/* this will be closed almost right away */
|
|
|
|
|
pipedes[1] = _open_osfhandle((long) hChildOutWr, O_APPEND);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef __MSDOS__
|
|
|
|
|
/*
|
|
|
|
|
untested
|
|
|
|
|
*/
|
|
|
|
|
int
|
|
|
|
|
msdos_openpipe (int* pipedes, int *pidp, char *text)
|
|
|
|
|
{
|
|
|
|
|
FILE *fpipe=0;
|
|
|
|
|
/* MSDOS can't fork, but it has `popen'.
|
|
|
|
|
(Bwt, why isn't `popen' used in all the versions?) */
|
|
|
|
|
struct variable *sh = lookup_variable ("SHELL", 5);
|
|
|
|
|
int e;
|
|
|
|
|
extern int dos_command_running, dos_status;
|
|
|
|
|
|
|
|
|
|
/* Make sure not to bother processing an empty line. */
|
|
|
|
|
while (isblank (*text))
|
|
|
|
|
++text;
|
|
|
|
|
if (*text == '\0')
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
if (sh)
|
|
|
|
|
{
|
|
|
|
|
char buf[PATH_MAX + 7];
|
|
|
|
|
/* This makes sure $SHELL value is used by $(shell), even
|
|
|
|
|
though the target environment is not passed to it. */
|
|
|
|
|
sprintf (buf, "SHELL=%s", sh->value);
|
|
|
|
|
putenv (buf);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
e = errno;
|
|
|
|
|
errno = 0;
|
|
|
|
|
dos_command_running = 1;
|
|
|
|
|
dos_status = 0;
|
|
|
|
|
fpipe = popen (text, "rt");
|
|
|
|
|
dos_command_running = 0;
|
|
|
|
|
if (!fpipe || dos_status)
|
|
|
|
|
{
|
|
|
|
|
pipedes[0] = -1;
|
|
|
|
|
*pidp = -1;
|
|
|
|
|
if (dos_status)
|
|
|
|
|
errno = EINTR;
|
|
|
|
|
else if (errno == 0)
|
|
|
|
|
errno = ENOMEM;
|
|
|
|
|
shell_function_completed = -1;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
pipedes[0] = fileno (fpipe);
|
|
|
|
|
*pidp = 42; /* uh? The meaning of Life?*/
|
|
|
|
|
errno = e;
|
|
|
|
|
shell_function_completed = 1;
|
|
|
|
|
}
|
|
|
|
|
return fpipe;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
Do shell spawning, with the naughty bits for different OSes.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
|
|
|
|
|
|
/* VMS can't do $(shell ...) */
|
|
|
|
|
#define func_shell 0
|
|
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
#ifndef _AMIGA
|
|
|
|
|
static char *
|
|
|
|
|
func_shell (o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
char* batch_filename = NULL;
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
#ifdef __MSDOS__
|
|
|
|
|
FILE *fpipe;
|
|
|
|
|
#endif
|
|
|
|
|
char **command_argv;
|
|
|
|
|
char *error_prefix;
|
|
|
|
|
char **envp;
|
|
|
|
|
int pipedes[2];
|
|
|
|
|
int pid;
|
|
|
|
|
|
|
|
|
|
#ifndef __MSDOS__
|
|
|
|
|
/* Construct the argument list. */
|
|
|
|
|
command_argv = construct_command_argv (argv[0],
|
|
|
|
|
(char **) NULL, (struct file *) 0,
|
|
|
|
|
&batch_filename);
|
|
|
|
|
if (command_argv == 0)
|
|
|
|
|
return o;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
/* Using a target environment for `shell' loses in cases like:
|
|
|
|
|
export var = $(shell echo foobie)
|
|
|
|
|
because target_environment hits a loop trying to expand $(var)
|
|
|
|
|
to put it in the environment. This is even more confusing when
|
|
|
|
|
var was not explicitly exported, but just appeared in the
|
|
|
|
|
calling environment. */
|
|
|
|
|
|
|
|
|
|
envp = environ;
|
|
|
|
|
|
|
|
|
|
/* For error messages. */
|
|
|
|
|
if (reading_file != 0)
|
|
|
|
|
{
|
|
|
|
|
error_prefix = (char *) alloca (strlen(reading_file->filenm)+100);
|
|
|
|
|
|
|
|
|
|
sprintf (error_prefix,
|
|
|
|
|
"%s:%lu: ", reading_file->filenm, reading_file->lineno);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
error_prefix = "";
|
|
|
|
|
|
|
|
|
|
#ifdef WINDOWS32
|
|
|
|
|
windows32_openpipe (pipedes, &pid, command_argv, envp);
|
|
|
|
|
#else /* WINDOWS32 */
|
|
|
|
|
|
|
|
|
|
# ifdef __MSDOS__
|
|
|
|
|
fpipe = msdos_openpipe (pipedes, argv[0], command_argv, envp);
|
|
|
|
|
if (!fpipe || pipedes[0] < 0)
|
|
|
|
|
{
|
|
|
|
|
perror_with_name (error_prefix, "pipe");
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
# else
|
|
|
|
|
if (pipe (pipedes) < 0)
|
|
|
|
|
{
|
|
|
|
|
perror_with_name (error_prefix, "pipe");
|
|
|
|
|
return o;
|
|
|
|
|
}
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
pid = vfork ();
|
|
|
|
|
if (pid < 0)
|
|
|
|
|
perror_with_name (error_prefix, "fork");
|
|
|
|
|
else if (pid == 0)
|
|
|
|
|
child_execute_job (0, pipedes[1], command_argv, envp);
|
|
|
|
|
else
|
|
|
|
|
# endif /* ! __MSDOS__ */
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
#endif /* WINDOWS32 */
|
|
|
|
|
{
|
|
|
|
|
/* We are the parent. */
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
char *buffer;
|
|
|
|
|
unsigned int maxlen;
|
|
|
|
|
int cc;
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Record the PID for reap_children. */
|
|
|
|
|
shell_function_pid = pid;
|
|
|
|
|
#ifndef __MSDOS__
|
|
|
|
|
shell_function_completed = 0;
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Free the storage only the child needed. */
|
|
|
|
|
free (command_argv[0]);
|
|
|
|
|
free ((char *) command_argv);
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Close the write side of the pipe. */
|
|
|
|
|
(void) close (pipedes[1]);
|
|
|
|
|
#endif
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Set up and read from the pipe. */
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
maxlen = 200;
|
|
|
|
|
buffer = (char *) xmalloc (maxlen + 1);
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Read from the pipe until it gets EOF. */
|
|
|
|
|
i = 0;
|
|
|
|
|
do
|
1992-01-11 19:37:36 +08:00
|
|
|
|
{
|
1999-03-26 12:04:42 +08:00
|
|
|
|
if (i == maxlen)
|
|
|
|
|
{
|
|
|
|
|
maxlen += 512;
|
|
|
|
|
buffer = (char *) xrealloc (buffer, maxlen + 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
errno = 0;
|
|
|
|
|
cc = read (pipedes[0], &buffer[i], maxlen - i);
|
|
|
|
|
if (cc > 0)
|
|
|
|
|
i += cc;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
}
|
1999-03-26 12:04:42 +08:00
|
|
|
|
#ifdef EINTR
|
|
|
|
|
while (cc > 0 || errno == EINTR);
|
|
|
|
|
#else
|
|
|
|
|
while (cc > 0);
|
|
|
|
|
#endif
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Close the read side of the pipe. */
|
|
|
|
|
#ifdef __MSDOS__
|
|
|
|
|
if (fpipe)
|
|
|
|
|
(void) pclose (fpipe);
|
|
|
|
|
#else
|
|
|
|
|
(void) close (pipedes[0]);
|
|
|
|
|
#endif
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Loop until child_handler sets shell_function_completed
|
|
|
|
|
to the status of our child shell. */
|
|
|
|
|
while (shell_function_completed == 0)
|
|
|
|
|
reap_children (1, 0);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
if (batch_filename) {
|
|
|
|
|
if (debug_flag)
|
|
|
|
|
printf("Cleaning up temporary batch file %s\n", batch_filename);
|
|
|
|
|
remove(batch_filename);
|
|
|
|
|
free(batch_filename);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
}
|
1999-03-26 12:04:42 +08:00
|
|
|
|
shell_function_pid = 0;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* The child_handler function will set shell_function_completed
|
|
|
|
|
to 1 when the child dies normally, or to -1 if it
|
|
|
|
|
dies with status 127, which is most likely an exec fail. */
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
if (shell_function_completed == -1)
|
1997-09-16 22:17:23 +08:00
|
|
|
|
{
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* This most likely means that the execvp failed,
|
|
|
|
|
so we should just write out the error message
|
|
|
|
|
that came in over the pipe from the child. */
|
|
|
|
|
fputs (buffer, stderr);
|
|
|
|
|
fflush (stderr);
|
1997-09-16 22:17:23 +08:00
|
|
|
|
}
|
1997-04-07 15:21:16 +08:00
|
|
|
|
else
|
1997-09-16 22:17:23 +08:00
|
|
|
|
{
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* The child finished normally. Replace all
|
|
|
|
|
newlines in its output with spaces, and put
|
|
|
|
|
that in the variable output buffer. */
|
|
|
|
|
fold_newlines (buffer, &i);
|
|
|
|
|
o = variable_buffer_output (o, buffer, i);
|
1997-09-16 22:17:23 +08:00
|
|
|
|
}
|
1997-04-07 15:21:16 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
free (buffer);
|
|
|
|
|
}
|
1997-04-07 15:21:16 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
return o;
|
|
|
|
|
}
|
1997-04-07 15:21:16 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
#else /* _AMIGA */
|
1997-04-07 15:21:16 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Do the Amiga version of func_shell. */
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
static char *
|
|
|
|
|
func_shell (char *o, char **argv, const char *funcname)
|
|
|
|
|
{
|
|
|
|
|
/* Amiga can't fork nor spawn, but I can start a program with
|
|
|
|
|
redirection of my choice. However, this means that we
|
|
|
|
|
don't have an opportunity to reopen stdout to trap it. Thus,
|
|
|
|
|
we save our own stdout onto a new descriptor and dup a temp
|
|
|
|
|
file's descriptor onto our stdout temporarily. After we
|
|
|
|
|
spawn the shell program, we dup our own stdout back to the
|
|
|
|
|
stdout descriptor. The buffer reading is the same as above,
|
|
|
|
|
except that we're now reading from a file. */
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
#include <dos/dos.h>
|
|
|
|
|
#include <proto/dos.h>
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
BPTR child_stdout;
|
|
|
|
|
char tmp_output[FILENAME_MAX];
|
|
|
|
|
unsigned int maxlen = 200;
|
|
|
|
|
int cc, i;
|
|
|
|
|
char * buffer, * ptr;
|
|
|
|
|
char ** aptr;
|
|
|
|
|
int len = 0;
|
|
|
|
|
char* batch_filename = NULL;
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/* Construct the argument list. */
|
|
|
|
|
command_argv = construct_command_argv (argv[0], (char **) NULL,
|
|
|
|
|
(struct file *) 0, &batch_filename);
|
|
|
|
|
if (command_argv == 0)
|
|
|
|
|
return o;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
strcpy (tmp_output, "t:MakeshXXXXXXXX");
|
|
|
|
|
mktemp (tmp_output);
|
|
|
|
|
child_stdout = Open (tmp_output, MODE_NEWFILE);
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
for (aptr=command_argv; *aptr; aptr++)
|
|
|
|
|
len += strlen (*aptr) + 1;
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
buffer = xmalloc (len + 1);
|
|
|
|
|
ptr = buffer;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
for (aptr=command_argv; *aptr; aptr++)
|
|
|
|
|
{
|
|
|
|
|
strcpy (ptr, *aptr);
|
|
|
|
|
ptr += strlen (ptr) + 1;
|
|
|
|
|
*ptr ++ = ' ';
|
|
|
|
|
*ptr = 0;
|
|
|
|
|
}
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
ptr[-1] = '\n';
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
Execute (buffer, NULL, child_stdout);
|
|
|
|
|
free (buffer);
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
Close (child_stdout);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
child_stdout = Open (tmp_output, MODE_OLDFILE);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
buffer = xmalloc (maxlen);
|
|
|
|
|
i = 0;
|
|
|
|
|
do
|
|
|
|
|
{
|
|
|
|
|
if (i == maxlen)
|
|
|
|
|
{
|
|
|
|
|
maxlen += 512;
|
|
|
|
|
buffer = (char *) xrealloc (buffer, maxlen + 1);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
}
|
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
cc = Read (child_stdout, &buffer[i], maxlen - i);
|
|
|
|
|
if (cc > 0)
|
|
|
|
|
i += cc;
|
|
|
|
|
} while (cc > 0);
|
|
|
|
|
|
|
|
|
|
Close (child_stdout);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
fold_newlines (buffer, &i);
|
|
|
|
|
o = variable_buffer_output (o, buffer, i);
|
|
|
|
|
free (buffer);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
return o;
|
|
|
|
|
}
|
1999-03-26 12:04:42 +08:00
|
|
|
|
#endif /* _AMIGA */
|
|
|
|
|
#endif /* !VMS */
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
#ifdef EXPERIMENTAL
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/*
|
|
|
|
|
equality. Return is string-boolean, ie, the empty string is false.
|
|
|
|
|
*/
|
|
|
|
|
static char *
|
|
|
|
|
func_eq (char* o, char **argv, char *funcname)
|
1992-01-11 19:37:36 +08:00
|
|
|
|
{
|
1999-03-26 12:04:42 +08:00
|
|
|
|
int result = ! strcmp (argv[0], argv[1]);
|
|
|
|
|
o = variable_buffer_output (o, result ? "1" : "", result);
|
|
|
|
|
return o;
|
|
|
|
|
}
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/*
|
|
|
|
|
string-boolean not operator.
|
|
|
|
|
*/
|
|
|
|
|
static char *
|
|
|
|
|
func_not (char* o, char **argv, char *funcname)
|
|
|
|
|
{
|
|
|
|
|
char * s = argv[0];
|
|
|
|
|
int result = 0;
|
|
|
|
|
while (isspace (*s))
|
|
|
|
|
s++;
|
|
|
|
|
result = ! (*s);
|
|
|
|
|
o = variable_buffer_output (o, result ? "1" : "", result);
|
|
|
|
|
return o;
|
|
|
|
|
}
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
/*
|
|
|
|
|
This is an experimental conditional function.
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
Syntax:
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
$(if condition, true-part, false-part)
|
1993-05-08 06:34:49 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
This is fully not consistent with make's syntax, but more in line
|
|
|
|
|
with `normal' programming languages.
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
Semantics:
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
- CONDITION is false iff it evaluates to an empty string. White
|
|
|
|
|
space before and after condition are stripped before evaluation.
|
|
|
|
|
|
|
|
|
|
- If CONDITION is true, then TRUE-PART is evaluated, otherwise
|
|
|
|
|
FALSE-PART is evaluated. Because only one of the two PARTs is
|
|
|
|
|
evaluated, you can use $(if ) to create side-effects with the
|
|
|
|
|
$(shell ) function
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
*/
|
1992-01-11 19:37:36 +08:00
|
|
|
|
static char *
|
1999-03-26 12:04:42 +08:00
|
|
|
|
func_if (char* o, char **argv, char *funcname)
|
1992-01-11 19:37:36 +08:00
|
|
|
|
{
|
1999-03-26 12:04:42 +08:00
|
|
|
|
char *begp = argv[0];
|
|
|
|
|
char *endp = argv[1]-2;
|
|
|
|
|
char *expansion =0;
|
|
|
|
|
int result = 0;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
strip_whitespace (&begp, &endp);
|
|
|
|
|
if(begp <= endp)
|
|
|
|
|
expansion = expand_argument (begp, endp + 1);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
result = expansion
|
|
|
|
|
? strlen (expansion)
|
|
|
|
|
: 0;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
result = !result;
|
|
|
|
|
free (expansion);
|
|
|
|
|
|
|
|
|
|
expansion = expand_argument (argv[1 + result], argv[2+result] -1);
|
|
|
|
|
o = variable_buffer_output (o, expansion, strlen (expansion));
|
|
|
|
|
|
|
|
|
|
return o;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
/* This might not be very useful, but the code was simple to
|
|
|
|
|
implement, I just had to do it.
|
|
|
|
|
|
|
|
|
|
Here goes anyway
|
|
|
|
|
|
|
|
|
|
Apply & User defined functions.
|
|
|
|
|
|
|
|
|
|
SYNTAX
|
|
|
|
|
|
|
|
|
|
$(apply funcname, arg1, arg2, .. )
|
|
|
|
|
|
|
|
|
|
SEMANTICS
|
|
|
|
|
|
|
|
|
|
You can specify a builtin function, for funcname, eg
|
|
|
|
|
|
|
|
|
|
f = addprefix
|
|
|
|
|
$(apply addprefix,a, b c d)
|
|
|
|
|
|
|
|
|
|
This will result in
|
|
|
|
|
|
|
|
|
|
ab ac ad
|
|
|
|
|
|
|
|
|
|
You can specify your own functions, eg
|
|
|
|
|
|
|
|
|
|
funcname=BODY
|
|
|
|
|
|
|
|
|
|
BODY contains $(1) .. $(N) as argument markers.
|
|
|
|
|
upon expansions the strings ARG1 .. ARGN are substituted for $(1) .. $(N)
|
|
|
|
|
into BODY
|
|
|
|
|
|
|
|
|
|
Because the funcname is computed as well you can combine this do some
|
|
|
|
|
funky things, eg
|
|
|
|
|
|
|
|
|
|
map=$(foreach a, $(2), $(apply $(1), $(a)))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
LIMITATIONS.
|
|
|
|
|
|
|
|
|
|
Make has no support for nested lists (or tuples), so you can't do
|
|
|
|
|
stuff like (Haskell notation):
|
|
|
|
|
|
|
|
|
|
f :: (a,b) -> c -- type of F
|
|
|
|
|
map :: (a->b) -> [a] -> b -- type of MAP
|
|
|
|
|
|
|
|
|
|
map f [(1,2), (2,3)] -- map F over list containing (1,2) and (2,3)
|
|
|
|
|
|
|
|
|
|
to get
|
|
|
|
|
|
|
|
|
|
[f (1, 2), f (2, 3)]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
If only we had nested lists and quotes, we could duplicate LISP in make by
|
|
|
|
|
transforming
|
|
|
|
|
|
|
|
|
|
$(a, b, c) <-> (a b c)
|
|
|
|
|
$(quote $(a, b, c)) <-> '(a b c)
|
|
|
|
|
|
|
|
|
|
(or something alike ;-) (We could have automatic integration of
|
|
|
|
|
GUILE and make :-)
|
|
|
|
|
|
|
|
|
|
[Actually -- why should this be a joke? If we could somehow integrate the
|
|
|
|
|
rules and targets into a functional model make could be a lot cleaner in
|
|
|
|
|
concept. ]
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
char *
|
|
|
|
|
func_apply (o, argv, funcname)
|
|
|
|
|
char *o;
|
|
|
|
|
char **argv;
|
|
|
|
|
const char *funcname;
|
|
|
|
|
{
|
|
|
|
|
char *userfunc_name;
|
|
|
|
|
int func_len;
|
|
|
|
|
char *body = 0;
|
|
|
|
|
char *expanded_body = 0;
|
|
|
|
|
int i;
|
|
|
|
|
const struct function_table_entry *entry_p;
|
|
|
|
|
|
|
|
|
|
userfunc_name = argv[0];
|
|
|
|
|
while (isspace (*userfunc_name))
|
|
|
|
|
++userfunc_name;
|
|
|
|
|
|
|
|
|
|
entry_p = lookup_function (function_table, userfunc_name);
|
|
|
|
|
|
|
|
|
|
/* builtin function? */
|
|
|
|
|
if (entry_p)
|
1992-01-11 19:37:36 +08:00
|
|
|
|
{
|
1999-03-26 12:04:42 +08:00
|
|
|
|
for (i=0; argv[i+1]; i++)
|
|
|
|
|
;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
o = expand_builtin_function (o, i, argv + 1, entry_p);
|
|
|
|
|
return o;
|
|
|
|
|
}
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
func_len = strlen (userfunc_name);
|
|
|
|
|
body = xmalloc (func_len + 4);
|
|
|
|
|
strcpy (body + 2, userfunc_name);
|
|
|
|
|
body [func_len+2]=')';
|
|
|
|
|
body [func_len+3]= 0;
|
|
|
|
|
body [1]='(';
|
|
|
|
|
body [0]='$';
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
push_new_variable_scope ();
|
|
|
|
|
|
|
|
|
|
/* set up arguments $(1) .. $(N) */
|
|
|
|
|
for (i=0; argv[i]; i++)
|
|
|
|
|
{
|
|
|
|
|
char num[10];
|
|
|
|
|
struct variable* var;
|
|
|
|
|
sprintf (num, "%d", i);
|
|
|
|
|
var = define_variable (num, strlen (num), argv[i], o_automatic, 0);
|
1992-01-11 19:37:36 +08:00
|
|
|
|
}
|
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
expanded_body = allocated_variable_expand (body);
|
|
|
|
|
o = variable_buffer_output (o, expanded_body, strlen (expanded_body));
|
|
|
|
|
free (expanded_body);
|
|
|
|
|
pop_variable_scope ();
|
1992-01-11 19:37:36 +08:00
|
|
|
|
|
1999-03-26 12:04:42 +08:00
|
|
|
|
free (body);
|
|
|
|
|
return o;
|
1992-01-11 19:37:36 +08:00
|
|
|
|
}
|
1999-03-26 12:04:42 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#define STRING_SIZE_TUPLE(s) (s), (sizeof(s)-1)
|
|
|
|
|
|
|
|
|
|
/* Lookup table for builtin functions.
|
|
|
|
|
|
|
|
|
|
This doesn't have to be sorted; we use a straight lookup. We might gain
|
|
|
|
|
some efficiency by moving most often used functions to the start of the
|
|
|
|
|
table.
|
|
|
|
|
|
|
|
|
|
REQUIRED_ARGUMENTS is the minimum number of arguments. A function
|
|
|
|
|
can have more, but they will be ignored.
|
|
|
|
|
|
|
|
|
|
EXPAND_ALL_ARGUMENTS means that all arguments should be expanded
|
|
|
|
|
before invocation. Functions that do namespace tricks (foreach)
|
|
|
|
|
don't automatically expand. */
|
|
|
|
|
|
|
|
|
|
static struct function_table_entry function_table[] =
|
|
|
|
|
{
|
|
|
|
|
/* Name/size */ /* ARG EXP? Function */
|
|
|
|
|
{ STRING_SIZE_TUPLE("addprefix"), 2, 1, func_addsuffix_addprefix},
|
|
|
|
|
{ STRING_SIZE_TUPLE("addsuffix"), 2, 1, func_addsuffix_addprefix},
|
|
|
|
|
{ STRING_SIZE_TUPLE("basename"), 1, 1, func_basename_dir},
|
|
|
|
|
{ STRING_SIZE_TUPLE("dir"), 1, 1, func_basename_dir},
|
|
|
|
|
{ STRING_SIZE_TUPLE("notdir"), 1, 1, func_notdir_suffix},
|
|
|
|
|
{ STRING_SIZE_TUPLE("subst"), 3, 1, func_subst},
|
|
|
|
|
{ STRING_SIZE_TUPLE("suffix"), 1, 1, func_notdir_suffix},
|
|
|
|
|
{ STRING_SIZE_TUPLE("filter"), 2, 1, func_filter_filterout},
|
|
|
|
|
{ STRING_SIZE_TUPLE("filter-out"), 2, 1, func_filter_filterout},
|
|
|
|
|
{ STRING_SIZE_TUPLE("findstring"), 2, 1, func_findstring},
|
|
|
|
|
{ STRING_SIZE_TUPLE("firstword"), 1, 1, func_firstword},
|
|
|
|
|
{ STRING_SIZE_TUPLE("join"), 2, 1, func_join},
|
|
|
|
|
{ STRING_SIZE_TUPLE("patsubst"), 3, 1, func_patsubst},
|
|
|
|
|
{ STRING_SIZE_TUPLE("shell"), 1, 1, func_shell},
|
|
|
|
|
{ STRING_SIZE_TUPLE("sort"), 1, 1, func_sort},
|
|
|
|
|
{ STRING_SIZE_TUPLE("strip"), 1, 1, func_strip},
|
|
|
|
|
{ STRING_SIZE_TUPLE("wildcard"), 1, 1, func_wildcard},
|
|
|
|
|
{ STRING_SIZE_TUPLE("word"), 2, 1, func_word},
|
|
|
|
|
{ STRING_SIZE_TUPLE("wordlist"), 3, 1, func_wordlist},
|
|
|
|
|
{ STRING_SIZE_TUPLE("words"), 1, 1, func_words},
|
|
|
|
|
{ STRING_SIZE_TUPLE("origin"), 1, 1, func_origin},
|
|
|
|
|
{ STRING_SIZE_TUPLE("error"), 1, 1, func_error},
|
|
|
|
|
{ STRING_SIZE_TUPLE("warning"), 1, 1, func_error},
|
|
|
|
|
{ STRING_SIZE_TUPLE("foreach"), 3, 0, func_foreach},
|
|
|
|
|
#ifdef EXPERIMENTAL
|
|
|
|
|
{ STRING_SIZE_TUPLE("apply"), 1, 1, func_apply},
|
|
|
|
|
{ STRING_SIZE_TUPLE("eq"), 2, 1, func_eq},
|
|
|
|
|
{ STRING_SIZE_TUPLE("if"), 3, 0, func_if},
|
|
|
|
|
{ STRING_SIZE_TUPLE("not"), 1, 1, func_not},
|
|
|
|
|
#endif
|
|
|
|
|
{ 0 }
|
|
|
|
|
};
|