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 
  27 /*
  28  *      dosys.cc
  29  *
  30  *      Execute one commandline
  31  */
  32 
  33 /*
  34  * Included files
  35  */
  36 #include <sys/wait.h>                     /* WIFEXITED(status) */
  37 #include <alloca.h>               /* alloca() */
  38 
  39 #include <stdio.h>                /* errno */
  40 #include <errno.h>                /* errno */
  41 #include <fcntl.h>                /* open() */
  42 #include <mksh/dosys.h>
  43 #include <mksh/macro.h>           /* getvar() */
  44 #include <mksh/misc.h>            /* getmem(), fatal_mksh(), errmsg() */
  45 #include <sys/signal.h>           /* SIG_DFL */
  46 #include <sys/stat.h>             /* open() */
  47 #include <sys/wait.h>             /* wait() */
  48 #include <ulimit.h>               /* ulimit() */
  49 #include <unistd.h>               /* close(), dup2() */
  50 #include <libintl.h>
  51 
  52 /*
  53  * typedefs & structs
  54  */
  55 
  56 /*
  57  * Static variables
  58  */
  59 
  60 /*
  61  * File table of contents
  62  */
  63 static Boolean  exec_vp(register char *name, register char **argv, char **envp, register Boolean ignore_error, pathpt vroot_path);
  64 
  65 /*
  66  * Workaround for NFS bug. Sometimes, when running 'open' on a remote
  67  * dmake server, it fails with "Stale NFS file handle" error.
  68  * The second attempt seems to work.
  69  */
  70 int
  71 my_open(const char *path, int oflag, mode_t mode) {
  72         int res = open(path, oflag, mode);
  73         if (res < 0 && (errno == ESTALE || errno == EAGAIN)) {
  74                 /* Stale NFS file handle. Try again */
  75                 res = open(path, oflag, mode);
  76         }
  77         return res;
  78 }
  79 
  80 /*
  81  *      void
  82  *      redirect_io(char *stdout_file, char *stderr_file)
  83  *
  84  *      Redirects stdout and stderr for a child mksh process.
  85  */
  86 void
  87 redirect_io(char *stdout_file, char *stderr_file)
  88 {        
  89         long            descriptor_limit;
  90         int             i;
  91 
  92         if ((descriptor_limit = ulimit(UL_GDESLIM)) < 0) {
  93                 fatal_mksh(gettext("ulimit() failed: %s"), errmsg(errno));
  94         }
  95         for (i = 3; i < descriptor_limit; i++) {
  96                 (void) close(i);
  97         }
  98         if ((i = my_open(stdout_file,
  99                  O_WRONLY | O_CREAT | O_TRUNC | O_DSYNC,
 100                  S_IREAD | S_IWRITE)) < 0) {
 101                 fatal_mksh(gettext("Couldn't open standard out temp file `%s': %s"),
 102                       stdout_file,
 103                       errmsg(errno));
 104         } else {
 105                 if (dup2(i, 1) == -1) {
 106                         fatal_mksh("*** Error: dup2(3, 1) failed: %s",
 107                                 errmsg(errno));
 108                 }
 109                 close(i);
 110         }
 111         if (stderr_file == NULL) {
 112                 if (dup2(1, 2) == -1) {
 113                         fatal_mksh("*** Error: dup2(1, 2) failed: %s",
 114                                 errmsg(errno));
 115                 }
 116         } else if ((i = my_open(stderr_file,
 117                         O_WRONLY | O_CREAT | O_TRUNC | O_DSYNC,
 118                         S_IREAD | S_IWRITE)) < 0) {
 119                 fatal_mksh(gettext("Couldn't open standard error temp file `%s': %s"),
 120                       stderr_file,
 121                       errmsg(errno));
 122         } else {
 123                 if (dup2(i, 2) == -1) {
 124                         fatal_mksh("*** Error: dup2(3, 2) failed: %s",
 125                                 errmsg(errno));
 126                 }
 127                 close(i);
 128         }
 129 }
 130 
 131 /*
 132  *      doshell(command, ignore_error)
 133  *
 134  *      Used to run command lines that include shell meta-characters.
 135  *      The make macro SHELL is supposed to contain a path to the shell.
 136  *
 137  *      Return value:
 138  *                              The pid of the process we started
 139  *
 140  *      Parameters:
 141  *              command         The command to run
 142  *              ignore_error    Should we abort on error?
 143  *
 144  *      Global variables used:
 145  *              filter_stderr   If -X is on we redirect stderr
 146  *              shell_name      The Name "SHELL", used to get the path to shell
 147  */
 148 int
 149 doshell(wchar_t *command, register Boolean ignore_error, char *stdout_file, char *stderr_file, int nice_prio)
 150 {
 151         char                    *argv[6];
 152         int                     argv_index = 0;
 153         int                     cmd_argv_index;
 154         int                     length;
 155         char                    nice_prio_buf[MAXPATHLEN];
 156         register Name           shell = getvar(shell_name);
 157         register char           *shellname;
 158         char                    *tmp_mbs_buffer;
 159 
 160 
 161         if (IS_EQUAL(shell->string_mb, "")) {
 162                 shell = shell_name;
 163         }
 164         if ((shellname = strrchr(shell->string_mb, (int) slash_char)) == NULL) {
 165                 shellname = shell->string_mb;
 166         } else {
 167                 shellname++;
 168         }
 169 
 170         /*
 171          * Only prepend the /usr/bin/nice command to the original command
 172          * if the nice priority, nice_prio, is NOT zero (0).
 173          * Nice priorities can be a positive or a negative number.
 174          */
 175         if (nice_prio != 0) {
 176                 argv[argv_index++] = (char *)"nice";
 177                 (void) sprintf(nice_prio_buf, "-%d", nice_prio);
 178                 argv[argv_index++] = strdup(nice_prio_buf);
 179         }
 180         argv[argv_index++] = shellname;
 181         argv[argv_index++] = (char*)(ignore_error ? "-c" : "-ce");
 182         if ((length = wcslen(command)) >= MAXPATHLEN) {
 183                 tmp_mbs_buffer = getmem((length * MB_LEN_MAX) + 1);
 184                 (void) wcstombs(tmp_mbs_buffer, command, (length * MB_LEN_MAX) + 1);
 185                 cmd_argv_index = argv_index;
 186                 argv[argv_index++] = strdup(tmp_mbs_buffer);
 187                 retmem_mb(tmp_mbs_buffer);
 188         } else {
 189                 WCSTOMBS(mbs_buffer, command);
 190                 cmd_argv_index = argv_index;
 191                 argv[argv_index++] = strdup(mbs_buffer);
 192         }
 193         argv[argv_index] = NULL;
 194         (void) fflush(stdout);
 195         if ((childPid = fork()) == 0) {
 196                 enable_interrupt((void (*) (int)) SIG_DFL);
 197 #if 0
 198                 if (filter_stderr) {
 199                         redirect_stderr();
 200                 }
 201 #endif
 202                 if (nice_prio != 0) {
 203                         (void) execve("/usr/bin/nice", argv, environ);
 204                         fatal_mksh(gettext("Could not load `/usr/bin/nice': %s"),
 205                               errmsg(errno));
 206                 } else {
 207                         (void) execve(shell->string_mb, argv, environ);
 208                         fatal_mksh(gettext("Could not load Shell from `%s': %s"),
 209                               shell->string_mb,
 210                               errmsg(errno));
 211                 }
 212         }
 213         if (childPid  == -1) {
 214                 fatal_mksh(gettext("fork failed: %s"),
 215                       errmsg(errno));
 216         }
 217         retmem_mb(argv[cmd_argv_index]);
 218         return childPid;
 219 }
 220 
 221 /*
 222  *      exec_vp(name, argv, envp, ignore_error)
 223  *
 224  *      Like execve, but does path search.
 225  *      This starts command when make invokes it directly (without a shell).
 226  *
 227  *      Return value:
 228  *                              Returns false if the exec failed
 229  *
 230  *      Parameters:
 231  *              name            The name of the command to run
 232  *              argv            Arguments for the command
 233  *              envp            The environment for it
 234  *              ignore_error    Should we abort on error?
 235  *
 236  *      Global variables used:
 237  *              shell_name      The Name "SHELL", used to get the path to shell
 238  *              vroot_path      The path used by the vroot package
 239  */
 240 static Boolean
 241 exec_vp(register char *name, register char **argv, char **envp, register Boolean ignore_error, pathpt vroot_path)
 242 {
 243         register Name           shell = getvar(shell_name);
 244         register char           *shellname;
 245         char                    *shargv[4];
 246         Name                    tmp_shell;
 247 
 248         if (IS_EQUAL(shell->string_mb, "")) {
 249                 shell = shell_name;
 250         }
 251 
 252         for (int i = 0; i < 5; i++) {
 253                 (void) execve_vroot(name,
 254                                     argv + 1,
 255                                     envp,
 256                                     vroot_path,
 257                                     VROOT_DEFAULT);
 258                 switch (errno) {
 259                 case ENOEXEC:
 260                 case ENOENT:
 261                         /* That failed. Let the shell handle it */
 262                         shellname = strrchr(shell->string_mb, (int) slash_char);
 263                         if (shellname == NULL) {
 264                                 shellname = shell->string_mb;
 265                         } else {
 266                                 shellname++;
 267                         }
 268                         shargv[0] = shellname;
 269                         shargv[1] = (char*)(ignore_error ? "-c" : "-ce");
 270                         shargv[2] = argv[0];
 271                         shargv[3] = NULL;
 272                         tmp_shell = getvar(shell_name);
 273                         if (IS_EQUAL(tmp_shell->string_mb, "")) {
 274                                 tmp_shell = shell_name;
 275                         }
 276                         (void) execve_vroot(tmp_shell->string_mb,
 277                                             shargv,
 278                                             envp,
 279                                             vroot_path,
 280                                             VROOT_DEFAULT);
 281                         return failed;
 282                 case ETXTBSY:
 283                         /*
 284                          * The program is busy (debugged?).
 285                          * Wait and then try again.
 286                          */
 287                         (void) sleep((unsigned) i);
 288                 case EAGAIN:
 289                         break;
 290                 default:
 291                         return failed;
 292                 }
 293         }
 294         return failed;
 295 }
 296 
 297 /*
 298  *      doexec(command, ignore_error)
 299  *
 300  *      Will scan an argument string and split it into words
 301  *      thus building an argument list that can be passed to exec_ve()
 302  *
 303  *      Return value:
 304  *                              The pid of the process started here
 305  *
 306  *      Parameters:
 307  *              command         The command to run
 308  *              ignore_error    Should we abort on error?
 309  *
 310  *      Global variables used:
 311  *              filter_stderr   If -X is on we redirect stderr
 312  */
 313 int
 314 doexec(register wchar_t *command, register Boolean ignore_error, char *stdout_file, char *stderr_file, pathpt vroot_path, int nice_prio)
 315 {
 316         int                     arg_count = 5;
 317         char                    **argv;
 318         int                     length;
 319         char                    nice_prio_buf[MAXPATHLEN];
 320         register char           **p;
 321         wchar_t                 *q;
 322         register wchar_t        *t;
 323         char                    *tmp_mbs_buffer;
 324 
 325         /*
 326          * Only prepend the /usr/bin/nice command to the original command
 327          * if the nice priority, nice_prio, is NOT zero (0).
 328          * Nice priorities can be a positive or a negative number.
 329          */
 330         if (nice_prio != 0) {
 331                 arg_count += 2;
 332         }
 333         for (t = command; *t != (int) nul_char; t++) {
 334                 if (iswspace(*t)) {
 335                         arg_count++;
 336                 }
 337         }
 338         argv = (char **)alloca(arg_count * (sizeof(char *)));
 339         /*
 340          * Reserve argv[0] for sh in case of exec_vp failure.
 341          * Don't worry about prepending /usr/bin/nice command to argv[0].
 342          * In fact, doing it may cause the sh command to fail!
 343          */
 344         p = &argv[1];
 345         if ((length = wcslen(command)) >= MAXPATHLEN) {
 346                 tmp_mbs_buffer = getmem((length * MB_LEN_MAX) + 1);
 347                 (void) wcstombs(tmp_mbs_buffer, command, (length * MB_LEN_MAX) + 1);
 348                 argv[0] = strdup(tmp_mbs_buffer);
 349                 retmem_mb(tmp_mbs_buffer);
 350         } else {
 351                 WCSTOMBS(mbs_buffer, command);
 352                 argv[0] = strdup(mbs_buffer);
 353         }
 354 
 355         if (nice_prio != 0) {
 356                 *p++ = strdup("/usr/bin/nice");
 357                 (void) sprintf(nice_prio_buf, "-%d", nice_prio);
 358                 *p++ = strdup(nice_prio_buf);
 359         }
 360         /* Build list of argument words. */
 361         for (t = command; *t;) {
 362                 if (p >= &argv[arg_count]) {
 363                         /* This should never happen, right? */
 364                         WCSTOMBS(mbs_buffer, command);
 365                         fatal_mksh(gettext("Command `%s' has more than %d arguments"),
 366                               mbs_buffer,
 367                               arg_count);
 368                 }
 369                 q = t;
 370                 while (!iswspace(*t) && (*t != (int) nul_char)) {
 371                         t++;
 372                 }
 373                 if (*t) {
 374                         for (*t++ = (int) nul_char; iswspace(*t); t++);
 375                 }
 376                 if ((length = wcslen(q)) >= MAXPATHLEN) {
 377                         tmp_mbs_buffer = getmem((length * MB_LEN_MAX) + 1);
 378                         (void) wcstombs(tmp_mbs_buffer, q, (length * MB_LEN_MAX) + 1);
 379                         *p++ = strdup(tmp_mbs_buffer);
 380                         retmem_mb(tmp_mbs_buffer);
 381                 } else {
 382                         WCSTOMBS(mbs_buffer, q);
 383                         *p++ = strdup(mbs_buffer);
 384                 }
 385         }
 386         *p = NULL;
 387 
 388         /* Then exec the command with that argument list. */
 389         (void) fflush(stdout);
 390         if ((childPid = fork()) == 0) {
 391                 enable_interrupt((void (*) (int)) SIG_DFL);
 392 #if 0
 393                 if (filter_stderr) {
 394                         redirect_stderr();
 395                 }
 396 #endif
 397                 (void) exec_vp(argv[1], argv, environ, ignore_error, vroot_path);
 398                 fatal_mksh(gettext("Cannot load command `%s': %s"), argv[1], errmsg(errno));
 399         }
 400         if (childPid  == -1) {
 401                 fatal_mksh(gettext("fork failed: %s"),
 402                       errmsg(errno));
 403         }
 404         for (int i = 0; argv[i] != NULL; i++) {
 405                 retmem_mb(argv[i]);
 406         }
 407         return childPid;
 408 }
 409 
 410 /*
 411  *      await(ignore_error, silent_error, target, command, running_pid)
 412  *
 413  *      Wait for one child process and analyzes
 414  *      the returned status when the child process terminates.
 415  *
 416  *      Return value:
 417  *                              Returns true if commands ran OK
 418  *
 419  *      Parameters:
 420  *              ignore_error    Should we abort on error?
 421  *              silent_error    Should error messages be suppressed for dmake?
 422  *              target          The target we are building, for error msgs
 423  *              command         The command we ran, for error msgs
 424  *              running_pid     The pid of the process we are waiting for
 425  *              
 426  *      Static variables used:
 427  *              filter_file     The fd for the filter file
 428  *              filter_file_name The name of the filter file
 429  *
 430  *      Global variables used:
 431  *              filter_stderr   Set if -X is on
 432  */
 433 Boolean
 434 await(register Boolean ignore_error, register Boolean silent_error, Name target, wchar_t *command, pid_t running_pid, void *xdrs_p, int job_msg_id)
 435 {
 436         int                     status;
 437         char                    *buffer;
 438         int                     core_dumped;
 439         int                     exit_status;
 440         FILE                    *outfp;
 441         register pid_t          pid;
 442         struct stat             stat_buff;
 443         int                     termination_signal;
 444         char                    tmp_buf[MAXPATHLEN];
 445 
 446         while ((pid = wait(&status)) != running_pid) {
 447                 if (pid == -1) {
 448                         fatal_mksh(gettext("wait() failed: %s"), errmsg(errno));
 449                 }
 450         }
 451         (void) fflush(stdout);
 452         (void) fflush(stderr);
 453 
 454         if (status == 0) {
 455 
 456 #ifdef PRINT_EXIT_STATUS
 457                 warning_mksh("I'm in await(), and status is 0.");
 458 #endif
 459 
 460                 return succeeded;
 461         }
 462 
 463 #ifdef PRINT_EXIT_STATUS
 464         warning_mksh("I'm in await(), and status is *NOT* 0.");
 465 #endif
 466 
 467 
 468         exit_status = WEXITSTATUS(status);
 469 
 470 #ifdef PRINT_EXIT_STATUS
 471         warning_mksh("I'm in await(), and exit_status is %d.", exit_status);
 472 #endif
 473 
 474         termination_signal = WTERMSIG(status);
 475         core_dumped = WCOREDUMP(status);
 476 
 477         /*
 478          * If the child returned an error, we now try to print a
 479          * nice message about it.
 480          */
 481         
 482         tmp_buf[0] = (int) nul_char;
 483         if (!silent_error) {
 484                 if (exit_status != 0) {
 485                         (void) fprintf(stdout,
 486                                        gettext("*** Error code %d"),
 487                                        exit_status);
 488                 } else {
 489                                 (void) fprintf(stdout,
 490                                                gettext("*** Signal %d"),
 491                                                termination_signal);
 492                         if (core_dumped) {
 493                                 (void) fprintf(stdout,
 494                                                gettext(" - core dumped"));
 495                         }
 496                 }
 497                 if (ignore_error) {
 498                         (void) fprintf(stdout,
 499                                        gettext(" (ignored)"));
 500                 }
 501                 (void) fprintf(stdout, "\n");
 502                 (void) fflush(stdout);
 503         }
 504 
 505 #ifdef PRINT_EXIT_STATUS
 506         warning_mksh("I'm in await(), returning failed.");
 507 #endif
 508 
 509         return failed;
 510 }
 511 
 512 /*
 513  *      sh_command2string(command, destination)
 514  *
 515  *      Run one sh command and capture the output from it.
 516  *
 517  *      Return value:
 518  *
 519  *      Parameters:
 520  *              command         The command to run
 521  *              destination     Where to deposit the output from the command
 522  *              
 523  *      Static variables used:
 524  *
 525  *      Global variables used:
 526  */
 527 void
 528 sh_command2string(register String command, register String destination)
 529 {
 530         register FILE           *fd;
 531         register int            chr;
 532         int                     status;
 533         Boolean                 command_generated_output = false;
 534 
 535         command->text.p = (int) nul_char;
 536         WCSTOMBS(mbs_buffer, command->buffer.start);
 537         if ((fd = popen(mbs_buffer, "r")) == NULL) {
 538                 WCSTOMBS(mbs_buffer, command->buffer.start);
 539                 fatal_mksh(gettext("Could not run command `%s' for :sh transformation"),
 540                       mbs_buffer);
 541         }
 542         while ((chr = getc(fd)) != EOF) {
 543                 if (chr == (int) newline_char) {
 544                         chr = (int) space_char;
 545                 }
 546                 command_generated_output = true;
 547                 append_char(chr, destination);
 548         }
 549 
 550         /*
 551          * We don't want to keep the last LINE_FEED since usually
 552          * the output of the 'sh:' command is used to evaluate
 553          * some MACRO. ( /bin/sh and other shell add a line feed
 554          * to the output so that the prompt appear in the right place.
 555          * We don't need that
 556          */
 557         if (command_generated_output){
 558                 if ( *(destination->text.p-1) == (int) space_char) {
 559                         * (-- destination->text.p) = '\0';
 560                 } 
 561         } else {
 562                 /*
 563                  * If the command didn't generate any output,
 564                  * set the buffer to a null string.
 565                  */
 566                 *(destination->text.p) = '\0';
 567         }
 568                         
 569         status = pclose(fd);
 570         if (status != 0) {
 571                 WCSTOMBS(mbs_buffer, command->buffer.start);
 572                 fatal_mksh(gettext("The command `%s' returned status `%d'"),
 573                       mbs_buffer,
 574                       WEXITSTATUS(status));
 575         }
 576 }
 577 
 578