Print this page
make: restore a couple of blocks of code from DISTRIBUTED that should have been TEAMWARE_MAKE_CMN
Split |
Close |
Expand all |
Collapse all |
--- old/usr/src/cmd/make/lib/mksh/macro.cc
+++ new/usr/src/cmd/make/lib/mksh/macro.cc
1 1 /*
2 2 * CDDL HEADER START
3 3 *
4 4 * The contents of this file are subject to the terms of the
5 5 * Common Development and Distribution License (the "License").
6 6 * You may not use this file except in compliance with the License.
7 7 *
8 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 9 * or http://www.opensolaris.org/os/licensing.
10 10 * See the License for the specific language governing permissions
11 11 * and limitations under the License.
12 12 *
13 13 * When distributing Covered Code, include this CDDL HEADER in each
14 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 15 * If applicable, add the following below this CDDL HEADER, with the
16 16 * fields enclosed by brackets "[]" replaced with your own identifying
17 17 * information: Portions Copyright [yyyy] [name of copyright owner]
18 18 *
19 19 * CDDL HEADER END
20 20 */
21 21 /*
22 22 * Copyright 2006 Sun Microsystems, Inc. All rights reserved.
23 23 * Use is subject to license terms.
24 24 */
25 25
26 26
27 27 /*
28 28 * macro.cc
29 29 *
30 30 * Handle expansion of make macros
31 31 */
32 32
33 33 /*
34 34 * Included files
35 35 */
36 36 #include <mksh/dosys.h> /* sh_command2string() */
37 37 #include <mksh/i18n.h> /* get_char_semantics_value() */
38 38 #include <mksh/macro.h>
39 39 #include <mksh/misc.h> /* retmem() */
40 40 #include <mksh/read.h> /* get_next_block_fn() */
41 41 #include <mksdmsi18n/mksdmsi18n.h> /* libmksdmsi18n_init() */
42 42
43 43 #include <widec.h>
44 44
45 45 /*
46 46 * File table of contents
47 47 */
48 48 static void add_macro_to_global_list(Name macro_to_add);
49 49 static void expand_value_with_daemon(Name, register Property macro, register String destination, Boolean cmd);
50 50
51 51 static void init_arch_macros(void);
52 52 static void init_mach_macros(void);
53 53 static Boolean init_arch_done = false;
54 54 static Boolean init_mach_done = false;
55 55
56 56
57 57 long env_alloc_num = 0;
58 58 long env_alloc_bytes = 0;
59 59
60 60 /*
61 61 * getvar(name)
62 62 *
63 63 * Return expanded value of macro.
64 64 *
65 65 * Return value:
66 66 * The expanded value of the macro
67 67 *
68 68 * Parameters:
69 69 * name The name of the macro we want the value for
70 70 *
71 71 * Global variables used:
72 72 */
73 73 Name
74 74 getvar(register Name name)
75 75 {
76 76 String_rec destination;
77 77 wchar_t buffer[STRING_BUFFER_LENGTH];
78 78 register Name result;
79 79
80 80 if ((name == host_arch) || (name == target_arch)) {
81 81 if (!init_arch_done) {
82 82 init_arch_done = true;
83 83 init_arch_macros();
84 84 }
85 85 }
86 86 if ((name == host_mach) || (name == target_mach)) {
87 87 if (!init_mach_done) {
88 88 init_mach_done = true;
89 89 init_mach_macros();
90 90 }
91 91 }
92 92
93 93 INIT_STRING_FROM_STACK(destination, buffer);
94 94 expand_value(maybe_append_prop(name, macro_prop)->body.macro.value,
95 95 &destination,
96 96 false);
97 97 result = GETNAME(destination.buffer.start, FIND_LENGTH);
98 98 if (destination.free_after_use) {
99 99 retmem(destination.buffer.start);
100 100 }
101 101 return result;
102 102 }
103 103
104 104 /*
105 105 * expand_value(value, destination, cmd)
106 106 *
107 107 * Recursively expands all macros in the string value.
108 108 * destination is where the expanded value should be appended.
109 109 *
110 110 * Parameters:
111 111 * value The value we are expanding
112 112 * destination Where to deposit the expansion
113 113 * cmd If we are evaluating a command line we
114 114 * turn \ quoting off
115 115 *
116 116 * Global variables used:
117 117 */
118 118 void
119 119 expand_value(Name value, register String destination, Boolean cmd)
120 120 {
121 121 Source_rec sourceb;
122 122 register Source source = &sourceb;
123 123 register wchar_t *source_p = NULL;
124 124 register wchar_t *source_end = NULL;
125 125 wchar_t *block_start = NULL;
126 126 int quote_seen = 0;
127 127
128 128 if (value == NULL) {
129 129 /*
130 130 * Make sure to get a string allocated even if it
131 131 * will be empty.
132 132 */
133 133 MBSTOWCS(wcs_buffer, "");
134 134 append_string(wcs_buffer, destination, FIND_LENGTH);
135 135 destination->text.end = destination->text.p;
136 136 return;
137 137 }
138 138 if (!value->dollar) {
139 139 /*
140 140 * If the value we are expanding does not contain
141 141 * any $, we don't have to parse it.
142 142 */
143 143 APPEND_NAME(value,
144 144 destination,
145 145 (int) value->hash.length
146 146 );
147 147 destination->text.end = destination->text.p;
148 148 return;
149 149 }
150 150
151 151 if (value->being_expanded) {
152 152 fatal_reader_mksh(catgets(libmksdmsi18n_catd, 1, 113, "Loop detected when expanding macro value `%s'"),
153 153 value->string_mb);
154 154 }
155 155 value->being_expanded = true;
156 156 /* Setup the structure we read from */
157 157 Wstring vals(value);
158 158 sourceb.string.text.p = sourceb.string.buffer.start = wsdup(vals.get_string());
159 159 sourceb.string.free_after_use = true;
160 160 sourceb.string.text.end =
161 161 sourceb.string.buffer.end =
162 162 sourceb.string.text.p + value->hash.length;
163 163 sourceb.previous = NULL;
164 164 sourceb.fd = -1;
165 165 sourceb.inp_buf =
166 166 sourceb.inp_buf_ptr =
167 167 sourceb.inp_buf_end = NULL;
168 168 sourceb.error_converting = false;
169 169 /* Lift some pointers from the struct to local register variables */
170 170 CACHE_SOURCE(0);
171 171 /* We parse the string in segments */
172 172 /* We read chars until we find a $, then we append what we have read so far */
173 173 /* (since last $ processing) to the destination. When we find a $ we call */
174 174 /* expand_macro() and let it expand that particular $ reference into dest */
175 175 block_start = source_p;
176 176 quote_seen = 0;
177 177 for (; 1; source_p++) {
178 178 switch (GET_CHAR()) {
179 179 case backslash_char:
180 180 /* Quote $ in macro value */
181 181 if (!cmd) {
182 182 quote_seen = ~quote_seen;
183 183 }
184 184 continue;
185 185 case dollar_char:
186 186 /* Save the plain string we found since */
187 187 /* start of string or previous $ */
188 188 if (quote_seen) {
189 189 append_string(block_start,
190 190 destination,
191 191 source_p - block_start - 1);
192 192 block_start = source_p;
193 193 break;
194 194 }
195 195 append_string(block_start,
196 196 destination,
197 197 source_p - block_start);
198 198 source->string.text.p = ++source_p;
199 199 UNCACHE_SOURCE();
200 200 /* Go expand the macro reference */
201 201 expand_macro(source, destination, sourceb.string.buffer.start, cmd);
202 202 CACHE_SOURCE(1);
203 203 block_start = source_p + 1;
204 204 break;
205 205 case nul_char:
206 206 /* The string ran out. Get some more */
207 207 append_string(block_start,
208 208 destination,
209 209 source_p - block_start);
210 210 GET_NEXT_BLOCK_NOCHK(source);
211 211 if (source == NULL) {
212 212 destination->text.end = destination->text.p;
213 213 value->being_expanded = false;
214 214 return;
215 215 }
216 216 if (source->error_converting) {
217 217 fatal_reader_mksh(NOCATGETS("Internal error: Invalid byte sequence in expand_value()"));
218 218 }
219 219 block_start = source_p;
220 220 source_p--;
221 221 continue;
222 222 }
223 223 quote_seen = 0;
224 224 }
225 225 retmem(sourceb.string.buffer.start);
226 226 }
227 227
228 228 /*
229 229 * expand_macro(source, destination, current_string, cmd)
230 230 *
231 231 * Should be called with source->string.text.p pointing to
232 232 * the first char after the $ that starts a macro reference.
233 233 * source->string.text.p is returned pointing to the first char after
234 234 * the macro name.
235 235 * It will read the macro name, expanding any macros in it,
236 236 * and get the value. The value is then expanded.
237 237 * destination is a String that is filled in with the expanded macro.
238 238 * It may be passed in referencing a buffer to expand the macro into.
239 239 * Note that most expansions are done on demand, e.g. right
240 240 * before the command is executed and not while the file is
241 241 * being parsed.
242 242 *
243 243 * Parameters:
244 244 * source The source block that references the string
245 245 * to expand
246 246 * destination Where to put the result
247 247 * current_string The string we are expanding, for error msg
248 248 * cmd If we are evaluating a command line we
249 249 * turn \ quoting off
250 250 *
251 251 * Global variables used:
252 252 * funny Vector of semantic tags for characters
253 253 * is_conditional Set if a conditional macro is refd
254 254 * make_word_mentioned Set if the word "MAKE" is mentioned
255 255 * makefile_type We deliver extra msg when reading makefiles
256 256 * query The Name "?", compared against
257 257 * query_mentioned Set if the word "?" is mentioned
258 258 */
259 259 void
260 260 expand_macro(register Source source, register String destination, wchar_t *current_string, Boolean cmd)
261 261 {
262 262 static Name make = (Name)NULL;
263 263 static wchar_t colon_sh[4];
264 264 static wchar_t colon_shell[7];
265 265 String_rec string;
266 266 wchar_t buffer[STRING_BUFFER_LENGTH];
267 267 register wchar_t *source_p = source->string.text.p;
268 268 register wchar_t *source_end = source->string.text.end;
269 269 register int closer = 0;
270 270 wchar_t *block_start = (wchar_t *)NULL;
271 271 int quote_seen = 0;
272 272 register int closer_level = 1;
273 273 Name name = (Name)NULL;
274 274 wchar_t *colon = (wchar_t *)NULL;
275 275 wchar_t *percent = (wchar_t *)NULL;
276 276 wchar_t *eq = (wchar_t *) NULL;
277 277 Property macro = NULL;
278 278 wchar_t *p = (wchar_t*)NULL;
279 279 String_rec extracted;
280 280 wchar_t extracted_string[MAXPATHLEN];
281 281 wchar_t *left_head = NULL;
282 282 wchar_t *left_tail = NULL;
283 283 wchar_t *right_tail = NULL;
284 284 int left_head_len = 0;
285 285 int left_tail_len = 0;
286 286 int tmp_len = 0;
287 287 wchar_t *right_hand[128];
288 288 int i = 0;
289 289 enum {
290 290 no_extract,
291 291 dir_extract,
292 292 file_extract
293 293 } extraction = no_extract;
294 294 enum {
295 295 no_replace,
296 296 suffix_replace,
297 297 pattern_replace,
298 298 sh_replace
299 299 } replacement = no_replace;
300 300
301 301 if (make == NULL) {
302 302 MBSTOWCS(wcs_buffer, NOCATGETS("MAKE"));
303 303 make = GETNAME(wcs_buffer, FIND_LENGTH);
304 304
305 305 MBSTOWCS(colon_sh, NOCATGETS(":sh"));
306 306 MBSTOWCS(colon_shell, NOCATGETS(":shell"));
307 307 }
308 308
309 309 right_hand[0] = NULL;
310 310
311 311 /* First copy the (macro-expanded) macro name into string. */
312 312 INIT_STRING_FROM_STACK(string, buffer);
313 313 recheck_first_char:
314 314 /* Check the first char of the macro name to figure out what to do. */
315 315 switch (GET_CHAR()) {
316 316 case nul_char:
317 317 GET_NEXT_BLOCK_NOCHK(source);
318 318 if (source == NULL) {
319 319 WCSTOMBS(mbs_buffer, current_string);
320 320 fatal_reader_mksh(catgets(libmksdmsi18n_catd, 1, 114, "'$' at end of string `%s'"),
321 321 mbs_buffer);
322 322 }
323 323 if (source->error_converting) {
324 324 fatal_reader_mksh(NOCATGETS("Internal error: Invalid byte sequence in expand_macro()"));
325 325 }
326 326 goto recheck_first_char;
327 327 case parenleft_char:
328 328 /* Multi char name. */
329 329 closer = (int) parenright_char;
330 330 break;
331 331 case braceleft_char:
332 332 /* Multi char name. */
333 333 closer = (int) braceright_char;
334 334 break;
335 335 case newline_char:
336 336 fatal_reader_mksh(catgets(libmksdmsi18n_catd, 1, 115, "'$' at end of line"));
337 337 default:
338 338 /* Single char macro name. Just suck it up */
339 339 append_char(*source_p, &string);
340 340 source->string.text.p = source_p + 1;
341 341 goto get_macro_value;
342 342 }
343 343
344 344 /* Handle multi-char macro names */
345 345 block_start = ++source_p;
346 346 quote_seen = 0;
347 347 for (; 1; source_p++) {
348 348 switch (GET_CHAR()) {
349 349 case nul_char:
350 350 append_string(block_start,
351 351 &string,
352 352 source_p - block_start);
353 353 GET_NEXT_BLOCK_NOCHK(source);
354 354 if (source == NULL) {
355 355 if (current_string != NULL) {
356 356 WCSTOMBS(mbs_buffer, current_string);
357 357 fatal_reader_mksh(catgets(libmksdmsi18n_catd, 1, 116, "Unmatched `%c' in string `%s'"),
358 358 closer ==
359 359 (int) braceright_char ?
360 360 (int) braceleft_char :
361 361 (int) parenleft_char,
362 362 mbs_buffer);
363 363 } else {
364 364 fatal_reader_mksh(catgets(libmksdmsi18n_catd, 1, 117, "Premature EOF"));
365 365 }
366 366 }
367 367 if (source->error_converting) {
368 368 fatal_reader_mksh(NOCATGETS("Internal error: Invalid byte sequence in expand_macro()"));
369 369 }
370 370 block_start = source_p;
371 371 source_p--;
372 372 continue;
373 373 case newline_char:
374 374 fatal_reader_mksh(catgets(libmksdmsi18n_catd, 1, 118, "Unmatched `%c' on line"),
375 375 closer == (int) braceright_char ?
376 376 (int) braceleft_char :
377 377 (int) parenleft_char);
378 378 case backslash_char:
379 379 /* Quote dollar in macro value. */
380 380 if (!cmd) {
381 381 quote_seen = ~quote_seen;
382 382 }
383 383 continue;
384 384 case dollar_char:
385 385 /*
386 386 * Macro names may reference macros.
387 387 * This expands the value of such macros into the
388 388 * macro name string.
389 389 */
390 390 if (quote_seen) {
391 391 append_string(block_start,
392 392 &string,
393 393 source_p - block_start - 1);
394 394 block_start = source_p;
395 395 break;
396 396 }
397 397 append_string(block_start,
398 398 &string,
399 399 source_p - block_start);
400 400 source->string.text.p = ++source_p;
401 401 UNCACHE_SOURCE();
402 402 expand_macro(source, &string, current_string, cmd);
403 403 CACHE_SOURCE(0);
404 404 block_start = source_p;
405 405 source_p--;
406 406 break;
407 407 case parenleft_char:
408 408 /* Allow nested pairs of () in the macro name. */
409 409 if (closer == (int) parenright_char) {
410 410 closer_level++;
411 411 }
412 412 break;
413 413 case braceleft_char:
414 414 /* Allow nested pairs of {} in the macro name. */
415 415 if (closer == (int) braceright_char) {
416 416 closer_level++;
417 417 }
418 418 break;
419 419 case parenright_char:
420 420 case braceright_char:
421 421 /*
422 422 * End of the name. Save the string in the macro
423 423 * name string.
424 424 */
425 425 if ((*source_p == closer) && (--closer_level <= 0)) {
426 426 source->string.text.p = source_p + 1;
427 427 append_string(block_start,
428 428 &string,
429 429 source_p - block_start);
430 430 goto get_macro_value;
431 431 }
432 432 break;
433 433 }
434 434 quote_seen = 0;
435 435 }
436 436 /*
437 437 * We got the macro name. We now inspect it to see if it
438 438 * specifies any translations of the value.
439 439 */
440 440 get_macro_value:
441 441 name = NULL;
442 442 /* First check if we have a $(@D) type translation. */
443 443 if ((get_char_semantics_value(string.buffer.start[0]) &
444 444 (int) special_macro_sem) &&
445 445 (string.text.p - string.buffer.start >= 2) &&
446 446 ((string.buffer.start[1] == 'D') ||
447 447 (string.buffer.start[1] == 'F'))) {
448 448 switch (string.buffer.start[1]) {
449 449 case 'D':
450 450 extraction = dir_extract;
451 451 break;
452 452 case 'F':
453 453 extraction = file_extract;
454 454 break;
455 455 default:
456 456 WCSTOMBS(mbs_buffer, string.buffer.start);
457 457 fatal_reader_mksh(catgets(libmksdmsi18n_catd, 1, 119, "Illegal macro reference `%s'"),
458 458 mbs_buffer);
459 459 }
460 460 /* Internalize the macro name using the first char only. */
461 461 name = GETNAME(string.buffer.start, 1);
462 462 (void) wscpy(string.buffer.start, string.buffer.start + 2);
463 463 }
464 464 /* Check for other kinds of translations. */
465 465 if ((colon = (wchar_t *) wschr(string.buffer.start,
466 466 (int) colon_char)) != NULL) {
467 467 /*
468 468 * We have a $(FOO:.c=.o) type translation.
469 469 * Get the name of the macro proper.
470 470 */
471 471 if (name == NULL) {
472 472 name = GETNAME(string.buffer.start,
473 473 colon - string.buffer.start);
474 474 }
475 475 /* Pickup all the translations. */
476 476 if (IS_WEQUAL(colon, colon_sh) || IS_WEQUAL(colon, colon_shell)) {
477 477 replacement = sh_replace;
478 478 } else if ((svr4) ||
479 479 ((percent = (wchar_t *) wschr(colon + 1,
480 480 (int) percent_char)) == NULL)) {
481 481 while (colon != NULL) {
482 482 if ((eq = (wchar_t *) wschr(colon + 1,
483 483 (int) equal_char)) == NULL) {
484 484 fatal_reader_mksh(catgets(libmksdmsi18n_catd, 1, 120, "= missing from replacement macro reference"));
485 485 }
486 486 left_tail_len = eq - colon - 1;
487 487 if(left_tail) {
488 488 retmem(left_tail);
489 489 }
490 490 left_tail = ALLOC_WC(left_tail_len + 1);
491 491 (void) wsncpy(left_tail,
492 492 colon + 1,
493 493 eq - colon - 1);
494 494 left_tail[eq - colon - 1] = (int) nul_char;
495 495 replacement = suffix_replace;
496 496 if ((colon = (wchar_t *) wschr(eq + 1,
497 497 (int) colon_char)) != NULL) {
498 498 tmp_len = colon - eq;
499 499 if(right_tail) {
500 500 retmem(right_tail);
501 501 }
502 502 right_tail = ALLOC_WC(tmp_len);
503 503 (void) wsncpy(right_tail,
504 504 eq + 1,
505 505 colon - eq - 1);
506 506 right_tail[colon - eq - 1] =
507 507 (int) nul_char;
508 508 } else {
509 509 if(right_tail) {
510 510 retmem(right_tail);
511 511 }
512 512 right_tail = ALLOC_WC(wslen(eq) + 1);
513 513 (void) wscpy(right_tail, eq + 1);
514 514 }
515 515 }
516 516 } else {
517 517 if ((eq = (wchar_t *) wschr(colon + 1,
518 518 (int) equal_char)) == NULL) {
519 519 fatal_reader_mksh(catgets(libmksdmsi18n_catd, 1, 121, "= missing from replacement macro reference"));
520 520 }
521 521 if ((percent = (wchar_t *) wschr(colon + 1,
522 522 (int) percent_char)) == NULL) {
523 523 fatal_reader_mksh(catgets(libmksdmsi18n_catd, 1, 122, "%% missing from replacement macro reference"));
524 524 }
525 525 if (eq < percent) {
526 526 fatal_reader_mksh(catgets(libmksdmsi18n_catd, 1, 123, "%% missing from replacement macro reference"));
527 527 }
528 528
529 529 if (percent > (colon + 1)) {
530 530 tmp_len = percent - colon;
531 531 if(left_head) {
532 532 retmem(left_head);
533 533 }
534 534 left_head = ALLOC_WC(tmp_len);
535 535 (void) wsncpy(left_head,
536 536 colon + 1,
537 537 percent - colon - 1);
538 538 left_head[percent-colon-1] = (int) nul_char;
539 539 left_head_len = percent-colon-1;
540 540 } else {
541 541 left_head = NULL;
542 542 left_head_len = 0;
543 543 }
544 544
545 545 if (eq > percent+1) {
546 546 tmp_len = eq - percent;
547 547 if(left_tail) {
548 548 retmem(left_tail);
549 549 }
550 550 left_tail = ALLOC_WC(tmp_len);
551 551 (void) wsncpy(left_tail,
552 552 percent + 1,
553 553 eq - percent - 1);
554 554 left_tail[eq-percent-1] = (int) nul_char;
555 555 left_tail_len = eq-percent-1;
556 556 } else {
557 557 left_tail = NULL;
558 558 left_tail_len = 0;
559 559 }
560 560
561 561 if ((percent = (wchar_t *) wschr(++eq,
562 562 (int) percent_char)) == NULL) {
563 563
564 564 right_hand[0] = ALLOC_WC(wslen(eq) + 1);
565 565 right_hand[1] = NULL;
566 566 (void) wscpy(right_hand[0], eq);
567 567 } else {
568 568 i = 0;
569 569 do {
570 570 right_hand[i] = ALLOC_WC(percent-eq+1);
571 571 (void) wsncpy(right_hand[i],
572 572 eq,
573 573 percent - eq);
574 574 right_hand[i][percent-eq] =
575 575 (int) nul_char;
576 576 if (i++ >= VSIZEOF(right_hand)) {
577 577 fatal_mksh(catgets(libmksdmsi18n_catd, 1, 124, "Too many %% in pattern"));
578 578 }
579 579 eq = percent + 1;
580 580 if (eq[0] == (int) nul_char) {
581 581 MBSTOWCS(wcs_buffer, "");
582 582 right_hand[i] = (wchar_t *) wsdup(wcs_buffer);
583 583 i++;
584 584 break;
585 585 }
586 586 } while ((percent = (wchar_t *) wschr(eq, (int) percent_char)) != NULL);
587 587 if (eq[0] != (int) nul_char) {
588 588 right_hand[i] = ALLOC_WC(wslen(eq) + 1);
589 589 (void) wscpy(right_hand[i], eq);
590 590 i++;
591 591 }
592 592 right_hand[i] = NULL;
593 593 }
594 594 replacement = pattern_replace;
595 595 }
596 596 }
597 597 if (name == NULL) {
598 598 /*
599 599 * No translations found.
600 600 * Use the whole string as the macro name.
601 601 */
602 602 name = GETNAME(string.buffer.start,
603 603 string.text.p - string.buffer.start);
604 604 }
605 605 if (string.free_after_use) {
606 606 retmem(string.buffer.start);
607 607 }
608 608 if (name == make) {
609 609 make_word_mentioned = true;
610 610 }
611 611 if (name == query) {
612 612 query_mentioned = true;
613 613 }
614 614 if ((name == host_arch) || (name == target_arch)) {
615 615 if (!init_arch_done) {
616 616 init_arch_done = true;
617 617 init_arch_macros();
618 618 }
619 619 }
620 620 if ((name == host_mach) || (name == target_mach)) {
621 621 if (!init_mach_done) {
622 622 init_mach_done = true;
623 623 init_mach_macros();
624 624 }
625 625 }
626 626 /* Get the macro value. */
627 627 macro = get_prop(name->prop, macro_prop);
628 628 if ((macro != NULL) && macro->body.macro.is_conditional) {
629 629 conditional_macro_used = true;
630 630 /*
631 631 * Add this conditional macro to the beginning of the
632 632 * global list.
633 633 */
634 634 add_macro_to_global_list(name);
635 635 if (makefile_type == reading_makefile) {
636 636 warning_mksh(catgets(libmksdmsi18n_catd, 1, 164, "Conditional macro `%s' referenced in file `%ws', line %d"),
637 637 name->string_mb, file_being_read, line_number);
638 638 }
639 639 }
640 640 /* Macro name read and parsed. Expand the value. */
641 641 if ((macro == NULL) || (macro->body.macro.value == NULL)) {
642 642 /* If the value is empty, we just get out of here. */
643 643 goto exit;
644 644 }
645 645 if (replacement == sh_replace) {
646 646 /* If we should do a :sh transform, we expand the command
647 647 * and process it.
648 648 */
649 649 INIT_STRING_FROM_STACK(string, buffer);
650 650 /* Expand the value into a local string buffer and run cmd. */
651 651 expand_value_with_daemon(name, macro, &string, cmd);
652 652 sh_command2string(&string, destination);
653 653 } else if ((replacement != no_replace) || (extraction != no_extract)) {
654 654 /*
655 655 * If there were any transforms specified in the macro
656 656 * name, we deal with them here.
657 657 */
658 658 INIT_STRING_FROM_STACK(string, buffer);
659 659 /* Expand the value into a local string buffer. */
660 660 expand_value_with_daemon(name, macro, &string, cmd);
661 661 /* Scan the expanded string. */
662 662 p = string.buffer.start;
663 663 while (*p != (int) nul_char) {
664 664 wchar_t chr;
665 665
666 666 /*
667 667 * First skip over any white space and append
668 668 * that to the destination string.
669 669 */
670 670 block_start = p;
671 671 while ((*p != (int) nul_char) && iswspace(*p)) {
672 672 p++;
673 673 }
674 674 append_string(block_start,
675 675 destination,
676 676 p - block_start);
677 677 /* Then find the end of the next word. */
678 678 block_start = p;
679 679 while ((*p != (int) nul_char) && !iswspace(*p)) {
680 680 p++;
681 681 }
682 682 /* If we cant find another word we are done */
683 683 if (block_start == p) {
684 684 break;
685 685 }
686 686 /* Then apply the transforms to the word */
687 687 INIT_STRING_FROM_STACK(extracted, extracted_string);
688 688 switch (extraction) {
689 689 case dir_extract:
690 690 /*
691 691 * $(@D) type transform. Extract the
692 692 * path from the word. Deliver "." if
693 693 * none is found.
694 694 */
695 695 if (p != NULL) {
696 696 chr = *p;
697 697 *p = (int) nul_char;
698 698 }
699 699 eq = (wchar_t *) wsrchr(block_start, (int) slash_char);
700 700 if (p != NULL) {
701 701 *p = chr;
702 702 }
703 703 if ((eq == NULL) || (eq > p)) {
704 704 MBSTOWCS(wcs_buffer, ".");
705 705 append_string(wcs_buffer, &extracted, 1);
706 706 } else {
707 707 append_string(block_start,
708 708 &extracted,
709 709 eq - block_start);
710 710 }
711 711 break;
712 712 case file_extract:
713 713 /*
714 714 * $(@F) type transform. Remove the path
715 715 * from the word if any.
716 716 */
717 717 if (p != NULL) {
718 718 chr = *p;
719 719 *p = (int) nul_char;
720 720 }
721 721 eq = (wchar_t *) wsrchr(block_start, (int) slash_char);
722 722 if (p != NULL) {
723 723 *p = chr;
724 724 }
725 725 if ((eq == NULL) || (eq > p)) {
726 726 append_string(block_start,
727 727 &extracted,
728 728 p - block_start);
729 729 } else {
730 730 append_string(eq + 1,
731 731 &extracted,
732 732 p - eq - 1);
733 733 }
734 734 break;
735 735 case no_extract:
736 736 append_string(block_start,
737 737 &extracted,
738 738 p - block_start);
739 739 break;
740 740 }
741 741 switch (replacement) {
742 742 case suffix_replace:
743 743 /*
744 744 * $(FOO:.o=.c) type transform.
745 745 * Maybe replace the tail of the word.
746 746 */
747 747 if (((extracted.text.p -
748 748 extracted.buffer.start) >=
749 749 left_tail_len) &&
750 750 IS_WEQUALN(extracted.text.p - left_tail_len,
751 751 left_tail,
752 752 left_tail_len)) {
753 753 append_string(extracted.buffer.start,
754 754 destination,
755 755 (extracted.text.p -
756 756 extracted.buffer.start)
757 757 - left_tail_len);
758 758 append_string(right_tail,
759 759 destination,
760 760 FIND_LENGTH);
761 761 } else {
762 762 append_string(extracted.buffer.start,
763 763 destination,
764 764 FIND_LENGTH);
765 765 }
766 766 break;
767 767 case pattern_replace:
768 768 /* $(X:a%b=c%d) type transform. */
769 769 if (((extracted.text.p -
770 770 extracted.buffer.start) >=
771 771 left_head_len+left_tail_len) &&
772 772 IS_WEQUALN(left_head,
773 773 extracted.buffer.start,
774 774 left_head_len) &&
775 775 IS_WEQUALN(left_tail,
776 776 extracted.text.p - left_tail_len,
777 777 left_tail_len)) {
778 778 i = 0;
779 779 while (right_hand[i] != NULL) {
780 780 append_string(right_hand[i],
781 781 destination,
782 782 FIND_LENGTH);
783 783 i++;
784 784 if (right_hand[i] != NULL) {
785 785 append_string(extracted.buffer.
786 786 start +
787 787 left_head_len,
788 788 destination,
789 789 (extracted.text.p - extracted.buffer.start)-left_head_len-left_tail_len);
790 790 }
791 791 }
792 792 } else {
793 793 append_string(extracted.buffer.start,
794 794 destination,
795 795 FIND_LENGTH);
796 796 }
797 797 break;
798 798 case no_replace:
799 799 append_string(extracted.buffer.start,
800 800 destination,
801 801 FIND_LENGTH);
802 802 break;
803 803 case sh_replace:
804 804 break;
805 805 }
806 806 }
807 807 if (string.free_after_use) {
808 808 retmem(string.buffer.start);
809 809 }
810 810 } else {
811 811 /*
812 812 * This is for the case when the macro name did not
813 813 * specify transforms.
814 814 */
815 815 if (!strncmp(name->string_mb, NOCATGETS("GET"), 3)) {
816 816 dollarget_seen = true;
817 817 }
818 818 dollarless_flag = false;
819 819 if (!strncmp(name->string_mb, "<", 1) &&
820 820 dollarget_seen) {
821 821 dollarless_flag = true;
822 822 dollarget_seen = false;
823 823 }
824 824 expand_value_with_daemon(name, macro, destination, cmd);
825 825 }
826 826 exit:
827 827 if(left_tail) {
828 828 retmem(left_tail);
829 829 }
830 830 if(right_tail) {
831 831 retmem(right_tail);
832 832 }
833 833 if(left_head) {
834 834 retmem(left_head);
835 835 }
836 836 i = 0;
837 837 while (right_hand[i] != NULL) {
838 838 retmem(right_hand[i]);
839 839 i++;
840 840 }
841 841 *destination->text.p = (int) nul_char;
842 842 destination->text.end = destination->text.p;
843 843 }
844 844
845 845 static void
846 846 add_macro_to_global_list(Name macro_to_add)
847 847 {
848 848 Macro_list new_macro;
849 849 Macro_list macro_on_list;
850 850 char *name_on_list = (char*)NULL;
851 851 char *name_to_add = macro_to_add->string_mb;
852 852 char *value_on_list = (char*)NULL;
853 853 const char *value_to_add = (char*)NULL;
854 854
855 855 if (macro_to_add->prop->body.macro.value != NULL) {
856 856 value_to_add = macro_to_add->prop->body.macro.value->string_mb;
857 857 } else {
858 858 value_to_add = "";
859 859 }
860 860
861 861 /*
862 862 * Check if this macro is already on list, if so, do nothing
863 863 */
864 864 for (macro_on_list = cond_macro_list;
865 865 macro_on_list != NULL;
866 866 macro_on_list = macro_on_list->next) {
867 867
868 868 name_on_list = macro_on_list->macro_name;
869 869 value_on_list = macro_on_list->value;
870 870
871 871 if (IS_EQUAL(name_on_list, name_to_add)) {
872 872 if (IS_EQUAL(value_on_list, value_to_add)) {
873 873 return;
874 874 }
875 875 }
876 876 }
877 877 new_macro = (Macro_list) malloc(sizeof(Macro_list_rec));
878 878 new_macro->macro_name = strdup(name_to_add);
879 879 new_macro->value = strdup(value_to_add);
880 880 new_macro->next = cond_macro_list;
881 881 cond_macro_list = new_macro;
882 882 }
883 883
884 884 /*
885 885 * init_arch_macros(void)
886 886 *
887 887 * Set the magic macros TARGET_ARCH, HOST_ARCH,
888 888 *
889 889 * Parameters:
890 890 *
891 891 * Global variables used:
892 892 * host_arch Property for magic macro HOST_ARCH
893 893 * target_arch Property for magic macro TARGET_ARCH
894 894 *
895 895 * Return value:
896 896 * The function does not return a value, but can
897 897 * call fatal() in case of error.
898 898 */
899 899 static void
900 900 init_arch_macros(void)
901 901 {
902 902 String_rec result_string;
903 903 wchar_t wc_buf[STRING_BUFFER_LENGTH];
904 904 char mb_buf[STRING_BUFFER_LENGTH];
905 905 FILE *pipe;
906 906 Name value;
907 907 int set_host, set_target;
908 908 const char *mach_command = NOCATGETS("/bin/mach");
909 909
910 910 set_host = (get_prop(host_arch->prop, macro_prop) == NULL);
911 911 set_target = (get_prop(target_arch->prop, macro_prop) == NULL);
912 912
913 913 if (set_host || set_target) {
914 914 INIT_STRING_FROM_STACK(result_string, wc_buf);
915 915 append_char((int) hyphen_char, &result_string);
916 916
917 917 if ((pipe = popen(mach_command, "r")) == NULL) {
918 918 fatal_mksh(catgets(libmksdmsi18n_catd, 1, 185, "Execute of %s failed"), mach_command);
919 919 }
920 920 while (fgets(mb_buf, sizeof(mb_buf), pipe) != NULL) {
921 921 MBSTOWCS(wcs_buffer, mb_buf);
922 922 append_string(wcs_buffer, &result_string, wslen(wcs_buffer));
923 923 }
924 924 if (pclose(pipe) != 0) {
925 925 fatal_mksh(catgets(libmksdmsi18n_catd, 1, 186, "Execute of %s failed"), mach_command);
926 926 }
927 927
928 928 value = GETNAME(result_string.buffer.start, wslen(result_string.buffer.start));
929 929
930 930 if (set_host) {
931 931 (void) setvar_daemon(host_arch, value, false, no_daemon, true, 0);
932 932 }
933 933 if (set_target) {
934 934 (void) setvar_daemon(target_arch, value, false, no_daemon, true, 0);
935 935 }
936 936 }
937 937 }
938 938
939 939 /*
940 940 * init_mach_macros(void)
941 941 *
942 942 * Set the magic macros TARGET_MACH, HOST_MACH,
943 943 *
944 944 * Parameters:
945 945 *
946 946 * Global variables used:
947 947 * host_mach Property for magic macro HOST_MACH
948 948 * target_mach Property for magic macro TARGET_MACH
949 949 *
950 950 * Return value:
951 951 * The function does not return a value, but can
952 952 * call fatal() in case of error.
953 953 */
954 954 static void
955 955 init_mach_macros(void)
956 956 {
957 957 String_rec result_string;
958 958 wchar_t wc_buf[STRING_BUFFER_LENGTH];
959 959 char mb_buf[STRING_BUFFER_LENGTH];
960 960 FILE *pipe;
961 961 Name value;
962 962 int set_host, set_target;
963 963 const char *arch_command = NOCATGETS("/bin/arch");
964 964
965 965 set_host = (get_prop(host_mach->prop, macro_prop) == NULL);
966 966 set_target = (get_prop(target_mach->prop, macro_prop) == NULL);
967 967
968 968 if (set_host || set_target) {
969 969 INIT_STRING_FROM_STACK(result_string, wc_buf);
970 970 append_char((int) hyphen_char, &result_string);
971 971
972 972 if ((pipe = popen(arch_command, "r")) == NULL) {
973 973 fatal_mksh(catgets(libmksdmsi18n_catd, 1, 183, "Execute of %s failed"), arch_command);
974 974 }
975 975 while (fgets(mb_buf, sizeof(mb_buf), pipe) != NULL) {
976 976 MBSTOWCS(wcs_buffer, mb_buf);
977 977 append_string(wcs_buffer, &result_string, wslen(wcs_buffer));
978 978 }
979 979 if (pclose(pipe) != 0) {
980 980 fatal_mksh(catgets(libmksdmsi18n_catd, 1, 184, "Execute of %s failed"), arch_command);
981 981 }
982 982
983 983 value = GETNAME(result_string.buffer.start, wslen(result_string.buffer.start));
984 984
985 985 if (set_host) {
986 986 (void) setvar_daemon(host_mach, value, false, no_daemon, true, 0);
987 987 }
988 988 if (set_target) {
989 989 (void) setvar_daemon(target_mach, value, false, no_daemon, true, 0);
990 990 }
991 991 }
992 992 }
993 993
994 994 /*
995 995 * expand_value_with_daemon(name, macro, destination, cmd)
996 996 *
997 997 * Checks for daemons and then maybe calls expand_value().
998 998 *
999 999 * Parameters:
1000 1000 * name Name of the macro (Added by the NSE)
1001 1001 * macro The property block with the value to expand
1002 1002 * destination Where the result should be deposited
1003 1003 * cmd If we are evaluating a command line we
1004 1004 * turn \ quoting off
1005 1005 *
1006 1006 * Global variables used:
1007 1007 */
1008 1008 static void
1009 1009 expand_value_with_daemon(Name, register Property macro, register String destination, Boolean cmd)
1010 1010 {
1011 1011 register Chain chain;
1012 1012
1013 1013
1014 1014 switch (macro->body.macro.daemon) {
1015 1015 case no_daemon:
1016 1016 if (!svr4 && !posix) {
1017 1017 expand_value(macro->body.macro.value, destination, cmd);
1018 1018 } else {
1019 1019 if (dollarless_flag && tilde_rule) {
1020 1020 expand_value(dollarless_value, destination, cmd);
1021 1021 dollarless_flag = false;
1022 1022 tilde_rule = false;
1023 1023 } else {
1024 1024 expand_value(macro->body.macro.value, destination, cmd);
1025 1025 }
1026 1026 }
1027 1027 return;
1028 1028 case chain_daemon:
1029 1029 /* If this is a $? value we call the daemon to translate the */
1030 1030 /* list of names to a string */
1031 1031 for (chain = (Chain) macro->body.macro.value;
1032 1032 chain != NULL;
1033 1033 chain = chain->next) {
1034 1034 APPEND_NAME(chain->name,
1035 1035 destination,
1036 1036 (int) chain->name->hash.length);
1037 1037 if (chain->next != NULL) {
1038 1038 append_char((int) space_char, destination);
1039 1039 }
1040 1040 }
1041 1041 return;
1042 1042 }
1043 1043 }
1044 1044
1045 1045 /*
1046 1046 * We use a permanent buffer to reset SUNPRO_DEPENDENCIES value.
1047 1047 */
1048 1048 char *sunpro_dependencies_buf = NULL;
1049 1049 char *sunpro_dependencies_oldbuf = NULL;
1050 1050 int sunpro_dependencies_buf_size = 0;
1051 1051
1052 1052 /*
1053 1053 * setvar_daemon(name, value, append, daemon, strip_trailing_spaces)
1054 1054 *
1055 1055 * Set a macro value, possibly supplying a daemon to be used
1056 1056 * when referencing the value.
1057 1057 *
1058 1058 * Return value:
1059 1059 * The property block with the new value
1060 1060 *
1061 1061 * Parameters:
1062 1062 * name Name of the macro to set
1063 1063 * value The value to set
1064 1064 * append Should we reset or append to the current value?
1065 1065 * daemon Special treatment when reading the value
1066 1066 * strip_trailing_spaces from the end of value->string
1067 1067 * debug_level Indicates how much tracing we should do
1068 1068 *
1069 1069 * Global variables used:
1070 1070 * makefile_type Used to check if we should enforce read only
1071 1071 * path_name The Name "PATH", compared against
1072 1072 * virtual_root The Name "VIRTUAL_ROOT", compared against
1073 1073 * vpath_defined Set if the macro VPATH is set
1074 1074 * vpath_name The Name "VPATH", compared against
1075 1075 * envvar A list of environment vars with $ in value
1076 1076 */
1077 1077 Property
1078 1078 setvar_daemon(register Name name, register Name value, Boolean append, Daemon daemon, Boolean strip_trailing_spaces, short debug_level)
1079 1079 {
1080 1080 register Property macro = maybe_append_prop(name, macro_prop);
1081 1081 register Property macro_apx = get_prop(name->prop, macro_append_prop);
1082 1082 int length = 0;
1083 1083 String_rec destination;
1084 1084 wchar_t buffer[STRING_BUFFER_LENGTH];
1085 1085 register Chain chain;
1086 1086 Name val;
1087 1087 wchar_t *val_string = (wchar_t*)NULL;
1088 1088 Wstring wcb;
1089 1089
1090 1090
1091 1091 if ((makefile_type != reading_nothing) &&
1092 1092 macro->body.macro.read_only) {
1093 1093 return macro;
1094 1094 }
1095 1095 /* Strip spaces from the end of the value */
1096 1096 if (daemon == no_daemon) {
1097 1097 if(value != NULL) {
1098 1098 wcb.init(value);
1099 1099 length = wcb.length();
1100 1100 val_string = wcb.get_string();
1101 1101 }
1102 1102 if ((length > 0) && iswspace(val_string[length-1])) {
1103 1103 INIT_STRING_FROM_STACK(destination, buffer);
1104 1104 buffer[0] = 0;
1105 1105 append_string(val_string, &destination, length);
1106 1106 if (strip_trailing_spaces) {
1107 1107 while ((length > 0) &&
1108 1108 iswspace(destination.buffer.start[length-1])) {
1109 1109 destination.buffer.start[--length] = 0;
1110 1110 }
1111 1111 }
1112 1112 value = GETNAME(destination.buffer.start, FIND_LENGTH);
1113 1113 }
1114 1114 }
1115 1115
1116 1116 if(macro_apx != NULL) {
1117 1117 val = macro_apx->body.macro_appendix.value;
1118 1118 } else {
1119 1119 val = macro->body.macro.value;
1120 1120 }
1121 1121
1122 1122 if (append) {
1123 1123 /*
1124 1124 * If we are appending, we just tack the new value after
1125 1125 * the old one with a space in between.
1126 1126 */
1127 1127 INIT_STRING_FROM_STACK(destination, buffer);
1128 1128 buffer[0] = 0;
1129 1129 if ((macro != NULL) && (val != NULL)) {
1130 1130 APPEND_NAME(val,
1131 1131 &destination,
1132 1132 (int) val->hash.length);
1133 1133 if (value != NULL) {
1134 1134 wcb.init(value);
1135 1135 if(wcb.length() > 0) {
1136 1136 MBTOWC(wcs_buffer, " ");
1137 1137 append_char(wcs_buffer[0], &destination);
1138 1138 }
1139 1139 }
1140 1140 }
1141 1141 if (value != NULL) {
1142 1142 APPEND_NAME(value,
1143 1143 &destination,
1144 1144 (int) value->hash.length);
1145 1145 }
1146 1146 value = GETNAME(destination.buffer.start, FIND_LENGTH);
1147 1147 wcb.init(value);
1148 1148 if (destination.free_after_use) {
1149 1149 retmem(destination.buffer.start);
1150 1150 }
1151 1151 }
1152 1152
1153 1153 /* Debugging trace */
1154 1154 if (debug_level > 1) {
1155 1155 if (value != NULL) {
1156 1156 switch (daemon) {
1157 1157 case chain_daemon:
1158 1158 (void) printf("%s =", name->string_mb);
1159 1159 for (chain = (Chain) value;
1160 1160 chain != NULL;
1161 1161 chain = chain->next) {
1162 1162 (void) printf(" %s", chain->name->string_mb);
1163 1163 }
1164 1164 (void) printf("\n");
1165 1165 break;
1166 1166 case no_daemon:
1167 1167 (void) printf("%s= %s\n",
1168 1168 name->string_mb,
1169 1169 value->string_mb);
1170 1170 break;
1171 1171 }
1172 1172 } else {
1173 1173 (void) printf("%s =\n", name->string_mb);
1174 1174 }
1175 1175 }
1176 1176 /* Set the new values in the macro property block */
1177 1177 /**/
1178 1178 if(macro_apx != NULL) {
1179 1179 macro_apx->body.macro_appendix.value = value;
1180 1180 INIT_STRING_FROM_STACK(destination, buffer);
1181 1181 buffer[0] = 0;
1182 1182 if (value != NULL) {
1183 1183 APPEND_NAME(value,
1184 1184 &destination,
1185 1185 (int) value->hash.length);
1186 1186 if (macro_apx->body.macro_appendix.value_to_append != NULL) {
1187 1187 MBTOWC(wcs_buffer, " ");
1188 1188 append_char(wcs_buffer[0], &destination);
1189 1189 }
1190 1190 }
1191 1191 if (macro_apx->body.macro_appendix.value_to_append != NULL) {
1192 1192 APPEND_NAME(macro_apx->body.macro_appendix.value_to_append,
1193 1193 &destination,
1194 1194 (int) macro_apx->body.macro_appendix.value_to_append->hash.length);
1195 1195 }
1196 1196 value = GETNAME(destination.buffer.start, FIND_LENGTH);
1197 1197 if (destination.free_after_use) {
1198 1198 retmem(destination.buffer.start);
1199 1199 }
1200 1200 }
1201 1201 /**/
1202 1202 macro->body.macro.value = value;
1203 1203 macro->body.macro.daemon = daemon;
1204 1204 /*
1205 1205 * If the user changes the VIRTUAL_ROOT, we need to flush
1206 1206 * the vroot package cache.
1207 1207 */
1208 1208 if (name == path_name) {
1209 1209 flush_path_cache();
1210 1210 }
1211 1211 if (name == virtual_root) {
1212 1212 flush_vroot_cache();
1213 1213 }
1214 1214 /* If this sets the VPATH we remember that */
1215 1215 if ((name == vpath_name) &&
1216 1216 (value != NULL) &&
↓ open down ↓ |
1216 lines elided |
↑ open up ↑ |
1217 1217 (value->hash.length > 0)) {
1218 1218 vpath_defined = true;
1219 1219 }
1220 1220 /*
1221 1221 * For environment variables we also set the
1222 1222 * environment value each time.
1223 1223 */
1224 1224 if (macro->body.macro.exported) {
1225 1225 static char *env;
1226 1226
1227 - if (!reading_environment && (value != NULL) && value->dollar) {
1227 + if (!reading_environment && (value != NULL)) {
1228 1228 Envvar p;
1229 1229
1230 1230 for (p = envvar; p != NULL; p = p->next) {
1231 1231 if (p->name == name) {
1232 1232 p->value = value;
1233 1233 p->already_put = false;
1234 1234 goto found_it;
1235 1235 }
1236 1236 }
1237 1237 p = ALLOC(Envvar);
1238 1238 p->name = name;
1239 1239 p->value = value;
1240 1240 p->next = envvar;
1241 1241 p->env_string = NULL;
1242 1242 p->already_put = false;
1243 1243 envvar = p;
1244 1244 found_it:;
1245 - } else {
1245 + } if (reading_environment || (value == NULL) || !value->dollar) {
1246 1246 length = 2 + strlen(name->string_mb);
1247 1247 if (value != NULL) {
1248 1248 length += strlen(value->string_mb);
1249 1249 }
1250 1250 Property env_prop = maybe_append_prop(name, env_mem_prop);
1251 1251 /*
1252 1252 * We use a permanent buffer to reset SUNPRO_DEPENDENCIES value.
1253 1253 */
1254 1254 if (!strncmp(name->string_mb, NOCATGETS("SUNPRO_DEPENDENCIES"), 19)) {
1255 1255 if (length >= sunpro_dependencies_buf_size) {
1256 1256 sunpro_dependencies_buf_size=length*2;
1257 1257 if (sunpro_dependencies_buf_size < 4096)
1258 1258 sunpro_dependencies_buf_size = 4096; // Default minimum size
1259 1259 if (sunpro_dependencies_buf)
1260 1260 sunpro_dependencies_oldbuf = sunpro_dependencies_buf;
1261 1261 sunpro_dependencies_buf=getmem(sunpro_dependencies_buf_size);
1262 1262 }
1263 1263 env = sunpro_dependencies_buf;
1264 1264 } else {
1265 1265 env = getmem(length);
1266 1266 }
1267 1267 env_alloc_num++;
1268 1268 env_alloc_bytes += length;
1269 1269 (void) sprintf(env,
1270 1270 "%s=%s",
1271 1271 name->string_mb,
1272 1272 value == NULL ?
1273 1273 "" : value->string_mb);
1274 1274 (void) putenv(env);
1275 1275 env_prop->body.env_mem.value = env;
1276 1276 if (sunpro_dependencies_oldbuf) {
1277 1277 /* Return old buffer */
1278 1278 retmem_mb(sunpro_dependencies_oldbuf);
1279 1279 sunpro_dependencies_oldbuf = NULL;
1280 1280 }
1281 1281 }
1282 1282 }
1283 1283 if (name == target_arch) {
1284 1284 Name ha = getvar(host_arch);
1285 1285 Name ta = getvar(target_arch);
1286 1286 Name vr = getvar(virtual_root);
1287 1287 int length;
1288 1288 wchar_t *new_value;
1289 1289 wchar_t *old_vr;
1290 1290 Boolean new_value_allocated = false;
1291 1291
1292 1292 Wstring ha_str(ha);
1293 1293 Wstring ta_str(ta);
1294 1294 Wstring vr_str(vr);
1295 1295
1296 1296 wchar_t * wcb_ha = ha_str.get_string();
1297 1297 wchar_t * wcb_ta = ta_str.get_string();
1298 1298 wchar_t * wcb_vr = vr_str.get_string();
1299 1299
1300 1300 length = 32 +
1301 1301 wslen(wcb_ha) +
1302 1302 wslen(wcb_ta) +
1303 1303 wslen(wcb_vr);
1304 1304 old_vr = wcb_vr;
1305 1305 MBSTOWCS(wcs_buffer, NOCATGETS("/usr/arch/"));
1306 1306 if (IS_WEQUALN(old_vr,
1307 1307 wcs_buffer,
1308 1308 wslen(wcs_buffer))) {
1309 1309 old_vr = (wchar_t *) wschr(old_vr, (int) colon_char) + 1;
1310 1310 }
1311 1311 if ( (ha == ta) || (wslen(wcb_ta) == 0) ) {
1312 1312 new_value = old_vr;
1313 1313 } else {
1314 1314 new_value = ALLOC_WC(length);
1315 1315 new_value_allocated = true;
1316 1316 WCSTOMBS(mbs_buffer, old_vr);
1317 1317 (void) wsprintf(new_value,
1318 1318 NOCATGETS("/usr/arch/%s/%s:%s"),
1319 1319 ha->string_mb + 1,
1320 1320 ta->string_mb + 1,
1321 1321 mbs_buffer);
1322 1322 }
1323 1323 if (new_value[0] != 0) {
1324 1324 (void) setvar_daemon(virtual_root,
1325 1325 GETNAME(new_value, FIND_LENGTH),
1326 1326 false,
1327 1327 no_daemon,
1328 1328 true,
1329 1329 debug_level);
1330 1330 }
1331 1331 if (new_value_allocated) {
1332 1332 retmem(new_value);
1333 1333 }
1334 1334 }
1335 1335 return macro;
1336 1336 }
↓ open down ↓ |
81 lines elided |
↑ open up ↑ |
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX