/*
* CDDL HEADER START
*
* The contents of this file are subject to the terms of the
* Common Development and Distribution License (the "License").
* You may not use this file except in compliance with the License.
*
* You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
* or http://www.opensolaris.org/os/licensing.
* See the License for the specific language governing permissions
* and limitations under the License.
*
* When distributing Covered Code, include this CDDL HEADER in each
* file and include the License file at usr/src/OPENSOLARIS.LICENSE.
* If applicable, add the following below this CDDL HEADER, with the
* fields enclosed by brackets "[]" replaced with your own identifying
* information: Portions Copyright [yyyy] [name of copyright owner]
*
* CDDL HEADER END
*/
/*
* Copyright 2006 Sun Microsystems, Inc. All rights reserved.
* Use is subject to license terms.
*/
/*
* macro.cc
*
* Handle expansion of make macros
*/
/*
* Included files
*/
#include <mksh/dosys.h> /* sh_command2string() */
#include <mksh/i18n.h> /* get_char_semantics_value() */
#include <mksh/macro.h>
#include <mksh/misc.h> /* retmem() */
#include <mksh/read.h> /* get_next_block_fn() */
#include <libintl.h>
/*
* File table of contents
*/
static void add_macro_to_global_list(Name macro_to_add);
static void expand_value_with_daemon(Name, register Property macro, register String destination, Boolean cmd);
static void init_arch_macros(void);
static void init_mach_macros(void);
static Boolean init_arch_done = false;
static Boolean init_mach_done = false;
long env_alloc_num = 0;
long env_alloc_bytes = 0;
/*
* getvar(name)
*
* Return expanded value of macro.
*
* Return value:
* The expanded value of the macro
*
* Parameters:
* name The name of the macro we want the value for
*
* Global variables used:
*/
Name
getvar(register Name name)
{
String_rec destination;
wchar_t buffer[STRING_BUFFER_LENGTH];
register Name result;
if ((name == host_arch) || (name == target_arch)) {
if (!init_arch_done) {
init_arch_done = true;
init_arch_macros();
}
}
if ((name == host_mach) || (name == target_mach)) {
if (!init_mach_done) {
init_mach_done = true;
init_mach_macros();
}
}
INIT_STRING_FROM_STACK(destination, buffer);
expand_value(maybe_append_prop(name, macro_prop)->body.macro.value,
&destination,
false);
result = GETNAME(destination.buffer.start, FIND_LENGTH);
if (destination.free_after_use) {
retmem(destination.buffer.start);
}
return result;
}
/*
* expand_value(value, destination, cmd)
*
* Recursively expands all macros in the string value.
* destination is where the expanded value should be appended.
*
* Parameters:
* value The value we are expanding
* destination Where to deposit the expansion
* cmd If we are evaluating a command line we
* turn \ quoting off
*
* Global variables used:
*/
void
expand_value(Name value, register String destination, Boolean cmd)
{
Source_rec sourceb;
register Source source = &sourceb;
register wchar_t *source_p = NULL;
register wchar_t *source_end = NULL;
wchar_t *block_start = NULL;
int quote_seen = 0;
if (value == NULL) {
/*
* Make sure to get a string allocated even if it
* will be empty.
*/
MBSTOWCS(wcs_buffer, "");
append_string(wcs_buffer, destination, FIND_LENGTH);
destination->text.end = destination->text.p;
return;
}
if (!value->dollar) {
/*
* If the value we are expanding does not contain
* any $, we don't have to parse it.
*/
APPEND_NAME(value,
destination,
(int) value->hash.length
);
destination->text.end = destination->text.p;
return;
}
if (value->being_expanded) {
fatal_reader_mksh(gettext("Loop detected when expanding macro value `%s'"),
value->string_mb);
}
value->being_expanded = true;
/* Setup the structure we read from */
Wstring vals(value);
sourceb.string.text.p = sourceb.string.buffer.start = wcsdup(vals.get_string());
sourceb.string.free_after_use = true;
sourceb.string.text.end =
sourceb.string.buffer.end =
sourceb.string.text.p + value->hash.length;
sourceb.previous = NULL;
sourceb.fd = -1;
sourceb.inp_buf =
sourceb.inp_buf_ptr =
sourceb.inp_buf_end = NULL;
sourceb.error_converting = false;
/* Lift some pointers from the struct to local register variables */
CACHE_SOURCE(0);
/* We parse the string in segments */
/* We read chars until we find a $, then we append what we have read so far */
/* (since last $ processing) to the destination. When we find a $ we call */
/* expand_macro() and let it expand that particular $ reference into dest */
block_start = source_p;
quote_seen = 0;
for (; 1; source_p++) {
switch (GET_CHAR()) {
case backslash_char:
/* Quote $ in macro value */
if (!cmd) {
quote_seen = ~quote_seen;
}
continue;
case dollar_char:
/* Save the plain string we found since */
/* start of string or previous $ */
if (quote_seen) {
append_string(block_start,
destination,
source_p - block_start - 1);
block_start = source_p;
break;
}
append_string(block_start,
destination,
source_p - block_start);
source->string.text.p = ++source_p;
UNCACHE_SOURCE();
/* Go expand the macro reference */
expand_macro(source, destination, sourceb.string.buffer.start, cmd);
CACHE_SOURCE(1);
block_start = source_p + 1;
break;
case nul_char:
/* The string ran out. Get some more */
append_string(block_start,
destination,
source_p - block_start);
GET_NEXT_BLOCK_NOCHK(source);
if (source == NULL) {
destination->text.end = destination->text.p;
value->being_expanded = false;
return;
}
if (source->error_converting) {
fatal_reader_mksh("Internal error: Invalid byte sequence in expand_value()");
}
block_start = source_p;
source_p--;
continue;
}
quote_seen = 0;
}
retmem(sourceb.string.buffer.start);
}
/*
* expand_macro(source, destination, current_string, cmd)
*
* Should be called with source->string.text.p pointing to
* the first char after the $ that starts a macro reference.
* source->string.text.p is returned pointing to the first char after
* the macro name.
* It will read the macro name, expanding any macros in it,
* and get the value. The value is then expanded.
* destination is a String that is filled in with the expanded macro.
* It may be passed in referencing a buffer to expand the macro into.
* Note that most expansions are done on demand, e.g. right
* before the command is executed and not while the file is
* being parsed.
*
* Parameters:
* source The source block that references the string
* to expand
* destination Where to put the result
* current_string The string we are expanding, for error msg
* cmd If we are evaluating a command line we
* turn \ quoting off
*
* Global variables used:
* funny Vector of semantic tags for characters
* is_conditional Set if a conditional macro is refd
* make_word_mentioned Set if the word "MAKE" is mentioned
* makefile_type We deliver extra msg when reading makefiles
* query The Name "?", compared against
* query_mentioned Set if the word "?" is mentioned
*/
void
expand_macro(register Source source, register String destination, wchar_t *current_string, Boolean cmd)
{
static Name make = (Name)NULL;
static wchar_t colon_sh[4];
static wchar_t colon_shell[7];
String_rec string;
wchar_t buffer[STRING_BUFFER_LENGTH];
register wchar_t *source_p = source->string.text.p;
register wchar_t *source_end = source->string.text.end;
register int closer = 0;
wchar_t *block_start = (wchar_t *)NULL;
int quote_seen = 0;
register int closer_level = 1;
Name name = (Name)NULL;
wchar_t *colon = (wchar_t *)NULL;
wchar_t *percent = (wchar_t *)NULL;
wchar_t *eq = (wchar_t *) NULL;
Property macro = NULL;
wchar_t *p = (wchar_t*)NULL;
String_rec extracted;
wchar_t extracted_string[MAXPATHLEN];
wchar_t *left_head = NULL;
wchar_t *left_tail = NULL;
wchar_t *right_tail = NULL;
int left_head_len = 0;
int left_tail_len = 0;
int tmp_len = 0;
wchar_t *right_hand[128];
int i = 0;
enum {
no_extract,
dir_extract,
file_extract
} extraction = no_extract;
enum {
no_replace,
suffix_replace,
pattern_replace,
sh_replace
} replacement = no_replace;
if (make == NULL) {
MBSTOWCS(wcs_buffer, "MAKE");
make = GETNAME(wcs_buffer, FIND_LENGTH);
MBSTOWCS(colon_sh, ":sh");
MBSTOWCS(colon_shell, ":shell");
}
right_hand[0] = NULL;
/* First copy the (macro-expanded) macro name into string. */
INIT_STRING_FROM_STACK(string, buffer);
recheck_first_char:
/* Check the first char of the macro name to figure out what to do. */
switch (GET_CHAR()) {
case nul_char:
GET_NEXT_BLOCK_NOCHK(source);
if (source == NULL) {
WCSTOMBS(mbs_buffer, current_string);
fatal_reader_mksh(gettext("'$' at end of string `%s'"),
mbs_buffer);
}
if (source->error_converting) {
fatal_reader_mksh("Internal error: Invalid byte sequence in expand_macro()");
}
goto recheck_first_char;
case parenleft_char:
/* Multi char name. */
closer = (int) parenright_char;
break;
case braceleft_char:
/* Multi char name. */
closer = (int) braceright_char;
break;
case newline_char:
fatal_reader_mksh(gettext("'$' at end of line"));
default:
/* Single char macro name. Just suck it up */
append_char(*source_p, &string);
source->string.text.p = source_p + 1;
goto get_macro_value;
}
/* Handle multi-char macro names */
block_start = ++source_p;
quote_seen = 0;
for (; 1; source_p++) {
switch (GET_CHAR()) {
case nul_char:
append_string(block_start,
&string,
source_p - block_start);
GET_NEXT_BLOCK_NOCHK(source);
if (source == NULL) {
if (current_string != NULL) {
WCSTOMBS(mbs_buffer, current_string);
fatal_reader_mksh(gettext("Unmatched `%c' in string `%s'"),
closer ==
(int) braceright_char ?
(int) braceleft_char :
(int) parenleft_char,
mbs_buffer);
} else {
fatal_reader_mksh(gettext("Premature EOF"));
}
}
if (source->error_converting) {
fatal_reader_mksh("Internal error: Invalid byte sequence in expand_macro()");
}
block_start = source_p;
source_p--;
continue;
case newline_char:
fatal_reader_mksh(gettext("Unmatched `%c' on line"),
closer == (int) braceright_char ?
(int) braceleft_char :
(int) parenleft_char);
case backslash_char:
/* Quote dollar in macro value. */
if (!cmd) {
quote_seen = ~quote_seen;
}
continue;
case dollar_char:
/*
* Macro names may reference macros.
* This expands the value of such macros into the
* macro name string.
*/
if (quote_seen) {
append_string(block_start,
&string,
source_p - block_start - 1);
block_start = source_p;
break;
}
append_string(block_start,
&string,
source_p - block_start);
source->string.text.p = ++source_p;
UNCACHE_SOURCE();
expand_macro(source, &string, current_string, cmd);
CACHE_SOURCE(0);
block_start = source_p;
source_p--;
break;
case parenleft_char:
/* Allow nested pairs of () in the macro name. */
if (closer == (int) parenright_char) {
closer_level++;
}
break;
case braceleft_char:
/* Allow nested pairs of {} in the macro name. */
if (closer == (int) braceright_char) {
closer_level++;
}
break;
case parenright_char:
case braceright_char:
/*
* End of the name. Save the string in the macro
* name string.
*/
if ((*source_p == closer) && (--closer_level <= 0)) {
source->string.text.p = source_p + 1;
append_string(block_start,
&string,
source_p - block_start);
goto get_macro_value;
}
break;
}
quote_seen = 0;
}
/*
* We got the macro name. We now inspect it to see if it
* specifies any translations of the value.
*/
get_macro_value:
name = NULL;
/* First check if we have a $(@D) type translation. */
if ((get_char_semantics_value(string.buffer.start[0]) &
(int) special_macro_sem) &&
(string.text.p - string.buffer.start >= 2) &&
((string.buffer.start[1] == 'D') ||
(string.buffer.start[1] == 'F'))) {
switch (string.buffer.start[1]) {
case 'D':
extraction = dir_extract;
break;
case 'F':
extraction = file_extract;
break;
default:
WCSTOMBS(mbs_buffer, string.buffer.start);
fatal_reader_mksh(gettext("Illegal macro reference `%s'"),
mbs_buffer);
}
/* Internalize the macro name using the first char only. */
name = GETNAME(string.buffer.start, 1);
(void) wcscpy(string.buffer.start, string.buffer.start + 2);
}
/* Check for other kinds of translations. */
if ((colon = (wchar_t *) wcschr(string.buffer.start,
(int) colon_char)) != NULL) {
/*
* We have a $(FOO:.c=.o) type translation.
* Get the name of the macro proper.
*/
if (name == NULL) {
name = GETNAME(string.buffer.start,
colon - string.buffer.start);
}
/* Pickup all the translations. */
if (IS_WEQUAL(colon, colon_sh) || IS_WEQUAL(colon, colon_shell)) {
replacement = sh_replace;
} else if ((svr4) ||
((percent = (wchar_t *) wcschr(colon + 1,
(int) percent_char)) == NULL)) {
while (colon != NULL) {
if ((eq = (wchar_t *) wcschr(colon + 1,
(int) equal_char)) == NULL) {
fatal_reader_mksh(gettext("= missing from replacement macro reference"));
}
left_tail_len = eq - colon - 1;
if(left_tail) {
retmem(left_tail);
}
left_tail = ALLOC_WC(left_tail_len + 1);
(void) wcsncpy(left_tail,
colon + 1,
eq - colon - 1);
left_tail[eq - colon - 1] = (int) nul_char;
replacement = suffix_replace;
if ((colon = (wchar_t *) wcschr(eq + 1,
(int) colon_char)) != NULL) {
tmp_len = colon - eq;
if(right_tail) {
retmem(right_tail);
}
right_tail = ALLOC_WC(tmp_len);
(void) wcsncpy(right_tail,
eq + 1,
colon - eq - 1);
right_tail[colon - eq - 1] =
(int) nul_char;
} else {
if(right_tail) {
retmem(right_tail);
}
right_tail = ALLOC_WC(wcslen(eq) + 1);
(void) wcscpy(right_tail, eq + 1);
}
}
} else {
if ((eq = (wchar_t *) wcschr(colon + 1,
(int) equal_char)) == NULL) {
fatal_reader_mksh(gettext("= missing from replacement macro reference"));
}
if ((percent = (wchar_t *) wcschr(colon + 1,
(int) percent_char)) == NULL) {
fatal_reader_mksh(gettext("%% missing from replacement macro reference"));
}
if (eq < percent) {
fatal_reader_mksh(gettext("%% missing from replacement macro reference"));
}
if (percent > (colon + 1)) {
tmp_len = percent - colon;
if(left_head) {
retmem(left_head);
}
left_head = ALLOC_WC(tmp_len);
(void) wcsncpy(left_head,
colon + 1,
percent - colon - 1);
left_head[percent-colon-1] = (int) nul_char;
left_head_len = percent-colon-1;
} else {
left_head = NULL;
left_head_len = 0;
}
if (eq > percent+1) {
tmp_len = eq - percent;
if(left_tail) {
retmem(left_tail);
}
left_tail = ALLOC_WC(tmp_len);
(void) wcsncpy(left_tail,
percent + 1,
eq - percent - 1);
left_tail[eq-percent-1] = (int) nul_char;
left_tail_len = eq-percent-1;
} else {
left_tail = NULL;
left_tail_len = 0;
}
if ((percent = (wchar_t *) wcschr(++eq,
(int) percent_char)) == NULL) {
right_hand[0] = ALLOC_WC(wcslen(eq) + 1);
right_hand[1] = NULL;
(void) wcscpy(right_hand[0], eq);
} else {
i = 0;
do {
right_hand[i] = ALLOC_WC(percent-eq+1);
(void) wcsncpy(right_hand[i],
eq,
percent - eq);
right_hand[i][percent-eq] =
(int) nul_char;
if (i++ >= VSIZEOF(right_hand)) {
fatal_mksh(gettext("Too many %% in pattern"));
}
eq = percent + 1;
if (eq[0] == (int) nul_char) {
MBSTOWCS(wcs_buffer, "");
right_hand[i] = (wchar_t *) wcsdup(wcs_buffer);
i++;
break;
}
} while ((percent = (wchar_t *) wcschr(eq, (int) percent_char)) != NULL);
if (eq[0] != (int) nul_char) {
right_hand[i] = ALLOC_WC(wcslen(eq) + 1);
(void) wcscpy(right_hand[i], eq);
i++;
}
right_hand[i] = NULL;
}
replacement = pattern_replace;
}
}
if (name == NULL) {
/*
* No translations found.
* Use the whole string as the macro name.
*/
name = GETNAME(string.buffer.start,
string.text.p - string.buffer.start);
}
if (string.free_after_use) {
retmem(string.buffer.start);
}
if (name == make) {
make_word_mentioned = true;
}
if (name == query) {
query_mentioned = true;
}
if ((name == host_arch) || (name == target_arch)) {
if (!init_arch_done) {
init_arch_done = true;
init_arch_macros();
}
}
if ((name == host_mach) || (name == target_mach)) {
if (!init_mach_done) {
init_mach_done = true;
init_mach_macros();
}
}
/* Get the macro value. */
macro = get_prop(name->prop, macro_prop);
if ((macro != NULL) && macro->body.macro.is_conditional) {
conditional_macro_used = true;
/*
* Add this conditional macro to the beginning of the
* global list.
*/
add_macro_to_global_list(name);
if (makefile_type == reading_makefile) {
warning_mksh(gettext("Conditional macro `%s' referenced in file `%ws', line %d"),
name->string_mb, file_being_read, line_number);
}
}
/* Macro name read and parsed. Expand the value. */
if ((macro == NULL) || (macro->body.macro.value == NULL)) {
/* If the value is empty, we just get out of here. */
goto exit;
}
if (replacement == sh_replace) {
/* If we should do a :sh transform, we expand the command
* and process it.
*/
INIT_STRING_FROM_STACK(string, buffer);
/* Expand the value into a local string buffer and run cmd. */
expand_value_with_daemon(name, macro, &string, cmd);
sh_command2string(&string, destination);
} else if ((replacement != no_replace) || (extraction != no_extract)) {
/*
* If there were any transforms specified in the macro
* name, we deal with them here.
*/
INIT_STRING_FROM_STACK(string, buffer);
/* Expand the value into a local string buffer. */
expand_value_with_daemon(name, macro, &string, cmd);
/* Scan the expanded string. */
p = string.buffer.start;
while (*p != (int) nul_char) {
wchar_t chr;
/*
* First skip over any white space and append
* that to the destination string.
*/
block_start = p;
while ((*p != (int) nul_char) && iswspace(*p)) {
p++;
}
append_string(block_start,
destination,
p - block_start);
/* Then find the end of the next word. */
block_start = p;
while ((*p != (int) nul_char) && !iswspace(*p)) {
p++;
}
/* If we cant find another word we are done */
if (block_start == p) {
break;
}
/* Then apply the transforms to the word */
INIT_STRING_FROM_STACK(extracted, extracted_string);
switch (extraction) {
case dir_extract:
/*
* $(@D) type transform. Extract the
* path from the word. Deliver "." if
* none is found.
*/
if (p != NULL) {
chr = *p;
*p = (int) nul_char;
}
eq = (wchar_t *) wcsrchr(block_start, (int) slash_char);
if (p != NULL) {
*p = chr;
}
if ((eq == NULL) || (eq > p)) {
MBSTOWCS(wcs_buffer, ".");
append_string(wcs_buffer, &extracted, 1);
} else {
append_string(block_start,
&extracted,
eq - block_start);
}
break;
case file_extract:
/*
* $(@F) type transform. Remove the path
* from the word if any.
*/
if (p != NULL) {
chr = *p;
*p = (int) nul_char;
}
eq = (wchar_t *) wcsrchr(block_start, (int) slash_char);
if (p != NULL) {
*p = chr;
}
if ((eq == NULL) || (eq > p)) {
append_string(block_start,
&extracted,
p - block_start);
} else {
append_string(eq + 1,
&extracted,
p - eq - 1);
}
break;
case no_extract:
append_string(block_start,
&extracted,
p - block_start);
break;
}
switch (replacement) {
case suffix_replace:
/*
* $(FOO:.o=.c) type transform.
* Maybe replace the tail of the word.
*/
if (((extracted.text.p -
extracted.buffer.start) >=
left_tail_len) &&
IS_WEQUALN(extracted.text.p - left_tail_len,
left_tail,
left_tail_len)) {
append_string(extracted.buffer.start,
destination,
(extracted.text.p -
extracted.buffer.start)
- left_tail_len);
append_string(right_tail,
destination,
FIND_LENGTH);
} else {
append_string(extracted.buffer.start,
destination,
FIND_LENGTH);
}
break;
case pattern_replace:
/* $(X:a%b=c%d) type transform. */
if (((extracted.text.p -
extracted.buffer.start) >=
left_head_len+left_tail_len) &&
IS_WEQUALN(left_head,
extracted.buffer.start,
left_head_len) &&
IS_WEQUALN(left_tail,
extracted.text.p - left_tail_len,
left_tail_len)) {
i = 0;
while (right_hand[i] != NULL) {
append_string(right_hand[i],
destination,
FIND_LENGTH);
i++;
if (right_hand[i] != NULL) {
append_string(extracted.buffer.
start +
left_head_len,
destination,
(extracted.text.p - extracted.buffer.start)-left_head_len-left_tail_len);
}
}
} else {
append_string(extracted.buffer.start,
destination,
FIND_LENGTH);
}
break;
case no_replace:
append_string(extracted.buffer.start,
destination,
FIND_LENGTH);
break;
case sh_replace:
break;
}
}
if (string.free_after_use) {
retmem(string.buffer.start);
}
} else {
/*
* This is for the case when the macro name did not
* specify transforms.
*/
if (!strncmp(name->string_mb, "GET", 3)) {
dollarget_seen = true;
}
dollarless_flag = false;
if (!strncmp(name->string_mb, "<", 1) &&
dollarget_seen) {
dollarless_flag = true;
dollarget_seen = false;
}
expand_value_with_daemon(name, macro, destination, cmd);
}
exit:
if(left_tail) {
retmem(left_tail);
}
if(right_tail) {
retmem(right_tail);
}
if(left_head) {
retmem(left_head);
}
i = 0;
while (right_hand[i] != NULL) {
retmem(right_hand[i]);
i++;
}
*destination->text.p = (int) nul_char;
destination->text.end = destination->text.p;
}
static void
add_macro_to_global_list(Name macro_to_add)
{
Macro_list new_macro;
Macro_list macro_on_list;
char *name_on_list = (char*)NULL;
char *name_to_add = macro_to_add->string_mb;
char *value_on_list = (char*)NULL;
const char *value_to_add = (char*)NULL;
if (macro_to_add->prop->body.macro.value != NULL) {
value_to_add = macro_to_add->prop->body.macro.value->string_mb;
} else {
value_to_add = "";
}
/*
* Check if this macro is already on list, if so, do nothing
*/
for (macro_on_list = cond_macro_list;
macro_on_list != NULL;
macro_on_list = macro_on_list->next) {
name_on_list = macro_on_list->macro_name;
value_on_list = macro_on_list->value;
if (IS_EQUAL(name_on_list, name_to_add)) {
if (IS_EQUAL(value_on_list, value_to_add)) {
return;
}
}
}
new_macro = (Macro_list) malloc(sizeof(Macro_list_rec));
new_macro->macro_name = strdup(name_to_add);
new_macro->value = strdup(value_to_add);
new_macro->next = cond_macro_list;
cond_macro_list = new_macro;
}
/*
* init_arch_macros(void)
*
* Set the magic macros TARGET_ARCH, HOST_ARCH,
*
* Parameters:
*
* Global variables used:
* host_arch Property for magic macro HOST_ARCH
* target_arch Property for magic macro TARGET_ARCH
*
* Return value:
* The function does not return a value, but can
* call fatal() in case of error.
*/
static void
init_arch_macros(void)
{
String_rec result_string;
wchar_t wc_buf[STRING_BUFFER_LENGTH];
char mb_buf[STRING_BUFFER_LENGTH];
FILE *pipe;
Name value;
int set_host, set_target;
const char *mach_command = "/bin/mach";
set_host = (get_prop(host_arch->prop, macro_prop) == NULL);
set_target = (get_prop(target_arch->prop, macro_prop) == NULL);
if (set_host || set_target) {
INIT_STRING_FROM_STACK(result_string, wc_buf);
append_char((int) hyphen_char, &result_string);
if ((pipe = popen(mach_command, "r")) == NULL) {
fatal_mksh(gettext("Execute of %s failed"), mach_command);
}
while (fgets(mb_buf, sizeof(mb_buf), pipe) != NULL) {
MBSTOWCS(wcs_buffer, mb_buf);
append_string(wcs_buffer, &result_string, wcslen(wcs_buffer));
}
if (pclose(pipe) != 0) {
fatal_mksh(gettext("Execute of %s failed"), mach_command);
}
value = GETNAME(result_string.buffer.start, wcslen(result_string.buffer.start));
if (set_host) {
(void) setvar_daemon(host_arch, value, false, no_daemon, true, 0);
}
if (set_target) {
(void) setvar_daemon(target_arch, value, false, no_daemon, true, 0);
}
}
}
/*
* init_mach_macros(void)
*
* Set the magic macros TARGET_MACH, HOST_MACH,
*
* Parameters:
*
* Global variables used:
* host_mach Property for magic macro HOST_MACH
* target_mach Property for magic macro TARGET_MACH
*
* Return value:
* The function does not return a value, but can
* call fatal() in case of error.
*/
static void
init_mach_macros(void)
{
String_rec result_string;
wchar_t wc_buf[STRING_BUFFER_LENGTH];
char mb_buf[STRING_BUFFER_LENGTH];
FILE *pipe;
Name value;
int set_host, set_target;
const char *arch_command = "/bin/arch";
set_host = (get_prop(host_mach->prop, macro_prop) == NULL);
set_target = (get_prop(target_mach->prop, macro_prop) == NULL);
if (set_host || set_target) {
INIT_STRING_FROM_STACK(result_string, wc_buf);
append_char((int) hyphen_char, &result_string);
if ((pipe = popen(arch_command, "r")) == NULL) {
fatal_mksh(gettext("Execute of %s failed"), arch_command);
}
while (fgets(mb_buf, sizeof(mb_buf), pipe) != NULL) {
MBSTOWCS(wcs_buffer, mb_buf);
append_string(wcs_buffer, &result_string, wcslen(wcs_buffer));
}
if (pclose(pipe) != 0) {
fatal_mksh(gettext("Execute of %s failed"), arch_command);
}
value = GETNAME(result_string.buffer.start, wcslen(result_string.buffer.start));
if (set_host) {
(void) setvar_daemon(host_mach, value, false, no_daemon, true, 0);
}
if (set_target) {
(void) setvar_daemon(target_mach, value, false, no_daemon, true, 0);
}
}
}
/*
* expand_value_with_daemon(name, macro, destination, cmd)
*
* Checks for daemons and then maybe calls expand_value().
*
* Parameters:
* name Name of the macro (Added by the NSE)
* macro The property block with the value to expand
* destination Where the result should be deposited
* cmd If we are evaluating a command line we
* turn \ quoting off
*
* Global variables used:
*/
static void
expand_value_with_daemon(Name, register Property macro, register String destination, Boolean cmd)
{
register Chain chain;
switch (macro->body.macro.daemon) {
case no_daemon:
if (!svr4 && !posix) {
expand_value(macro->body.macro.value, destination, cmd);
} else {
if (dollarless_flag && tilde_rule) {
expand_value(dollarless_value, destination, cmd);
dollarless_flag = false;
tilde_rule = false;
} else {
expand_value(macro->body.macro.value, destination, cmd);
}
}
return;
case chain_daemon:
/* If this is a $? value we call the daemon to translate the */
/* list of names to a string */
for (chain = (Chain) macro->body.macro.value;
chain != NULL;
chain = chain->next) {
APPEND_NAME(chain->name,
destination,
(int) chain->name->hash.length);
if (chain->next != NULL) {
append_char((int) space_char, destination);
}
}
return;
}
}
/*
* We use a permanent buffer to reset SUNPRO_DEPENDENCIES value.
*/
char *sunpro_dependencies_buf = NULL;
char *sunpro_dependencies_oldbuf = NULL;
int sunpro_dependencies_buf_size = 0;
/*
* setvar_daemon(name, value, append, daemon, strip_trailing_spaces)
*
* Set a macro value, possibly supplying a daemon to be used
* when referencing the value.
*
* Return value:
* The property block with the new value
*
* Parameters:
* name Name of the macro to set
* value The value to set
* append Should we reset or append to the current value?
* daemon Special treatment when reading the value
* strip_trailing_spaces from the end of value->string
* debug_level Indicates how much tracing we should do
*
* Global variables used:
* makefile_type Used to check if we should enforce read only
* path_name The Name "PATH", compared against
* virtual_root The Name "VIRTUAL_ROOT", compared against
* vpath_defined Set if the macro VPATH is set
* vpath_name The Name "VPATH", compared against
* envvar A list of environment vars with $ in value
*/
Property
setvar_daemon(register Name name, register Name value, Boolean append, Daemon daemon, Boolean strip_trailing_spaces, short debug_level)
{
register Property macro = maybe_append_prop(name, macro_prop);
register Property macro_apx = get_prop(name->prop, macro_append_prop);
int length = 0;
String_rec destination;
wchar_t buffer[STRING_BUFFER_LENGTH];
register Chain chain;
Name val;
wchar_t *val_string = (wchar_t*)NULL;
Wstring wcb;
if ((makefile_type != reading_nothing) &&
macro->body.macro.read_only) {
return macro;
}
/* Strip spaces from the end of the value */
if (daemon == no_daemon) {
if(value != NULL) {
wcb.init(value);
length = wcb.length();
val_string = wcb.get_string();
}
if ((length > 0) && iswspace(val_string[length-1])) {
INIT_STRING_FROM_STACK(destination, buffer);
buffer[0] = 0;
append_string(val_string, &destination, length);
if (strip_trailing_spaces) {
while ((length > 0) &&
iswspace(destination.buffer.start[length-1])) {
destination.buffer.start[--length] = 0;
}
}
value = GETNAME(destination.buffer.start, FIND_LENGTH);
}
}
if(macro_apx != NULL) {
val = macro_apx->body.macro_appendix.value;
} else {
val = macro->body.macro.value;
}
if (append) {
/*
* If we are appending, we just tack the new value after
* the old one with a space in between.
*/
INIT_STRING_FROM_STACK(destination, buffer);
buffer[0] = 0;
if ((macro != NULL) && (val != NULL)) {
APPEND_NAME(val,
&destination,
(int) val->hash.length);
if (value != NULL) {
wcb.init(value);
if(wcb.length() > 0) {
MBTOWC(wcs_buffer, " ");
append_char(wcs_buffer[0], &destination);
}
}
}
if (value != NULL) {
APPEND_NAME(value,
&destination,
(int) value->hash.length);
}
value = GETNAME(destination.buffer.start, FIND_LENGTH);
wcb.init(value);
if (destination.free_after_use) {
retmem(destination.buffer.start);
}
}
/* Debugging trace */
if (debug_level > 1) {
if (value != NULL) {
switch (daemon) {
case chain_daemon:
(void) printf("%s =", name->string_mb);
for (chain = (Chain) value;
chain != NULL;
chain = chain->next) {
(void) printf(" %s", chain->name->string_mb);
}
(void) printf("\n");
break;
case no_daemon:
(void) printf("%s= %s\n",
name->string_mb,
value->string_mb);
break;
}
} else {
(void) printf("%s =\n", name->string_mb);
}
}
/* Set the new values in the macro property block */
/**/
if(macro_apx != NULL) {
macro_apx->body.macro_appendix.value = value;
INIT_STRING_FROM_STACK(destination, buffer);
buffer[0] = 0;
if (value != NULL) {
APPEND_NAME(value,
&destination,
(int) value->hash.length);
if (macro_apx->body.macro_appendix.value_to_append != NULL) {
MBTOWC(wcs_buffer, " ");
append_char(wcs_buffer[0], &destination);
}
}
if (macro_apx->body.macro_appendix.value_to_append != NULL) {
APPEND_NAME(macro_apx->body.macro_appendix.value_to_append,
&destination,
(int) macro_apx->body.macro_appendix.value_to_append->hash.length);
}
value = GETNAME(destination.buffer.start, FIND_LENGTH);
if (destination.free_after_use) {
retmem(destination.buffer.start);
}
}
/**/
macro->body.macro.value = value;
macro->body.macro.daemon = daemon;
/*
* If the user changes the VIRTUAL_ROOT, we need to flush
* the vroot package cache.
*/
if (name == path_name) {
flush_path_cache();
}
if (name == virtual_root) {
flush_vroot_cache();
}
/* If this sets the VPATH we remember that */
if ((name == vpath_name) &&
(value != NULL) &&
(value->hash.length > 0)) {
vpath_defined = true;
}
/*
* For environment variables we also set the
* environment value each time.
*/
if (macro->body.macro.exported) {
static char *env;
if (!reading_environment && (value != NULL)) {
Envvar p;
for (p = envvar; p != NULL; p = p->next) {
if (p->name == name) {
p->value = value;
p->already_put = false;
goto found_it;
}
}
p = ALLOC(Envvar);
p->name = name;
p->value = value;
p->next = envvar;
p->env_string = NULL;
p->already_put = false;
envvar = p;
found_it:;
} if (reading_environment || (value == NULL) || !value->dollar) {
length = 2 + strlen(name->string_mb);
if (value != NULL) {
length += strlen(value->string_mb);
}
Property env_prop = maybe_append_prop(name, env_mem_prop);
/*
* We use a permanent buffer to reset SUNPRO_DEPENDENCIES value.
*/
if (!strncmp(name->string_mb, "SUNPRO_DEPENDENCIES", 19)) {
if (length >= sunpro_dependencies_buf_size) {
sunpro_dependencies_buf_size=length*2;
if (sunpro_dependencies_buf_size < 4096)
sunpro_dependencies_buf_size = 4096; // Default minimum size
if (sunpro_dependencies_buf)
sunpro_dependencies_oldbuf = sunpro_dependencies_buf;
sunpro_dependencies_buf=getmem(sunpro_dependencies_buf_size);
}
env = sunpro_dependencies_buf;
} else {
env = getmem(length);
}
env_alloc_num++;
env_alloc_bytes += length;
(void) sprintf(env,
"%s=%s",
name->string_mb,
value == NULL ?
"" : value->string_mb);
(void) putenv(env);
env_prop->body.env_mem.value = env;
if (sunpro_dependencies_oldbuf) {
/* Return old buffer */
retmem_mb(sunpro_dependencies_oldbuf);
sunpro_dependencies_oldbuf = NULL;
}
}
}
if (name == target_arch) {
Name ha = getvar(host_arch);
Name ta = getvar(target_arch);
Name vr = getvar(virtual_root);
int length;
wchar_t *new_value;
wchar_t *old_vr;
Boolean new_value_allocated = false;
Wstring ha_str(ha);
Wstring ta_str(ta);
Wstring vr_str(vr);
wchar_t * wcb_ha = ha_str.get_string();
wchar_t * wcb_ta = ta_str.get_string();
wchar_t * wcb_vr = vr_str.get_string();
length = 32 +
wcslen(wcb_ha) +
wcslen(wcb_ta) +
wcslen(wcb_vr);
old_vr = wcb_vr;
MBSTOWCS(wcs_buffer, "/usr/arch/");
if (IS_WEQUALN(old_vr,
wcs_buffer,
wcslen(wcs_buffer))) {
old_vr = (wchar_t *) wcschr(old_vr, (int) colon_char) + 1;
}
if ( (ha == ta) || (wcslen(wcb_ta) == 0) ) {
new_value = old_vr;
} else {
new_value = ALLOC_WC(length);
new_value_allocated = true;
WCSTOMBS(mbs_buffer, old_vr);
(void) swprintf(new_value, length * SIZEOFWCHAR_T,
L"/usr/arch/%s/%s:%s",
ha->string_mb + 1,
ta->string_mb + 1,
mbs_buffer);
}
if (new_value[0] != 0) {
(void) setvar_daemon(virtual_root,
GETNAME(new_value, FIND_LENGTH),
false,
no_daemon,
true,
debug_level);
}
if (new_value_allocated) {
retmem(new_value);
}
}
return macro;
}