1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 /* 22 * Copyright 2005 Sun Microsystems, Inc. All rights reserved. 23 * Use is subject to license terms. 24 */ 25 26 #ifdef TEAMWARE_MAKE_CMN 27 28 /* 29 * parallel.cc 30 * 31 * Deal with the parallel processing 32 */ 33 34 /* 35 * Included files 36 */ 37 #ifdef DISTRIBUTED 38 #include <avo/strings.h> /* AVO_STRDUP() */ 39 #include <dm/Avo_DoJobMsg.h> 40 #include <dm/Avo_MToolJobResultMsg.h> 41 #endif 42 #include <errno.h> /* errno */ 43 #include <fcntl.h> 44 #include <avo/util.h> /* avo_get_user(), avo_hostname() */ 45 #include <mk/defs.h> 46 #include <mksh/dosys.h> /* redirect_io() */ 47 #include <mksh/macro.h> /* expand_value() */ 48 #include <mksh/misc.h> /* getmem() */ 49 #include <sys/signal.h> 50 #include <sys/stat.h> 51 #include <sys/types.h> 52 #include <sys/utsname.h> 53 #include <sys/wait.h> 54 #include <unistd.h> 55 56 #ifdef SGE_SUPPORT 57 #include <dmthread/Avo_PathNames.h> 58 #endif 59 60 61 /* 62 * Defined macros 63 */ 64 #define MAXRULES 100 65 66 /* 67 * This const should be in avo_dms/include/AvoDmakeCommand.h 68 */ 69 const int local_host_mask = 0x20; 70 71 72 /* 73 * typedefs & structs 74 */ 75 76 77 /* 78 * Static variables 79 */ 80 #ifdef TEAMWARE_MAKE_CMN 81 static Boolean just_did_subtree = false; 82 static char local_host[MAXNAMELEN] = ""; 83 static char user_name[MAXNAMELEN] = ""; 84 #endif 85 static int pmake_max_jobs = 0; 86 static pid_t process_running = -1; 87 static Running *running_tail = &running_list; 88 static Name subtree_conflict; 89 static Name subtree_conflict2; 90 91 92 /* 93 * File table of contents 94 */ 95 #ifdef DISTRIBUTED 96 static void append_dmake_cmd(Avo_DoJobMsg *dmake_job_msg, char *orig_cmd_line, int cmd_options); 97 static void append_job_result_msg(Avo_MToolJobResultMsg *msg, char *outFn, char *errFn); 98 static void send_job_result_msg(Running rp); 99 #endif 100 static void delete_running_struct(Running rp); 101 static Boolean dependency_conflict(Name target); 102 static Doname distribute_process(char **commands, Property line); 103 static void doname_subtree(Name target, Boolean do_get, Boolean implicit); 104 static void dump_out_file(char *filename, Boolean err); 105 static void finish_doname(Running rp); 106 static void maybe_reread_make_state(void); 107 static void process_next(void); 108 static void reset_conditionals(int cnt, Name *targets, Property *locals); 109 static pid_t run_rule_commands(char *host, char **commands); 110 static Property *set_conditionals(int cnt, Name *targets); 111 static void store_conditionals(Running rp); 112 113 114 /* 115 * execute_parallel(line, waitflg) 116 * 117 * DMake 2.x: 118 * parallel mode: spawns a parallel process to execute the command group. 119 * distributed mode: sends the command group down the pipe to rxm. 120 * 121 * Return value: 122 * The result of the execution 123 * 124 * Parameters: 125 * line The command group to execute 126 */ 127 Doname 128 execute_parallel(Property line, Boolean waitflg, Boolean local) 129 { 130 int argcnt; 131 int cmd_options = 0; 132 char *commands[MAXRULES + 5]; 133 char *cp; 134 #ifdef DISTRIBUTED 135 Avo_DoJobMsg *dmake_job_msg = NULL; 136 #endif 137 Name dmake_name; 138 Name dmake_value; 139 int ignore; 140 Name make_machines_name; 141 char **p; 142 Property prop; 143 Doname result = build_ok; 144 Cmd_line rule; 145 Boolean silent_flag; 146 Name target = line->body.line.target; 147 Boolean wrote_state_file = false; 148 149 if ((pmake_max_jobs == 0) && 150 (dmake_mode_type == parallel_mode)) { 151 if (user_name[0] == '\0') { 152 avo_get_user(user_name, NULL); 153 } 154 if (local_host[0] == '\0') { 155 strcpy(local_host, avo_hostname()); 156 } 157 MBSTOWCS(wcs_buffer, NOCATGETS("DMAKE_MAX_JOBS")); 158 dmake_name = GETNAME(wcs_buffer, FIND_LENGTH); 159 if (((prop = get_prop(dmake_name->prop, macro_prop)) != NULL) && 160 ((dmake_value = prop->body.macro.value) != NULL)) { 161 pmake_max_jobs = atoi(dmake_value->string_mb); 162 if (pmake_max_jobs <= 0) { 163 warning(catgets(catd, 1, 308, "DMAKE_MAX_JOBS cannot be less than or equal to zero.")); 164 warning(catgets(catd, 1, 309, "setting DMAKE_MAX_JOBS to %d."), PMAKE_DEF_MAX_JOBS); 165 pmake_max_jobs = PMAKE_DEF_MAX_JOBS; 166 } 167 } else { 168 /* 169 * For backwards compatibility w/ PMake 1.x, when 170 * DMake 2.x is being run in parallel mode, DMake 171 * should parse the PMake startup file 172 * $(HOME)/.make.machines to get the pmake_max_jobs. 173 */ 174 MBSTOWCS(wcs_buffer, NOCATGETS("PMAKE_MACHINESFILE")); 175 dmake_name = GETNAME(wcs_buffer, FIND_LENGTH); 176 if (((prop = get_prop(dmake_name->prop, macro_prop)) != NULL) && 177 ((dmake_value = prop->body.macro.value) != NULL)) { 178 make_machines_name = dmake_value; 179 } else { 180 make_machines_name = NULL; 181 } 182 if ((pmake_max_jobs = read_make_machines(make_machines_name)) <= 0) { 183 pmake_max_jobs = PMAKE_DEF_MAX_JOBS; 184 } 185 } 186 #ifdef DISTRIBUTED 187 if (send_mtool_msgs) { 188 send_rsrc_info_msg(pmake_max_jobs, local_host, user_name); 189 } 190 #endif 191 } 192 193 if ((dmake_mode_type == serial_mode) || 194 ((dmake_mode_type == parallel_mode) && (waitflg))) { 195 return (execute_serial(line)); 196 } 197 198 #ifdef DISTRIBUTED 199 if (dmake_mode_type == distributed_mode) { 200 if(local) { 201 // return (execute_serial(line)); 202 waitflg = true; 203 } 204 dmake_job_msg = new Avo_DoJobMsg(); 205 dmake_job_msg->setJobId(++job_msg_id); 206 dmake_job_msg->setTarget(target->string_mb); 207 dmake_job_msg->setImmediateOutput(0); 208 called_make = false; 209 } else 210 #endif 211 { 212 p = commands; 213 } 214 215 argcnt = 0; 216 for (rule = line->body.line.command_used; 217 rule != NULL; 218 rule = rule->next) { 219 if (posix && (touch || quest) && !rule->always_exec) { 220 continue; 221 } 222 if (vpath_defined) { 223 rule->command_line = 224 vpath_translation(rule->command_line); 225 } 226 if (dmake_mode_type == distributed_mode) { 227 cmd_options = 0; 228 if(local) { 229 cmd_options |= local_host_mask; 230 } 231 } else { 232 silent_flag = false; 233 ignore = 0; 234 } 235 if (rule->command_line->hash.length > 0) { 236 if (++argcnt == MAXRULES) { 237 if (dmake_mode_type == distributed_mode) { 238 /* XXX - tell rxm to execute on local host. */ 239 /* I WAS HERE!!! */ 240 } else { 241 /* Too many rules, run serially instead. */ 242 return build_serial; 243 } 244 } 245 #ifdef DISTRIBUTED 246 if (dmake_mode_type == distributed_mode) { 247 /* 248 * XXX - set assign_mask to tell rxm 249 * to do the following. 250 */ 251 /* From execute_serial(): 252 if (rule->assign) { 253 result = build_ok; 254 do_assign(rule->command_line, target); 255 */ 256 if (0) { 257 } else if (report_dependencies_level == 0) { 258 if (rule->ignore_error) { 259 cmd_options |= ignore_mask; 260 } 261 if (rule->silent) { 262 cmd_options |= silent_mask; 263 } 264 if (rule->command_line->meta) { 265 cmd_options |= meta_mask; 266 } 267 if (rule->make_refd) { 268 cmd_options |= make_refd_mask; 269 } 270 if (do_not_exec_rule) { 271 cmd_options |= do_not_exec_mask; 272 } 273 append_dmake_cmd(dmake_job_msg, 274 rule->command_line->string_mb, 275 cmd_options); 276 /* Copying dosys()... */ 277 if (rule->make_refd) { 278 if (waitflg) { 279 dmake_job_msg->setImmediateOutput(1); 280 } 281 called_make = true; 282 if (command_changed && 283 !wrote_state_file) { 284 write_state_file(0, false); 285 wrote_state_file = true; 286 } 287 } 288 } 289 } else 290 #endif 291 { 292 if (rule->silent && !silent) { 293 silent_flag = true; 294 } 295 if (rule->ignore_error) { 296 ignore++; 297 } 298 /* XXX - need to add support for + prefix */ 299 if (silent_flag || ignore) { 300 *p = getmem((silent_flag ? 1 : 0) + 301 ignore + 302 (strlen(rule-> 303 command_line-> 304 string_mb)) + 305 1); 306 cp = *p++; 307 if (silent_flag) { 308 *cp++ = (int) at_char; 309 } 310 if (ignore) { 311 *cp++ = (int) hyphen_char; 312 } 313 (void) strcpy(cp, rule->command_line->string_mb); 314 } else { 315 *p++ = rule->command_line->string_mb; 316 } 317 } 318 } 319 } 320 if ((argcnt == 0) || 321 (report_dependencies_level > 0)) { 322 #ifdef DISTRIBUTED 323 if (dmake_job_msg) { 324 delete dmake_job_msg; 325 } 326 #endif 327 return build_ok; 328 } 329 #ifdef DISTRIBUTED 330 if (dmake_mode_type == distributed_mode) { 331 // Send a DoJob message to the rxm process. 332 distribute_rxm(dmake_job_msg); 333 334 // Wait for an acknowledgement. 335 Avo_AcknowledgeMsg *ackMsg = getAcknowledgeMsg(); 336 if (ackMsg) { 337 delete ackMsg; 338 } 339 340 if (waitflg) { 341 // Wait for, and process a job result. 342 result = await_dist(waitflg); 343 if (called_make) { 344 maybe_reread_make_state(); 345 } 346 check_state(temp_file_name); 347 if (result == build_failed) { 348 if (!continue_after_error) { 349 350 #ifdef PRINT_EXIT_STATUS 351 warning(NOCATGETS("I'm in execute_parallel. await_dist() returned build_failed")); 352 #endif 353 354 fatal(catgets(catd, 1, 252, "Command failed for target `%s'"), 355 target->string_mb); 356 } 357 /* 358 * Make sure a failing command is not 359 * saved in .make.state. 360 */ 361 line->body.line.command_used = NULL; 362 } 363 if (temp_file_name != NULL) { 364 free_name(temp_file_name); 365 } 366 temp_file_name = NULL; 367 Property spro = get_prop(sunpro_dependencies->prop, macro_prop); 368 if(spro != NULL) { 369 Name val = spro->body.macro.value; 370 if(val != NULL) { 371 free_name(val); 372 spro->body.macro.value = NULL; 373 } 374 } 375 spro = get_prop(sunpro_dependencies->prop, env_mem_prop); 376 if(spro) { 377 char *val = spro->body.env_mem.value; 378 if(val != NULL) { 379 retmem_mb(val); 380 spro->body.env_mem.value = NULL; 381 } 382 } 383 return result; 384 } else { 385 parallel_process_cnt++; 386 return build_running; 387 } 388 } else 389 #endif 390 { 391 *p = NULL; 392 393 Doname res = distribute_process(commands, line); 394 if (res == build_running) { 395 parallel_process_cnt++; 396 } 397 398 /* 399 * Return only those memory that were specially allocated 400 * for part of commands. 401 */ 402 for (int i = 0; commands[i] != NULL; i++) { 403 if ((commands[i][0] == (int) at_char) || 404 (commands[i][0] == (int) hyphen_char)) { 405 retmem_mb(commands[i]); 406 } 407 } 408 return res; 409 } 410 } 411 412 #ifdef DISTRIBUTED 413 /* 414 * append_dmake_cmd() 415 * 416 * Replaces all escaped newline's (\<cr>) 417 * in the original command line with space's, 418 * then append the new command line to the DoJobMsg object. 419 */ 420 static void 421 append_dmake_cmd(Avo_DoJobMsg *dmake_job_msg, 422 char *orig_cmd_line, 423 int cmd_options) 424 { 425 /* 426 Avo_DmakeCommand *tmp_dmake_command; 427 428 tmp_dmake_command = new Avo_DmakeCommand(orig_cmd_line, cmd_options); 429 dmake_job_msg->appendCmd(tmp_dmake_command); 430 delete tmp_dmake_command; 431 */ 432 dmake_job_msg->appendCmd(new Avo_DmakeCommand(orig_cmd_line, cmd_options)); 433 } 434 #endif 435 436 #ifdef TEAMWARE_MAKE_CMN 437 #define MAXJOBS_ADJUST_RFE4694000 438 439 #ifdef MAXJOBS_ADJUST_RFE4694000 440 441 #include <unistd.h> /* sysconf(_SC_NPROCESSORS_ONLN) */ 442 #include <sys/ipc.h> /* ftok() */ 443 #include <sys/shm.h> /* shmget(), shmat(), shmdt(), shmctl() */ 444 #include <semaphore.h> /* sem_init(), sem_trywait(), sem_post(), sem_destroy() */ 445 #if defined(linux) 446 #define LOADAVG_1MIN 0 447 #else 448 #include <sys/loadavg.h> /* getloadavg() */ 449 #endif /* linux */ 450 451 /* 452 * adjust_pmake_max_jobs (int pmake_max_jobs) 453 * 454 * Parameters: 455 * pmake_max_jobs - max jobs limit set by user 456 * 457 * External functions used: 458 * sysconf() 459 * getloadavg() 460 */ 461 static int 462 adjust_pmake_max_jobs (int pmake_max_jobs) 463 { 464 static int ncpu = 0; 465 double loadavg[3]; 466 int adjustment; 467 int adjusted_max_jobs; 468 469 if (ncpu <= 0) { 470 if ((ncpu = sysconf(_SC_NPROCESSORS_ONLN)) <= 0) { 471 ncpu = 1; 472 } 473 } 474 if (getloadavg(loadavg, 3) != 3) return(pmake_max_jobs); 475 adjustment = ((int)loadavg[LOADAVG_1MIN]); 476 if (adjustment < 2) return(pmake_max_jobs); 477 if (ncpu > 1) { 478 adjustment = adjustment / ncpu; 479 } 480 adjusted_max_jobs = pmake_max_jobs - adjustment; 481 if (adjusted_max_jobs < 1) adjusted_max_jobs = 1; 482 return(adjusted_max_jobs); 483 } 484 485 /* 486 * M2 adjust mode data and functions 487 * 488 * m2_init() - initializes M2 shared semaphore 489 * m2_acquire_job() - decrements M2 semaphore counter 490 * m2_release_job() - increments M2 semaphore counter 491 * m2_fini() - destroys M2 semaphore and shared memory* 492 * 493 * Environment variables: 494 * __DMAKE_M2_FILE__ 495 * 496 * External functions: 497 * ftok(), shmget(), shmat(), shmdt(), shmctl() 498 * sem_init(), sem_trywait(), sem_post(), sem_destroy() 499 * creat(), close(), unlink() 500 * getenv(), putenv() 501 * 502 * Static variables: 503 * m2_file - tmp file name to create ipc key for shared memory 504 * m2_shm_id - shared memory id 505 * m2_shm_sem - shared memory semaphore 506 */ 507 508 static char m2_file[MAXPATHLEN]; 509 static int m2_shm_id = -1; 510 static sem_t* m2_shm_sem = 0; 511 512 static int 513 m2_init() { 514 char *var; 515 key_t key; 516 517 if ((var = getenv(NOCATGETS("__DMAKE_M2_FILE__"))) == 0) { 518 /* compose tmp file name */ 519 sprintf(m2_file, NOCATGETS("%s/dmake.m2.%d.XXXXXX"), tmpdir, getpid()); 520 521 /* create tmp file */ 522 int fd = mkstemp(m2_file); 523 if (fd < 0) { 524 return -1; 525 } else { 526 close(fd); 527 } 528 } else { 529 /* using existing semaphore */ 530 strcpy(m2_file, var); 531 } 532 533 /* combine IPC key */ 534 if ((key = ftok(m2_file, 38)) == (key_t) -1) { 535 return -1; 536 } 537 538 /* create shared memory */ 539 if ((m2_shm_id = shmget(key, sizeof(*m2_shm_sem), 0666 | (var ? 0 : IPC_CREAT|IPC_EXCL))) == -1) { 540 return -1; 541 } 542 543 /* attach shared memory */ 544 if ((m2_shm_sem = (sem_t*) shmat(m2_shm_id, 0, 0666)) == (sem_t*)-1) { 545 return -1; 546 } 547 548 /* root process */ 549 if (var == 0) { 550 /* initialize semaphore */ 551 if (sem_init(m2_shm_sem, 1, pmake_max_jobs)) { 552 return -1; 553 } 554 555 /* alloc memory for env variable */ 556 if ((var = (char*) malloc(MAXPATHLEN)) == 0) { 557 return -1; 558 } 559 560 /* put key to env */ 561 sprintf(var, NOCATGETS("__DMAKE_M2_FILE__=%s"), m2_file); 562 if (putenv(var)) { 563 return -1; 564 } 565 } 566 return 0; 567 } 568 569 static void 570 m2_fini() { 571 if (m2_shm_id >= 0) { 572 struct shmid_ds stat; 573 574 /* determine the number of attached processes */ 575 if (shmctl(m2_shm_id, IPC_STAT, &stat) == 0) { 576 if (stat.shm_nattch <= 1) { 577 /* destroy semaphore */ 578 if (m2_shm_sem != 0) { 579 (void) sem_destroy(m2_shm_sem); 580 } 581 582 /* destroy shared memory */ 583 (void) shmctl(m2_shm_id, IPC_RMID, &stat); 584 585 /* remove tmp file created for the key */ 586 (void) unlink(m2_file); 587 } else { 588 /* detach shared memory */ 589 if (m2_shm_sem != 0) { 590 (void) shmdt((char*) m2_shm_sem); 591 } 592 } 593 } 594 595 m2_shm_id = -1; 596 m2_shm_sem = 0; 597 } 598 } 599 600 static int 601 m2_acquire_job() { 602 if ((m2_shm_id >= 0) && (m2_shm_sem != 0)) { 603 if (sem_trywait(m2_shm_sem) == 0) { 604 return 1; 605 } 606 if (errno == EAGAIN) { 607 return 0; 608 } 609 } 610 return -1; 611 } 612 613 static int 614 m2_release_job() { 615 if ((m2_shm_id >= 0) && (m2_shm_sem != 0)) { 616 if (sem_post(m2_shm_sem) == 0) { 617 return 0; 618 } 619 } 620 return -1; 621 } 622 623 /* 624 * job adjust mode 625 * 626 * Possible values: 627 * ADJUST_M1 - adjustment by system load (default) 628 * ADJUST_M2 - fixed limit of jobs for the group of nested dmakes 629 * ADJUST_NONE - no adjustment - fixed limit of jobs for the current dmake 630 */ 631 static enum { 632 ADJUST_UNKNOWN, 633 ADJUST_M1, 634 ADJUST_M2, 635 ADJUST_NONE 636 } job_adjust_mode = ADJUST_UNKNOWN; 637 638 /* 639 * void job_adjust_fini() 640 * 641 * Description: 642 * Cleans up job adjust data. 643 * 644 * Static variables: 645 * job_adjust_mode Current job adjust mode 646 */ 647 void 648 job_adjust_fini() { 649 if (job_adjust_mode == ADJUST_M2) { 650 m2_fini(); 651 } 652 } 653 654 /* 655 * void job_adjust_error() 656 * 657 * Description: 658 * Prints warning message, cleans up job adjust data, and disables job adjustment 659 * 660 * Environment: 661 * DMAKE_ADJUST_MAX_JOBS 662 * 663 * External functions: 664 * putenv() 665 * 666 * Static variables: 667 * job_adjust_mode Current job adjust mode 668 */ 669 static void 670 job_adjust_error() { 671 if (job_adjust_mode != ADJUST_NONE) { 672 /* cleanup internals */ 673 job_adjust_fini(); 674 675 /* warning message for the user */ 676 warning(catgets(catd, 1, 339, "Encountered max jobs auto adjustment error - disabling auto adjustment.")); 677 678 /* switch off job adjustment for the children */ 679 putenv(NOCATGETS("DMAKE_ADJUST_MAX_JOBS=NO")); 680 681 /* and for this dmake */ 682 job_adjust_mode = ADJUST_NONE; 683 } 684 } 685 686 /* 687 * void job_adjust_init() 688 * 689 * Description: 690 * Parses DMAKE_ADJUST_MAX_JOBS env variable 691 * and performs appropriate initializations. 692 * 693 * Environment: 694 * DMAKE_ADJUST_MAX_JOBS 695 * DMAKE_ADJUST_MAX_JOBS == "NO" - no adjustment 696 * DMAKE_ADJUST_MAX_JOBS == "M2" - M2 adjust mode 697 * other - M1 adjust mode 698 * 699 * External functions: 700 * getenv() 701 * 702 * Static variables: 703 * job_adjust_mode Current job adjust mode 704 */ 705 static void 706 job_adjust_init() { 707 if (job_adjust_mode == ADJUST_UNKNOWN) { 708 /* default mode */ 709 job_adjust_mode = ADJUST_M1; 710 711 /* determine adjust mode */ 712 if (char *var = getenv(NOCATGETS("DMAKE_ADJUST_MAX_JOBS"))) { 713 if (strcasecmp(var, NOCATGETS("NO")) == 0) { 714 job_adjust_mode = ADJUST_NONE; 715 } else if (strcasecmp(var, NOCATGETS("M2")) == 0) { 716 job_adjust_mode = ADJUST_M2; 717 } 718 } 719 720 /* M2 specific initialization */ 721 if (job_adjust_mode == ADJUST_M2) { 722 if (m2_init()) { 723 job_adjust_error(); 724 } 725 } 726 } 727 } 728 729 #endif /* MAXJOBS_ADJUST_RFE4694000 */ 730 #endif /* TEAMWARE_MAKE_CMN */ 731 732 /* 733 * distribute_process(char **commands, Property line) 734 * 735 * Parameters: 736 * commands argv vector of commands to execute 737 * 738 * Return value: 739 * The result of the execution 740 * 741 * Static variables used: 742 * process_running Set to the pid of the process set running 743 * #if defined (TEAMWARE_MAKE_CMN) && defined (MAXJOBS_ADJUST_RFE4694000) 744 * job_adjust_mode Current job adjust mode 745 * #endif 746 */ 747 static Doname 748 distribute_process(char **commands, Property line) 749 { 750 static unsigned file_number = 0; 751 wchar_t string[MAXPATHLEN]; 752 char mbstring[MAXPATHLEN]; 753 int filed; 754 int res; 755 int tmp_index; 756 char *tmp_index_str_ptr; 757 758 #if !defined (TEAMWARE_MAKE_CMN) || !defined (MAXJOBS_ADJUST_RFE4694000) 759 while (parallel_process_cnt >= pmake_max_jobs) { 760 await_parallel(false); 761 finish_children(true); 762 } 763 #else /* TEAMWARE_MAKE_CMN && MAXJOBS_ADJUST_RFE4694000 */ 764 /* initialize adjust mode, if not initialized */ 765 if (job_adjust_mode == ADJUST_UNKNOWN) { 766 job_adjust_init(); 767 } 768 769 /* actions depend on adjust mode */ 770 switch (job_adjust_mode) { 771 case ADJUST_M1: 772 while (parallel_process_cnt >= adjust_pmake_max_jobs (pmake_max_jobs)) { 773 await_parallel(false); 774 finish_children(true); 775 } 776 break; 777 case ADJUST_M2: 778 if ((res = m2_acquire_job()) == 0) { 779 if (parallel_process_cnt > 0) { 780 await_parallel(false); 781 finish_children(true); 782 783 if ((res = m2_acquire_job()) == 0) { 784 return build_serial; 785 } 786 } else { 787 return build_serial; 788 } 789 } 790 if (res < 0) { 791 /* job adjustment error */ 792 job_adjust_error(); 793 794 /* no adjustment */ 795 while (parallel_process_cnt >= pmake_max_jobs) { 796 await_parallel(false); 797 finish_children(true); 798 } 799 } 800 break; 801 default: 802 while (parallel_process_cnt >= pmake_max_jobs) { 803 await_parallel(false); 804 finish_children(true); 805 } 806 } 807 #endif /* TEAMWARE_MAKE_CMN && MAXJOBS_ADJUST_RFE4694000 */ 808 #ifdef DISTRIBUTED 809 if (send_mtool_msgs) { 810 send_job_start_msg(line); 811 } 812 #endif 813 #ifdef DISTRIBUTED 814 setvar_envvar((Avo_DoJobMsg *)NULL); 815 #else 816 setvar_envvar(); 817 #endif 818 /* 819 * Tell the user what DMake is doing. 820 */ 821 if (!silent && output_mode != txt2_mode) { 822 /* 823 * Print local_host --> x job(s). 824 */ 825 (void) fprintf(stdout, 826 catgets(catd, 1, 325, "%s --> %d %s\n"), 827 local_host, 828 parallel_process_cnt + 1, 829 (parallel_process_cnt == 0) ? catgets(catd, 1, 124, "job") : catgets(catd, 1, 125, "jobs")); 830 831 /* Print command line(s). */ 832 tmp_index = 0; 833 while (commands[tmp_index] != NULL) { 834 /* No @ char. */ 835 /* XXX - need to add [2] when + prefix is added */ 836 if ((commands[tmp_index][0] != (int) at_char) && 837 (commands[tmp_index][1] != (int) at_char)) { 838 tmp_index_str_ptr = commands[tmp_index]; 839 if (*tmp_index_str_ptr == (int) hyphen_char) { 840 tmp_index_str_ptr++; 841 } 842 (void) fprintf(stdout, "%s\n", tmp_index_str_ptr); 843 } 844 tmp_index++; 845 } 846 (void) fflush(stdout); 847 } 848 849 (void) sprintf(mbstring, 850 NOCATGETS("%s/dmake.stdout.%d.%d.XXXXXX"), 851 tmpdir, 852 getpid(), 853 file_number++); 854 855 mktemp(mbstring); 856 857 stdout_file = strdup(mbstring); 858 stderr_file = NULL; 859 #if defined (TEAMWARE_MAKE_CMN) && defined(REDIRECT_ERR) 860 if (!out_err_same) { 861 (void) sprintf(mbstring, 862 NOCATGETS("%s/dmake.stderr.%d.%d.XXXXXX"), 863 tmpdir, 864 getpid(), 865 file_number++); 866 867 mktemp(mbstring); 868 869 stderr_file = strdup(mbstring); 870 } 871 #endif 872 873 #ifdef SGE_SUPPORT 874 if (grid) { 875 static char *dir4gridscripts = NULL; 876 static char *hostName = NULL; 877 if (dir4gridscripts == NULL) { 878 Name dmakeOdir_name, dmakeOdir_value; 879 Property prop; 880 MBSTOWCS(wcs_buffer, NOCATGETS("DMAKE_ODIR")); 881 dmakeOdir_name = GETNAME(wcs_buffer, FIND_LENGTH); 882 if (((prop = get_prop(dmakeOdir_name->prop, macro_prop)) != NULL) && 883 ((dmakeOdir_value = prop->body.macro.value) != NULL)) { 884 dir4gridscripts = dmakeOdir_value->string_mb; 885 } 886 dir4gridscripts = Avo_PathNames::pathname_output_directory(dir4gridscripts); 887 hostName = Avo_PathNames::pathname_local_host(); 888 } 889 (void) sprintf(script_file, 890 NOCATGETS("%s/dmake.script.%s.%d.%d.XXXXXX"), 891 dir4gridscripts, 892 hostName, 893 getpid(), 894 file_number++); 895 } 896 #endif /* SGE_SUPPORT */ 897 process_running = run_rule_commands(local_host, commands); 898 899 return build_running; 900 } 901 902 /* 903 * doname_parallel(target, do_get, implicit) 904 * 905 * Processes the given target and finishes up any parallel 906 * processes left running. 907 * 908 * Return value: 909 * Result of target build 910 * 911 * Parameters: 912 * target Target to build 913 * do_get True if sccs get to be done 914 * implicit True if this is an implicit target 915 */ 916 Doname 917 doname_parallel(Name target, Boolean do_get, Boolean implicit) 918 { 919 Doname result; 920 921 result = doname_check(target, do_get, implicit, false); 922 if (result == build_ok || result == build_failed) { 923 return result; 924 } 925 finish_running(); 926 return (Doname) target->state; 927 } 928 929 /* 930 * doname_subtree(target, do_get, implicit) 931 * 932 * Completely computes an object and its dependents for a 933 * serial subtree build. 934 * 935 * Parameters: 936 * target Target to build 937 * do_get True if sccs get to be done 938 * implicit True if this is an implicit target 939 * 940 * Static variables used: 941 * running_tail Tail of the list of running processes 942 * 943 * Global variables used: 944 * running_list The list of running processes 945 */ 946 static void 947 doname_subtree(Name target, Boolean do_get, Boolean implicit) 948 { 949 Running save_running_list; 950 Running *save_running_tail; 951 952 save_running_list = running_list; 953 save_running_tail = running_tail; 954 running_list = NULL; 955 running_tail = &running_list; 956 target->state = build_subtree; 957 target->checking_subtree = true; 958 while(doname_check(target, do_get, implicit, false) == build_running) { 959 target->checking_subtree = false; 960 finish_running(); 961 target->state = build_subtree; 962 } 963 target->checking_subtree = false; 964 running_list = save_running_list; 965 running_tail = save_running_tail; 966 } 967 968 /* 969 * finish_running() 970 * 971 * Keeps processing until the running_list is emptied out. 972 * 973 * Parameters: 974 * 975 * Global variables used: 976 * running_list The list of running processes 977 */ 978 void 979 finish_running(void) 980 { 981 while (running_list != NULL) { 982 #ifdef DISTRIBUTED 983 if (dmake_mode_type == distributed_mode) { 984 if ((just_did_subtree) || 985 (parallel_process_cnt == 0)) { 986 just_did_subtree = false; 987 } else { 988 (void) await_dist(false); 989 finish_children(true); 990 } 991 } else 992 #endif 993 { 994 await_parallel(false); 995 finish_children(true); 996 } 997 if (running_list != NULL) { 998 process_next(); 999 } 1000 } 1001 } 1002 1003 /* 1004 * process_next() 1005 * 1006 * Searches the running list for any targets which can start processing. 1007 * This can be a pending target, a serial target, or a subtree target. 1008 * 1009 * Parameters: 1010 * 1011 * Static variables used: 1012 * running_tail The end of the list of running procs 1013 * subtree_conflict A target which conflicts with a subtree 1014 * subtree_conflict2 The other target which conflicts 1015 * 1016 * Global variables used: 1017 * commands_done True if commands executed 1018 * debug_level Controls debug output 1019 * parallel_process_cnt Number of parallel process running 1020 * recursion_level Indentation for debug output 1021 * running_list List of running processes 1022 */ 1023 static void 1024 process_next(void) 1025 { 1026 Running rp; 1027 Running *rp_prev; 1028 Property line; 1029 Chain target_group; 1030 Dependency dep; 1031 Boolean quiescent = true; 1032 Running *subtree_target; 1033 Boolean saved_commands_done; 1034 Property *conditionals; 1035 1036 subtree_target = NULL; 1037 subtree_conflict = NULL; 1038 subtree_conflict2 = NULL; 1039 /* 1040 * If nothing currently running, build a serial target, if any. 1041 */ 1042 start_loop_1: 1043 for (rp_prev = &running_list, rp = running_list; 1044 rp != NULL && parallel_process_cnt == 0; 1045 rp = rp->next) { 1046 if (rp->state == build_serial) { 1047 *rp_prev = rp->next; 1048 if (rp->next == NULL) { 1049 running_tail = rp_prev; 1050 } 1051 recursion_level = rp->recursion_level; 1052 rp->target->state = build_pending; 1053 (void) doname_check(rp->target, 1054 rp->do_get, 1055 rp->implicit, 1056 false); 1057 quiescent = false; 1058 delete_running_struct(rp); 1059 goto start_loop_1; 1060 } else { 1061 rp_prev = &rp->next; 1062 } 1063 } 1064 /* 1065 * Find a target to build. The target must be pending, have all 1066 * its dependencies built, and not be in a target group with a target 1067 * currently building. 1068 */ 1069 start_loop_2: 1070 for (rp_prev = &running_list, rp = running_list; 1071 rp != NULL; 1072 rp = rp->next) { 1073 if (!(rp->state == build_pending || 1074 rp->state == build_subtree)) { 1075 quiescent = false; 1076 rp_prev = &rp->next; 1077 } else if (rp->state == build_pending) { 1078 line = get_prop(rp->target->prop, line_prop); 1079 for (dep = line->body.line.dependencies; 1080 dep != NULL; 1081 dep = dep->next) { 1082 if (dep->name->state == build_running || 1083 dep->name->state == build_pending || 1084 dep->name->state == build_serial) { 1085 break; 1086 } 1087 } 1088 if (dep == NULL) { 1089 for (target_group = line->body.line.target_group; 1090 target_group != NULL; 1091 target_group = target_group->next) { 1092 if (is_running(target_group->name)) { 1093 break; 1094 } 1095 } 1096 if (target_group == NULL) { 1097 *rp_prev = rp->next; 1098 if (rp->next == NULL) { 1099 running_tail = rp_prev; 1100 } 1101 recursion_level = rp->recursion_level; 1102 rp->target->state = rp->redo ? 1103 build_dont_know : build_pending; 1104 saved_commands_done = commands_done; 1105 conditionals = 1106 set_conditionals 1107 (rp->conditional_cnt, 1108 rp->conditional_targets); 1109 rp->target->dont_activate_cond_values = true; 1110 if ((doname_check(rp->target, 1111 rp->do_get, 1112 rp->implicit, 1113 rp->target->has_target_prop ? true : false) != 1114 build_running) && 1115 !commands_done) { 1116 commands_done = 1117 saved_commands_done; 1118 } 1119 rp->target->dont_activate_cond_values = false; 1120 reset_conditionals 1121 (rp->conditional_cnt, 1122 rp->conditional_targets, 1123 conditionals); 1124 quiescent = false; 1125 delete_running_struct(rp); 1126 goto start_loop_2; 1127 } else { 1128 rp_prev = &rp->next; 1129 } 1130 } else { 1131 rp_prev = &rp->next; 1132 } 1133 } else { 1134 rp_prev = &rp->next; 1135 } 1136 } 1137 /* 1138 * If nothing has been found to build and there exists a subtree 1139 * target with no dependency conflicts, build it. 1140 */ 1141 if (quiescent) { 1142 start_loop_3: 1143 for (rp_prev = &running_list, rp = running_list; 1144 rp != NULL; 1145 rp = rp->next) { 1146 if (rp->state == build_subtree) { 1147 if (!dependency_conflict(rp->target)) { 1148 *rp_prev = rp->next; 1149 if (rp->next == NULL) { 1150 running_tail = rp_prev; 1151 } 1152 recursion_level = rp->recursion_level; 1153 doname_subtree(rp->target, 1154 rp->do_get, 1155 rp->implicit); 1156 #ifdef DISTRIBUTED 1157 just_did_subtree = true; 1158 #endif 1159 quiescent = false; 1160 delete_running_struct(rp); 1161 goto start_loop_3; 1162 } else { 1163 subtree_target = rp_prev; 1164 rp_prev = &rp->next; 1165 } 1166 } else { 1167 rp_prev = &rp->next; 1168 } 1169 } 1170 } 1171 /* 1172 * If still nothing found to build, we either have a deadlock 1173 * or a subtree with a dependency conflict with something waiting 1174 * to build. 1175 */ 1176 if (quiescent) { 1177 if (subtree_target == NULL) { 1178 fatal(catgets(catd, 1, 126, "Internal error: deadlock detected in process_next")); 1179 } else { 1180 rp = *subtree_target; 1181 if (debug_level > 0) { 1182 warning(catgets(catd, 1, 127, "Conditional macro conflict encountered for %s between %s and %s"), 1183 subtree_conflict2->string_mb, 1184 rp->target->string_mb, 1185 subtree_conflict->string_mb); 1186 } 1187 *subtree_target = (*subtree_target)->next; 1188 if (rp->next == NULL) { 1189 running_tail = subtree_target; 1190 } 1191 recursion_level = rp->recursion_level; 1192 doname_subtree(rp->target, rp->do_get, rp->implicit); 1193 #ifdef DISTRIBUTED 1194 just_did_subtree = true; 1195 #endif 1196 delete_running_struct(rp); 1197 } 1198 } 1199 } 1200 1201 /* 1202 * set_conditionals(cnt, targets) 1203 * 1204 * Sets the conditional macros for the targets given in the array of 1205 * targets. The old macro values are returned in an array of 1206 * Properties for later resetting. 1207 * 1208 * Return value: 1209 * Array of conditional macro settings 1210 * 1211 * Parameters: 1212 * cnt Number of targets 1213 * targets Array of targets 1214 */ 1215 static Property * 1216 set_conditionals(int cnt, Name *targets) 1217 { 1218 Property *locals, *lp; 1219 Name *tp; 1220 1221 locals = (Property *) getmem(cnt * sizeof(Property)); 1222 for (lp = locals, tp = targets; 1223 cnt > 0; 1224 cnt--, lp++, tp++) { 1225 *lp = (Property) getmem((*tp)->conditional_cnt * 1226 sizeof(struct _Property)); 1227 set_locals(*tp, *lp); 1228 } 1229 return locals; 1230 } 1231 1232 /* 1233 * reset_conditionals(cnt, targets, locals) 1234 * 1235 * Resets the conditional macros as saved in the given array of 1236 * Properties. The resets are done in reverse order. Afterwards the 1237 * data structures are freed. 1238 * 1239 * Parameters: 1240 * cnt Number of targets 1241 * targets Array of targets 1242 * locals Array of dependency macro settings 1243 */ 1244 static void 1245 reset_conditionals(int cnt, Name *targets, Property *locals) 1246 { 1247 Name *tp; 1248 Property *lp; 1249 1250 for (tp = targets + (cnt - 1), lp = locals + (cnt - 1); 1251 cnt > 0; 1252 cnt--, tp--, lp--) { 1253 reset_locals(*tp, 1254 *lp, 1255 get_prop((*tp)->prop, conditional_prop), 1256 0); 1257 retmem_mb((caddr_t) *lp); 1258 } 1259 retmem_mb((caddr_t) locals); 1260 } 1261 1262 /* 1263 * dependency_conflict(target) 1264 * 1265 * Returns true if there is an intersection between 1266 * the subtree of the target and any dependents of the pending targets. 1267 * 1268 * Return value: 1269 * True if conflict found 1270 * 1271 * Parameters: 1272 * target Subtree target to check 1273 * 1274 * Static variables used: 1275 * subtree_conflict Target conflict found 1276 * subtree_conflict2 Second conflict found 1277 * 1278 * Global variables used: 1279 * running_list List of running processes 1280 * wait_name .WAIT, not a real dependency 1281 */ 1282 static Boolean 1283 dependency_conflict(Name target) 1284 { 1285 Property line; 1286 Property pending_line; 1287 Dependency dp; 1288 Dependency pending_dp; 1289 Running rp; 1290 1291 /* Return if we are already checking this target */ 1292 if (target->checking_subtree) { 1293 return false; 1294 } 1295 target->checking_subtree = true; 1296 line = get_prop(target->prop, line_prop); 1297 if (line == NULL) { 1298 target->checking_subtree = false; 1299 return false; 1300 } 1301 /* Check each dependency of the target for conflicts */ 1302 for (dp = line->body.line.dependencies; dp != NULL; dp = dp->next) { 1303 /* Ignore .WAIT dependency */ 1304 if (dp->name == wait_name) { 1305 continue; 1306 } 1307 /* 1308 * For each pending target, look for a dependency which 1309 * is the same as a dependency of the subtree target. Since 1310 * we can't build the subtree until all pending targets have 1311 * finished which depend on the same dependency, this is 1312 * a conflict. 1313 */ 1314 for (rp = running_list; rp != NULL; rp = rp->next) { 1315 if (rp->state == build_pending) { 1316 pending_line = get_prop(rp->target->prop, 1317 line_prop); 1318 if (pending_line == NULL) { 1319 continue; 1320 } 1321 for(pending_dp = pending_line-> 1322 body.line.dependencies; 1323 pending_dp != NULL; 1324 pending_dp = pending_dp->next) { 1325 if (dp->name == pending_dp->name) { 1326 target->checking_subtree 1327 = false; 1328 subtree_conflict = rp->target; 1329 subtree_conflict2 = dp->name; 1330 return true; 1331 } 1332 } 1333 } 1334 } 1335 if (dependency_conflict(dp->name)) { 1336 target->checking_subtree = false; 1337 return true; 1338 } 1339 } 1340 target->checking_subtree = false; 1341 return false; 1342 } 1343 1344 /* 1345 * await_parallel(waitflg) 1346 * 1347 * Waits for parallel children to exit and finishes their processing. 1348 * If waitflg is false, the function returns after update_delay. 1349 * 1350 * Parameters: 1351 * waitflg dwight 1352 */ 1353 void 1354 await_parallel(Boolean waitflg) 1355 { 1356 #ifdef _CHECK_UPDATE_H 1357 static int number_of_unknown_children = 0; 1358 #endif /* _CHECK_UPDATE_H */ 1359 Boolean nohang; 1360 pid_t pid; 1361 int status; 1362 Running rp; 1363 int waiterr; 1364 1365 nohang = false; 1366 for ( ; ; ) { 1367 if (!nohang) { 1368 (void) alarm((int) update_delay); 1369 } 1370 pid = waitpid((pid_t)-1, 1371 &status, 1372 nohang ? WNOHANG : 0); 1373 waiterr = errno; 1374 if (!nohang) { 1375 (void) alarm(0); 1376 } 1377 if (pid <= 0) { 1378 if (waiterr == EINTR) { 1379 if (waitflg) { 1380 continue; 1381 } else { 1382 return; 1383 } 1384 } else { 1385 return; 1386 } 1387 } 1388 for (rp = running_list; 1389 (rp != NULL) && (rp->pid != pid); 1390 rp = rp->next) { 1391 ; 1392 } 1393 if (rp == NULL) { 1394 #ifdef _CHECK_UPDATE_H 1395 /* Ignore first child - it is check_update */ 1396 if (number_of_unknown_children <= 0) { 1397 number_of_unknown_children = 1; 1398 return; 1399 } 1400 #endif /* _CHECK_UPDATE_H */ 1401 if (send_mtool_msgs) { 1402 continue; 1403 } else { 1404 fatal(catgets(catd, 1, 128, "Internal error: returned child pid not in running_list")); 1405 } 1406 } else { 1407 rp->state = (WIFEXITED(status) && WEXITSTATUS(status) == 0) ? build_ok : build_failed; 1408 } 1409 nohang = true; 1410 parallel_process_cnt--; 1411 1412 #if defined (TEAMWARE_MAKE_CMN) && defined (MAXJOBS_ADJUST_RFE4694000) 1413 if (job_adjust_mode == ADJUST_M2) { 1414 if (m2_release_job()) { 1415 job_adjust_error(); 1416 } 1417 } 1418 #endif 1419 } 1420 } 1421 1422 /* 1423 * finish_children(docheck) 1424 * 1425 * Finishes the processing for all targets which were running 1426 * and have now completed. 1427 * 1428 * Parameters: 1429 * docheck Completely check the finished target 1430 * 1431 * Static variables used: 1432 * running_tail The tail of the running list 1433 * 1434 * Global variables used: 1435 * continue_after_error -k flag 1436 * fatal_in_progress True if we are finishing up after fatal err 1437 * running_list List of running processes 1438 */ 1439 void 1440 finish_children(Boolean docheck) 1441 { 1442 int cmds_length; 1443 Property line; 1444 Property line2; 1445 struct stat out_buf; 1446 Running rp; 1447 Running *rp_prev; 1448 Cmd_line rule; 1449 Boolean silent_flag; 1450 1451 for (rp_prev = &running_list, rp = running_list; 1452 rp != NULL; 1453 rp = rp->next) { 1454 bypass_for_loop_inc_4: 1455 /* 1456 * If the state is ok or failed, then this target has 1457 * finished building. 1458 * In parallel_mode, output the accumulated stdout/stderr. 1459 * Read the auto dependency stuff, handle a failed build, 1460 * update the target, then finish the doname process for 1461 * that target. 1462 */ 1463 if (rp->state == build_ok || rp->state == build_failed) { 1464 *rp_prev = rp->next; 1465 if (rp->next == NULL) { 1466 running_tail = rp_prev; 1467 } 1468 if ((line2 = rp->command) == NULL) { 1469 line2 = get_prop(rp->target->prop, line_prop); 1470 } 1471 if (dmake_mode_type == distributed_mode) { 1472 if (rp->make_refd) { 1473 maybe_reread_make_state(); 1474 } 1475 } else { 1476 /* 1477 * Send an Avo_MToolJobResultMsg to maketool. 1478 */ 1479 #ifdef DISTRIBUTED 1480 if (send_mtool_msgs) { 1481 send_job_result_msg(rp); 1482 } 1483 #endif 1484 /* 1485 * Check if there were any job output 1486 * from the parallel build. 1487 */ 1488 if (rp->stdout_file != NULL) { 1489 if (stat(rp->stdout_file, &out_buf) < 0) { 1490 fatal(catgets(catd, 1, 130, "stat of %s failed: %s"), 1491 rp->stdout_file, 1492 errmsg(errno)); 1493 } 1494 if ((line2 != NULL) && 1495 (out_buf.st_size > 0)) { 1496 cmds_length = 0; 1497 for (rule = line2->body.line.command_used, 1498 silent_flag = silent; 1499 rule != NULL; 1500 rule = rule->next) { 1501 cmds_length += rule->command_line->hash.length + 1; 1502 silent_flag = BOOLEAN(silent_flag || rule->silent); 1503 } 1504 if (out_buf.st_size != cmds_length || silent_flag || 1505 output_mode == txt2_mode) { 1506 dump_out_file(rp->stdout_file, false); 1507 } 1508 } 1509 (void) unlink(rp->stdout_file); 1510 retmem_mb(rp->stdout_file); 1511 rp->stdout_file = NULL; 1512 } 1513 #if defined(REDIRECT_ERR) 1514 if (!out_err_same && (rp->stderr_file != NULL)) { 1515 if (stat(rp->stderr_file, &out_buf) < 0) { 1516 fatal(catgets(catd, 1, 130, "stat of %s failed: %s"), 1517 rp->stderr_file, 1518 errmsg(errno)); 1519 } 1520 if ((line2 != NULL) && 1521 (out_buf.st_size > 0)) { 1522 dump_out_file(rp->stderr_file, true); 1523 } 1524 (void) unlink(rp->stderr_file); 1525 retmem_mb(rp->stderr_file); 1526 rp->stderr_file = NULL; 1527 } 1528 #endif 1529 } 1530 check_state(rp->temp_file); 1531 if (rp->temp_file != NULL) { 1532 free_name(rp->temp_file); 1533 } 1534 rp->temp_file = NULL; 1535 if (rp->state == build_failed) { 1536 line = get_prop(rp->target->prop, line_prop); 1537 if (line != NULL) { 1538 line->body.line.command_used = NULL; 1539 } 1540 if (continue_after_error || 1541 fatal_in_progress || 1542 !docheck) { 1543 warning(catgets(catd, 1, 256, "Command failed for target `%s'"), 1544 rp->command ? line2->body.line.target->string_mb : rp->target->string_mb); 1545 build_failed_seen = true; 1546 } else { 1547 /* 1548 * XXX??? - DMake needs to exit(), 1549 * but shouldn't call fatal(). 1550 */ 1551 #ifdef PRINT_EXIT_STATUS 1552 warning(NOCATGETS("I'm in finish_children. rp->state == build_failed.")); 1553 #endif 1554 1555 fatal(catgets(catd, 1, 258, "Command failed for target `%s'"), 1556 rp->command ? line2->body.line.target->string_mb : rp->target->string_mb); 1557 } 1558 } 1559 if (!docheck) { 1560 delete_running_struct(rp); 1561 rp = *rp_prev; 1562 if (rp == NULL) { 1563 break; 1564 } else { 1565 goto bypass_for_loop_inc_4; 1566 } 1567 } 1568 update_target(get_prop(rp->target->prop, line_prop), 1569 rp->state); 1570 finish_doname(rp); 1571 delete_running_struct(rp); 1572 rp = *rp_prev; 1573 if (rp == NULL) { 1574 break; 1575 } else { 1576 goto bypass_for_loop_inc_4; 1577 } 1578 } else { 1579 rp_prev = &rp->next; 1580 } 1581 } 1582 } 1583 1584 /* 1585 * dump_out_file(filename, err) 1586 * 1587 * Write the contents of the file to stdout, then unlink the file. 1588 * 1589 * Parameters: 1590 * filename Name of temp file containing output 1591 * 1592 * Global variables used: 1593 */ 1594 static void 1595 dump_out_file(char *filename, Boolean err) 1596 { 1597 int chars_read; 1598 char copybuf[BUFSIZ]; 1599 int fd; 1600 int out_fd = (err ? 2 : 1); 1601 1602 if ((fd = open(filename, O_RDONLY)) < 0) { 1603 fatal(catgets(catd, 1, 141, "open failed for output file %s: %s"), 1604 filename, 1605 errmsg(errno)); 1606 } 1607 if (!silent && output_mode != txt2_mode) { 1608 (void) fprintf(err ? stderr : stdout, 1609 err ? 1610 catgets(catd, 1, 338, "%s --> Job errors\n") : 1611 catgets(catd, 1, 259, "%s --> Job output\n"), 1612 local_host); 1613 (void) fflush(err ? stderr : stdout); 1614 } 1615 for (chars_read = read(fd, copybuf, BUFSIZ); 1616 chars_read > 0; 1617 chars_read = read(fd, copybuf, BUFSIZ)) { 1618 /* 1619 * Read buffers from the source file until end or error. 1620 */ 1621 if (write(out_fd, copybuf, chars_read) < 0) { 1622 fatal(catgets(catd, 1, 260, "write failed for output file %s: %s"), 1623 filename, 1624 errmsg(errno)); 1625 } 1626 } 1627 (void) close(fd); 1628 (void) unlink(filename); 1629 } 1630 1631 /* 1632 * finish_doname(rp) 1633 * 1634 * Completes the processing for a target which was left running. 1635 * 1636 * Parameters: 1637 * rp Running list entry for target 1638 * 1639 * Global variables used: 1640 * debug_level Debug flag 1641 * recursion_level Indentation for debug output 1642 */ 1643 static void 1644 finish_doname(Running rp) 1645 { 1646 int auto_count = rp->auto_count; 1647 Name *automatics = rp->automatics; 1648 Doname result = rp->state; 1649 Name target = rp->target; 1650 Name true_target = rp->true_target; 1651 Property *conditionals; 1652 1653 recursion_level = rp->recursion_level; 1654 if (result == build_ok) { 1655 if (true_target == NULL) { 1656 (void) printf(NOCATGETS("Target = %s\n"), target->string_mb); 1657 (void) printf(NOCATGETS(" State = %d\n"), result); 1658 fatal(NOCATGETS("Internal error: NULL true_target in finish_doname")); 1659 } 1660 /* If all went OK, set a nice timestamp */ 1661 if (true_target->stat.time == file_doesnt_exist) { 1662 true_target->stat.time = file_max_time; 1663 } 1664 } 1665 target->state = result; 1666 if (target->is_member) { 1667 Property member; 1668 1669 /* Propagate the timestamp from the member file to the member */ 1670 if ((target->stat.time != file_max_time) && 1671 ((member = get_prop(target->prop, member_prop)) != NULL) && 1672 (exists(member->body.member.member) > file_doesnt_exist)) { 1673 target->stat.time = 1674 /* 1675 exists(member->body.member.member); 1676 */ 1677 member->body.member.member->stat.time; 1678 } 1679 } 1680 /* 1681 * Check if we found any new auto dependencies when we 1682 * built the target. 1683 */ 1684 if ((result == build_ok) && check_auto_dependencies(target, 1685 auto_count, 1686 automatics)) { 1687 if (debug_level > 0) { 1688 (void) printf(catgets(catd, 1, 261, "%*sTarget `%s' acquired new dependencies from build, checking all dependencies\n"), 1689 recursion_level, 1690 "", 1691 true_target->string_mb); 1692 } 1693 target->rechecking_target = true; 1694 target->state = build_running; 1695 1696 /* [tolik, Tue Mar 25 1997] 1697 * Fix for bug 4038824: 1698 * command line options set by conditional macros get dropped 1699 * rp->conditional_cnt and rp->conditional_targets must be copied 1700 * to new 'rp' during add_pending(). Set_conditionals() stores 1701 * rp->conditional_targets to the global variable 'conditional_targets' 1702 * Add_pending() will use this variable to set up 'rp'. 1703 */ 1704 conditionals = set_conditionals(rp->conditional_cnt, rp->conditional_targets); 1705 add_pending(target, 1706 recursion_level, 1707 rp->do_get, 1708 rp->implicit, 1709 true); 1710 reset_conditionals(rp->conditional_cnt, rp->conditional_targets, conditionals); 1711 } 1712 } 1713 1714 /* 1715 * new_running_struct() 1716 * 1717 * Constructor for Running struct. Creates a structure and initializes 1718 * its fields. 1719 * 1720 */ 1721 static Running new_running_struct() 1722 { 1723 Running rp; 1724 1725 rp = ALLOC(Running); 1726 rp->target = NULL; 1727 rp->true_target = NULL; 1728 rp->command = NULL; 1729 rp->sprodep_value = NULL; 1730 rp->sprodep_env = NULL; 1731 rp->auto_count = 0; 1732 rp->automatics = NULL; 1733 rp->pid = -1; 1734 rp->job_msg_id = -1; 1735 rp->stdout_file = NULL; 1736 rp->stderr_file = NULL; 1737 rp->temp_file = NULL; 1738 rp->next = NULL; 1739 return rp; 1740 } 1741 1742 /* 1743 * add_running(target, true_target, command, recursion_level, auto_count, 1744 * automatics, do_get, implicit) 1745 * 1746 * Adds a record on the running list for this target, which 1747 * was just spawned and is running. 1748 * 1749 * Parameters: 1750 * target Target being built 1751 * true_target True target for target 1752 * command Running command. 1753 * recursion_level Debug indentation level 1754 * auto_count Count of automatic dependencies 1755 * automatics List of automatic dependencies 1756 * do_get Sccs get flag 1757 * implicit Implicit flag 1758 * 1759 * Static variables used: 1760 * running_tail Tail of running list 1761 * process_running PID of process 1762 * 1763 * Global variables used: 1764 * current_line Current line for target 1765 * current_target Current target being built 1766 * stderr_file Temporary file for stdout 1767 * stdout_file Temporary file for stdout 1768 * temp_file_name Temporary file for auto dependencies 1769 */ 1770 void 1771 add_running(Name target, Name true_target, Property command, int recursion_level, int auto_count, Name *automatics, Boolean do_get, Boolean implicit) 1772 { 1773 Running rp; 1774 Name *p; 1775 1776 rp = new_running_struct(); 1777 rp->state = build_running; 1778 rp->target = target; 1779 rp->true_target = true_target; 1780 rp->command = command; 1781 Property spro_val = get_prop(sunpro_dependencies->prop, macro_prop); 1782 if(spro_val) { 1783 rp->sprodep_value = spro_val->body.macro.value; 1784 spro_val->body.macro.value = NULL; 1785 spro_val = get_prop(sunpro_dependencies->prop, env_mem_prop); 1786 if(spro_val) { 1787 rp->sprodep_env = spro_val->body.env_mem.value; 1788 spro_val->body.env_mem.value = NULL; 1789 } 1790 } 1791 rp->recursion_level = recursion_level; 1792 rp->do_get = do_get; 1793 rp->implicit = implicit; 1794 rp->auto_count = auto_count; 1795 if (auto_count > 0) { 1796 rp->automatics = (Name *) getmem(auto_count * sizeof (Name)); 1797 for (p = rp->automatics; auto_count > 0; auto_count--) { 1798 *p++ = *automatics++; 1799 } 1800 } else { 1801 rp->automatics = NULL; 1802 } 1803 #ifdef DISTRIBUTED 1804 if (dmake_mode_type == distributed_mode) { 1805 rp->make_refd = called_make; 1806 called_make = false; 1807 } else 1808 #endif 1809 { 1810 rp->pid = process_running; 1811 process_running = -1; 1812 childPid = -1; 1813 } 1814 rp->job_msg_id = job_msg_id; 1815 rp->stdout_file = stdout_file; 1816 rp->stderr_file = stderr_file; 1817 rp->temp_file = temp_file_name; 1818 rp->redo = false; 1819 rp->next = NULL; 1820 store_conditionals(rp); 1821 stdout_file = NULL; 1822 stderr_file = NULL; 1823 temp_file_name = NULL; 1824 current_target = NULL; 1825 current_line = NULL; 1826 *running_tail = rp; 1827 running_tail = &rp->next; 1828 } 1829 1830 /* 1831 * add_pending(target, recursion_level, do_get, implicit, redo) 1832 * 1833 * Adds a record on the running list for a pending target 1834 * (waiting for its dependents to finish running). 1835 * 1836 * Parameters: 1837 * target Target being built 1838 * recursion_level Debug indentation level 1839 * do_get Sccs get flag 1840 * implicit Implicit flag 1841 * redo True if this target is being redone 1842 * 1843 * Static variables used: 1844 * running_tail Tail of running list 1845 */ 1846 void 1847 add_pending(Name target, int recursion_level, Boolean do_get, Boolean implicit, Boolean redo) 1848 { 1849 Running rp; 1850 rp = new_running_struct(); 1851 rp->state = build_pending; 1852 rp->target = target; 1853 rp->recursion_level = recursion_level; 1854 rp->do_get = do_get; 1855 rp->implicit = implicit; 1856 rp->redo = redo; 1857 store_conditionals(rp); 1858 *running_tail = rp; 1859 running_tail = &rp->next; 1860 } 1861 1862 /* 1863 * add_serial(target, recursion_level, do_get, implicit) 1864 * 1865 * Adds a record on the running list for a target which must be 1866 * executed in serial after others have finished. 1867 * 1868 * Parameters: 1869 * target Target being built 1870 * recursion_level Debug indentation level 1871 * do_get Sccs get flag 1872 * implicit Implicit flag 1873 * 1874 * Static variables used: 1875 * running_tail Tail of running list 1876 */ 1877 void 1878 add_serial(Name target, int recursion_level, Boolean do_get, Boolean implicit) 1879 { 1880 Running rp; 1881 1882 rp = new_running_struct(); 1883 rp->target = target; 1884 rp->recursion_level = recursion_level; 1885 rp->do_get = do_get; 1886 rp->implicit = implicit; 1887 rp->state = build_serial; 1888 rp->redo = false; 1889 store_conditionals(rp); 1890 *running_tail = rp; 1891 running_tail = &rp->next; 1892 } 1893 1894 /* 1895 * add_subtree(target, recursion_level, do_get, implicit) 1896 * 1897 * Adds a record on the running list for a target which must be 1898 * executed in isolation after others have finished. 1899 * 1900 * Parameters: 1901 * target Target being built 1902 * recursion_level Debug indentation level 1903 * do_get Sccs get flag 1904 * implicit Implicit flag 1905 * 1906 * Static variables used: 1907 * running_tail Tail of running list 1908 */ 1909 void 1910 add_subtree(Name target, int recursion_level, Boolean do_get, Boolean implicit) 1911 { 1912 Running rp; 1913 1914 rp = new_running_struct(); 1915 rp->target = target; 1916 rp->recursion_level = recursion_level; 1917 rp->do_get = do_get; 1918 rp->implicit = implicit; 1919 rp->state = build_subtree; 1920 rp->redo = false; 1921 store_conditionals(rp); 1922 *running_tail = rp; 1923 running_tail = &rp->next; 1924 } 1925 1926 /* 1927 * store_conditionals(rp) 1928 * 1929 * Creates an array of the currently active targets with conditional 1930 * macros (found in the chain conditional_targets) and puts that 1931 * array in the Running struct. 1932 * 1933 * Parameters: 1934 * rp Running struct for storing chain 1935 * 1936 * Global variables used: 1937 * conditional_targets Chain of current dynamic conditionals 1938 */ 1939 static void 1940 store_conditionals(Running rp) 1941 { 1942 int cnt; 1943 Chain cond_name; 1944 1945 if (conditional_targets == NULL) { 1946 rp->conditional_cnt = 0; 1947 rp->conditional_targets = NULL; 1948 return; 1949 } 1950 cnt = 0; 1951 for (cond_name = conditional_targets; 1952 cond_name != NULL; 1953 cond_name = cond_name->next) { 1954 cnt++; 1955 } 1956 rp->conditional_cnt = cnt; 1957 rp->conditional_targets = (Name *) getmem(cnt * sizeof(Name)); 1958 for (cond_name = conditional_targets; 1959 cond_name != NULL; 1960 cond_name = cond_name->next) { 1961 rp->conditional_targets[--cnt] = cond_name->name; 1962 } 1963 } 1964 1965 /* 1966 * parallel_ok(target, line_prop_must_exists) 1967 * 1968 * Returns true if the target can be run in parallel 1969 * 1970 * Return value: 1971 * True if can run in parallel 1972 * 1973 * Parameters: 1974 * target Target being tested 1975 * 1976 * Global variables used: 1977 * all_parallel True if all targets default to parallel 1978 * only_parallel True if no targets default to parallel 1979 */ 1980 Boolean 1981 parallel_ok(Name target, Boolean line_prop_must_exists) 1982 { 1983 Boolean assign; 1984 Boolean make_refd; 1985 Property line; 1986 Cmd_line rule; 1987 1988 assign = make_refd = false; 1989 if (((line = get_prop(target->prop, line_prop)) == NULL) && 1990 line_prop_must_exists) { 1991 return false; 1992 } 1993 if (line != NULL) { 1994 for (rule = line->body.line.command_used; 1995 rule != NULL; 1996 rule = rule->next) { 1997 if (rule->assign) { 1998 assign = true; 1999 } else if (rule->make_refd) { 2000 make_refd = true; 2001 } 2002 } 2003 } 2004 if (assign) { 2005 return false; 2006 } else if (target->parallel) { 2007 return true; 2008 } else if (target->no_parallel) { 2009 return false; 2010 } else if (all_parallel) { 2011 return true; 2012 } else if (only_parallel) { 2013 return false; 2014 } else if (make_refd) { 2015 return false; 2016 } else { 2017 return true; 2018 } 2019 } 2020 2021 /* 2022 * is_running(target) 2023 * 2024 * Returns true if the target is running. 2025 * 2026 * Return value: 2027 * True if target is running 2028 * 2029 * Parameters: 2030 * target Target to check 2031 * 2032 * Global variables used: 2033 * running_list List of running processes 2034 */ 2035 Boolean 2036 is_running(Name target) 2037 { 2038 Running rp; 2039 2040 if (target->state != build_running) { 2041 return false; 2042 } 2043 for (rp = running_list; 2044 rp != NULL && target != rp->target; 2045 rp = rp->next); 2046 if (rp == NULL) { 2047 return false; 2048 } else { 2049 return (rp->state == build_running) ? true : false; 2050 } 2051 } 2052 2053 /* 2054 * This function replaces the makesh binary. 2055 */ 2056 2057 #ifdef SGE_SUPPORT 2058 #define DO_CHECK(f) if (f <= 0) { \ 2059 fprintf(stderr, \ 2060 catgets(catd, 1, 347, "Could not write to file: %s: %s\n"), \ 2061 script_file, errmsg(errno)); \ 2062 _exit(1); \ 2063 } 2064 #endif /* SGE_SUPPORT */ 2065 2066 static pid_t 2067 run_rule_commands(char *host, char **commands) 2068 { 2069 Boolean always_exec; 2070 Name command; 2071 Boolean ignore; 2072 int length; 2073 Doname result; 2074 Boolean silent_flag; 2075 #ifdef SGE_SUPPORT 2076 wchar_t *wcmd, *tmp_wcs_buffer = NULL; 2077 char *cmd, *tmp_mbs_buffer = NULL; 2078 FILE *scrfp; 2079 Name shell = getvar(shell_name); 2080 #else 2081 wchar_t *tmp_wcs_buffer; 2082 #endif /* SGE_SUPPORT */ 2083 2084 childPid = fork(); 2085 switch (childPid) { 2086 case -1: /* Error */ 2087 fatal(catgets(catd, 1, 337, "Could not fork child process for dmake job: %s"), 2088 errmsg(errno)); 2089 break; 2090 case 0: /* Child */ 2091 /* To control the processed targets list is not the child's business */ 2092 running_list = NULL; 2093 #if defined(REDIRECT_ERR) 2094 if(out_err_same) { 2095 redirect_io(stdout_file, (char*)NULL); 2096 } else { 2097 redirect_io(stdout_file, stderr_file); 2098 } 2099 #else 2100 redirect_io(stdout_file, (char*)NULL); 2101 #endif 2102 #ifdef SGE_SUPPORT 2103 if (grid) { 2104 int fdes = mkstemp(script_file); 2105 if ((fdes < 0) || (scrfp = fdopen(fdes, "w")) == NULL) { 2106 fprintf(stderr, 2107 catgets(catd, 1, 341, "Could not create file: %s: %s\n"), 2108 script_file, errmsg(errno)); 2109 _exit(1); 2110 } 2111 if (IS_EQUAL(shell->string_mb, "")) { 2112 shell = shell_name; 2113 } 2114 } 2115 #endif /* SGE_SUPPORT */ 2116 for (commands = commands; 2117 (*commands != (char *)NULL); 2118 commands++) { 2119 silent_flag = silent; 2120 ignore = false; 2121 always_exec = false; 2122 while ((**commands == (int) at_char) || 2123 (**commands == (int) hyphen_char) || 2124 (**commands == (int) plus_char)) { 2125 if (**commands == (int) at_char) { 2126 silent_flag = true; 2127 } 2128 if (**commands == (int) hyphen_char) { 2129 ignore = true; 2130 } 2131 if (**commands == (int) plus_char) { 2132 always_exec = true; 2133 } 2134 (*commands)++; 2135 } 2136 #ifdef SGE_SUPPORT 2137 if (grid) { 2138 if ((length = strlen(*commands)) >= MAXPATHLEN / 2) { 2139 wcmd = tmp_wcs_buffer = ALLOC_WC(length * 2 + 1); 2140 (void) mbstowcs(tmp_wcs_buffer, *commands, length * 2 + 1); 2141 } else { 2142 MBSTOWCS(wcs_buffer, *commands); 2143 wcmd = wcs_buffer; 2144 cmd = mbs_buffer; 2145 } 2146 wchar_t *from = wcmd + wslen(wcmd); 2147 wchar_t *to = from + (from - wcmd); 2148 *to = (int) nul_char; 2149 while (from > wcmd) { 2150 *--to = *--from; 2151 if (*from == (int) newline_char) { // newline symbols are already quoted 2152 *--to = *--from; 2153 } else if (wschr(char_semantics_char, *from)) { 2154 *--to = (int) backslash_char; 2155 } 2156 } 2157 if (length >= MAXPATHLEN*MB_LEN_MAX/2) { // sizeof(mbs_buffer) / 2 2158 cmd = tmp_mbs_buffer = getmem((length * MB_LEN_MAX * 2) + 1); 2159 (void) wcstombs(tmp_mbs_buffer, to, (length * MB_LEN_MAX * 2) + 1); 2160 } else { 2161 WCSTOMBS(mbs_buffer, to); 2162 cmd = mbs_buffer; 2163 } 2164 char *mbst, *mbend; 2165 if ((length > 0) && 2166 !silent_flag) { 2167 for (mbst = cmd; (mbend = strstr(mbst, "\\\n")) != NULL; mbst = mbend + 2) { 2168 *mbend = '\0'; 2169 DO_CHECK(fprintf(scrfp, NOCATGETS("/usr/bin/printf '%%s\\n' %s\\\\\n"), mbst)); 2170 *mbend = '\\'; 2171 } 2172 DO_CHECK(fprintf(scrfp, NOCATGETS("/usr/bin/printf '%%s\\n' %s\n"), mbst)); 2173 } 2174 if (!do_not_exec_rule || 2175 !working_on_targets || 2176 always_exec) { 2177 #if defined(linux) 2178 if (0 != strcmp(shell->string_mb, (char*)NOCATGETS("/bin/sh"))) { 2179 DO_CHECK(fprintf(scrfp, NOCATGETS("%s -c %s\n"), shell->string_mb, cmd)); 2180 } else 2181 #endif 2182 DO_CHECK(fprintf(scrfp, NOCATGETS("%s -ce %s\n"), shell->string_mb, cmd)); 2183 DO_CHECK(fputs(NOCATGETS("__DMAKECMDEXITSTAT=$?\nif [ ${__DMAKECMDEXITSTAT} -ne 0 ]; then\n"), scrfp)); 2184 if (ignore) { 2185 DO_CHECK(fprintf(scrfp, NOCATGETS("\techo %s ${__DMAKECMDEXITSTAT} %s\n"), 2186 catgets(catd, 1, 343, "\"*** Error code"), 2187 catgets(catd, 1, 344, "(ignored)\""))); 2188 } else { 2189 DO_CHECK(fprintf(scrfp, NOCATGETS("\techo %s ${__DMAKECMDEXITSTAT}\n"), 2190 catgets(catd, 1, 342, "\"*** Error code\""))); 2191 } 2192 if (silent_flag) { 2193 DO_CHECK(fprintf(scrfp, NOCATGETS("\techo %s\n"), 2194 catgets(catd, 1, 345, "The following command caused the error:"))); 2195 for (mbst = cmd; (mbend = strstr(mbst, "\\\n")) != NULL; mbst = mbend + 2) { 2196 *mbend = '\0'; 2197 DO_CHECK(fprintf(scrfp, NOCATGETS("\t/usr/bin/printf '%%s\\n' %s\\\\\n"), mbst)); 2198 *mbend = '\\'; 2199 } 2200 DO_CHECK(fprintf(scrfp, NOCATGETS("\t/usr/bin/printf '%%s\\n' %s\n"), mbst)); 2201 } 2202 if (!ignore) { 2203 DO_CHECK(fputs(NOCATGETS("\texit ${__DMAKECMDEXITSTAT}\n"), scrfp)); 2204 } 2205 DO_CHECK(fputs(NOCATGETS("fi\n"), scrfp)); 2206 } 2207 if (tmp_wcs_buffer) { 2208 retmem_mb(tmp_mbs_buffer); 2209 tmp_mbs_buffer = NULL; 2210 } 2211 if (tmp_wcs_buffer) { 2212 retmem(tmp_wcs_buffer); 2213 tmp_wcs_buffer = NULL; 2214 } 2215 continue; 2216 } 2217 #endif /* SGE_SUPPORT */ 2218 if ((length = strlen(*commands)) >= MAXPATHLEN) { 2219 tmp_wcs_buffer = ALLOC_WC(length + 1); 2220 (void) mbstowcs(tmp_wcs_buffer, *commands, length + 1); 2221 command = GETNAME(tmp_wcs_buffer, FIND_LENGTH); 2222 retmem(tmp_wcs_buffer); 2223 } else { 2224 MBSTOWCS(wcs_buffer, *commands); 2225 command = GETNAME(wcs_buffer, FIND_LENGTH); 2226 } 2227 if ((command->hash.length > 0) && 2228 !silent_flag) { 2229 (void) printf("%s\n", command->string_mb); 2230 } 2231 result = dosys(command, 2232 ignore, 2233 false, 2234 false, /* bugs #4085164 & #4990057 */ 2235 /* BOOLEAN(silent_flag && ignore), */ 2236 always_exec, 2237 (Name) NULL, 2238 false); 2239 if (result == build_failed) { 2240 if (silent_flag) { 2241 (void) printf(catgets(catd, 1, 152, "The following command caused the error:\n%s\n"), command->string_mb); 2242 } 2243 if (!ignore) { 2244 _exit(1); 2245 } 2246 } 2247 } 2248 #ifndef SGE_SUPPORT 2249 _exit(0); 2250 #else 2251 if (!grid) { 2252 _exit(0); 2253 } 2254 DO_CHECK(fputs(NOCATGETS("exit 0\n"), scrfp)); 2255 if (fclose(scrfp) != 0) { 2256 fprintf(stderr, 2257 catgets(catd, 1, 346, "Could not close file: %s: %s\n"), 2258 script_file, errmsg(errno)); 2259 _exit(1); 2260 } 2261 { 2262 2263 #define DEFAULT_QRSH_TRIES_NUMBER 1 2264 #define DEFAULT_QRSH_TIMEOUT 0 2265 2266 static char *sge_env_var = NULL; 2267 static int qrsh_tries_number = DEFAULT_QRSH_TRIES_NUMBER; 2268 static int qrsh_timeout = DEFAULT_QRSH_TIMEOUT; 2269 #define SGE_DEBUG 2270 #ifdef SGE_DEBUG 2271 static Boolean do_not_remove = false; 2272 #endif /* SGE_DEBUG */ 2273 if (sge_env_var == NULL) { 2274 sge_env_var = getenv(NOCATGETS("__SPRO_DMAKE_SGE_TRIES")); 2275 if (sge_env_var != NULL) { 2276 qrsh_tries_number = atoi(sge_env_var); 2277 if (qrsh_tries_number < 1 || qrsh_tries_number > 9) { 2278 qrsh_tries_number = DEFAULT_QRSH_TRIES_NUMBER; 2279 } 2280 } 2281 sge_env_var = getenv(NOCATGETS("__SPRO_DMAKE_SGE_TIMEOUT")); 2282 if (sge_env_var != NULL) { 2283 qrsh_timeout = atoi(sge_env_var); 2284 if (qrsh_timeout <= 0) { 2285 qrsh_timeout = DEFAULT_QRSH_TIMEOUT; 2286 } 2287 } else { 2288 sge_env_var = ""; 2289 } 2290 #ifdef SGE_DEBUG 2291 sge_env_var = getenv(NOCATGETS("__SPRO_DMAKE_SGE_DEBUG")); 2292 if (sge_env_var == NULL) { 2293 sge_env_var = ""; 2294 } 2295 if (strstr(sge_env_var, NOCATGETS("noqrsh")) != NULL) 2296 qrsh_tries_number = 0; 2297 if (strstr(sge_env_var, NOCATGETS("donotremove")) != NULL) 2298 do_not_remove = true; 2299 #endif /* SGE_DEBUG */ 2300 } 2301 for (int i = qrsh_tries_number; ; i--) 2302 if ((childPid = fork()) < 0) { 2303 fatal(catgets(catd, 1, 348, "Could not fork child process for qrsh job: %s"), 2304 errmsg(errno)); 2305 _exit(1); 2306 } else if (childPid == 0) { 2307 enable_interrupt((void (*) (int))SIG_DFL); 2308 if (i > 0) { 2309 static char qrsh_cmd[50+MAXPATHLEN] = NOCATGETS("qrsh -cwd -V -noshell -nostdin /bin/sh "); 2310 static char *fname_ptr = NULL; 2311 static char *argv[] = { NOCATGETS("sh"), 2312 NOCATGETS("-fce"), 2313 qrsh_cmd, 2314 NULL}; 2315 if (fname_ptr == NULL) { 2316 fname_ptr = qrsh_cmd + strlen(qrsh_cmd); 2317 } 2318 strcpy(fname_ptr, script_file); 2319 (void) execve(NOCATGETS("/bin/sh"), argv, environ); 2320 } else { 2321 static char *argv[] = { NOCATGETS("sh"), 2322 script_file, 2323 NULL}; 2324 (void) execve(NOCATGETS("/bin/sh"), argv, environ); 2325 } 2326 fprintf(stderr, 2327 catgets(catd, 1, 349, "Could not load `qrsh': %s\n"), 2328 errmsg(errno)); 2329 _exit(1); 2330 } else { 2331 int status; 2332 pid_t pid; 2333 while ((pid = wait(&status)) != childPid) { 2334 if (pid == -1) { 2335 fprintf(stderr, 2336 catgets(catd, 1, 350, "wait() failed: %s\n"), 2337 errmsg(errno)); 2338 _exit(1); 2339 } 2340 } 2341 if (status != 0 && i > 0) { 2342 if (i > 1) { 2343 sleep(qrsh_timeout); 2344 } 2345 continue; 2346 } 2347 #ifdef SGE_DEBUG 2348 if (do_not_remove) { 2349 if (status) { 2350 fprintf(stderr, 2351 NOCATGETS("SGE script failed: %s\n"), 2352 script_file); 2353 } 2354 _exit(status ? 1 : 0); 2355 } 2356 #endif /* SGE_DEBUG */ 2357 (void) unlink(script_file); 2358 _exit(status ? 1 : 0); 2359 } 2360 } 2361 #endif /* SGE_SUPPORT */ 2362 break; 2363 default: 2364 break; 2365 } 2366 return childPid; 2367 } 2368 2369 static void 2370 maybe_reread_make_state(void) 2371 { 2372 /* Copying dosys()... */ 2373 if (report_dependencies_level == 0) { 2374 make_state->stat.time = file_no_time; 2375 (void) exists(make_state); 2376 if (make_state_before == make_state->stat.time) { 2377 return; 2378 } 2379 makefile_type = reading_statefile; 2380 if (read_trace_level > 1) { 2381 trace_reader = true; 2382 } 2383 temp_file_number++; 2384 (void) read_simple_file(make_state, 2385 false, 2386 false, 2387 false, 2388 false, 2389 false, 2390 true); 2391 trace_reader = false; 2392 } 2393 } 2394 2395 #ifdef DISTRIBUTED 2396 /* 2397 * Create and send an Avo_MToolJobResultMsg. 2398 */ 2399 static void 2400 send_job_result_msg(Running rp) 2401 { 2402 Avo_MToolJobResultMsg *msg; 2403 RWCollectable *xdr_msg; 2404 2405 msg = new Avo_MToolJobResultMsg(); 2406 msg->setResult(rp->job_msg_id, 2407 (rp->state == build_ok) ? 0 : 1, 2408 DONE); 2409 append_job_result_msg(msg, 2410 rp->stdout_file, 2411 rp->stderr_file); 2412 2413 xdr_msg = (RWCollectable *)msg; 2414 xdr(get_xdrs_ptr(), xdr_msg); 2415 (void) fflush(get_mtool_msgs_fp()); 2416 2417 delete msg; 2418 } 2419 2420 /* 2421 * Append the stdout/err to Avo_MToolJobResultMsg. 2422 */ 2423 static void 2424 append_job_result_msg(Avo_MToolJobResultMsg *msg, char *outFn, char *errFn) 2425 { 2426 FILE *fp; 2427 char line[MAXPATHLEN]; 2428 2429 fp = fopen(outFn, "r"); 2430 if (fp == NULL) { 2431 /* Hmmm... what should we do here? */ 2432 return; 2433 } 2434 while (fgets(line, MAXPATHLEN, fp) != NULL) { 2435 if (line[strlen(line) - 1] == '\n') { 2436 line[strlen(line) - 1] = '\0'; 2437 } 2438 msg->appendOutput(AVO_STRDUP(line)); 2439 } 2440 (void) fclose(fp); 2441 } 2442 #endif 2443 2444 static void 2445 delete_running_struct(Running rp) 2446 { 2447 if ((rp->conditional_cnt > 0) && 2448 (rp->conditional_targets != NULL)) { 2449 retmem_mb((char *) rp->conditional_targets); 2450 } 2451 /**/ 2452 if ((rp->auto_count > 0) && 2453 (rp->automatics != NULL)) { 2454 retmem_mb((char *) rp->automatics); 2455 } 2456 /**/ 2457 if(rp->sprodep_value) { 2458 free_name(rp->sprodep_value); 2459 } 2460 if(rp->sprodep_env) { 2461 retmem_mb(rp->sprodep_env); 2462 } 2463 retmem_mb((char *) rp); 2464 2465 } 2466 2467 #endif 2468