1 /* $Header: /src/pub/tcsh/tc.os.c,v 3.50 1998/10/25 15:10:35 christos Exp $ */ 2 /* 3 * tc.os.c: OS Dependent builtin functions 4 */ 5 /*- 6 * Copyright (c) 1980, 1991 The Regents of the University of California. 7 * All rights reserved. 8 * 9 * Redistribution and use in source and binary forms, with or without 10 * modification, are permitted provided that the following conditions 11 * are met: 12 * 1. Redistributions of source code must retain the above copyright 13 * notice, this list of conditions and the following disclaimer. 14 * 2. Redistributions in binary form must reproduce the above copyright 15 * notice, this list of conditions and the following disclaimer in the 16 * documentation and/or other materials provided with the distribution. 17 * 3. All advertising materials mentioning features or use of this software 18 * must display the following acknowledgement: 19 * This product includes software developed by the University of 20 * California, Berkeley and its contributors. 21 * 4. Neither the name of the University nor the names of its contributors 22 * may be used to endorse or promote products derived from this software 23 * without specific prior written permission. 24 * 25 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 26 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 27 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 28 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 29 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 30 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 31 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 32 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 33 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 34 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 35 * SUCH DAMAGE. 36 */ 37 #include "sh.h" 38 39 RCSID("$Id: tc.os.c,v 3.50 1998/10/25 15:10:35 christos Exp $") 40 41 #include "tw.h" 42 #include "ed.h" 43 #include "ed.defns.h" /* for the function names */ 44 #include "sh.decls.h" 45 46 #ifdef _UWIN 47 #define TIOCGPGRP TIOCGETPGRP 48 #define TIOCSPGRP TIOCSETPGRP 49 #endif 50 51 /*** 52 *** MACH 53 ***/ 54 55 #ifdef MACH 56 /* dosetpath -- setpath built-in command 57 * 58 ********************************************************************** 59 * HISTORY 60 * 08-May-88 Richard Draves (rpd) at Carnegie-Mellon University 61 * Major changes to remove artificial limits on sizes and numbers 62 * of paths. 63 * 64 ********************************************************************** 65 */ 66 67 #ifdef MACH 68 static Char STRCPATH[] = {'C', 'P', 'A', 'T', 'H', '\0'}; 69 static Char STRLPATH[] = {'L', 'P', 'A', 'T', 'H', '\0'}; 70 static Char STRMPATH[] = {'M', 'P', 'A', 'T', 'H', '\0'}; 71 # if EPATH 72 static Char STREPATH[] = {'E', 'P', 'A', 'T', 'H', '\0'}; 73 # endif 74 #endif /* MACH */ 75 static Char *syspaths[] = {STRKPATH, STRCPATH, STRLPATH, STRMPATH, 76 77 #if EPATH 78 STREPATH, 79 #endif 80 0}; 81 #define LOCALSYSPATH "/usr/local" 82 83 /*ARGSUSED*/ 84 void 85 dosetpath(arglist, c) 86 Char **arglist; 87 struct command *c; 88 { 89 extern char *getenv(); 90 sigmask_t omask; 91 Char **pathvars, **cmdargs; 92 char **spaths, **cpaths, **cmds; 93 char *tcp; 94 unsigned int npaths, ncmds; 95 int i, sysflag; 96 97 omask = sigsetmask(sigmask(SIGINT)); 98 99 /* 100 * setpath(3) uses stdio and we want 0, 1, 2 to work... 101 */ 102 if (!didfds) { 103 (void) dcopy(SHIN, 0); 104 (void) dcopy(SHOUT, 1); 105 (void) dcopy(SHDIAG, 2); 106 didfds = 1; 107 } 108 109 for (i = 1; arglist[i] && (arglist[i][0] != '-'); i++); 110 npaths = i - 1; 111 112 cmdargs = &arglist[i]; 113 for (; arglist[i]; i++); 114 ncmds = i - npaths - 1; 115 116 if (npaths) { 117 sysflag = 0; 118 pathvars = &arglist[1]; 119 } 120 else { 121 sysflag = 1; 122 npaths = (sizeof syspaths / sizeof *syspaths) - 1; 123 pathvars = syspaths; 124 } 125 126 /* note that npaths != 0 */ 127 128 spaths = (char **) xmalloc((size_t) npaths * sizeof *spaths); 129 setzero((char *) spaths, npaths * sizeof *spaths); 130 cpaths = (char **) xmalloc((size_t) (npaths + 1) * sizeof *cpaths); 131 setzero((char *) cpaths, (npaths + 1) * sizeof *cpaths); 132 cmds = (char **) xmalloc((size_t) (ncmds + 1) * sizeof *cmds); 133 setzero((char *) cmds, (ncmds + 1) * sizeof *cmds); 134 for (i = 0; i < npaths; i++) { 135 char *val = getenv(short2str(pathvars[i])); 136 137 if (val == NULL) 138 val = ""; 139 140 spaths[i] = (char *) xmalloc((size_t) (Strlen(pathvars[i]) + 141 strlen(val) + 2) * sizeof **spaths); 142 (void) strcpy(spaths[i], short2str(pathvars[i])); 143 (void) strcat(spaths[i], "="); 144 (void) strcat(spaths[i], val); 145 cpaths[i] = spaths[i]; 146 } 147 148 for (i = 0; i < ncmds; i++) { 149 Char *val = globone(cmdargs[i], G_ERROR); 150 151 if (val == NULL) 152 goto abortpath; 153 cmds[i] = (char *) xmalloc((size_t) Strlen(val) + 1); 154 (void) strcpy(cmds[i], short2str(val)); 155 } 156 157 158 if (setpath(cpaths, cmds, LOCALSYSPATH, sysflag, 1) < 0) { 159 abortpath: 160 if (spaths) { 161 for (i = 0; i < npaths; i++) 162 if (spaths[i]) 163 xfree((ptr_t) spaths[i]); 164 xfree((ptr_t) spaths); 165 } 166 if (cpaths) 167 xfree((ptr_t) cpaths); 168 if (cmds) { 169 for (i = 0; i < ncmds; i++) 170 if (cmds[i]) 171 xfree((ptr_t) cmds[i]); 172 xfree((ptr_t) cmds); 173 } 174 175 (void) sigsetmask(omask); 176 donefds(); 177 return; 178 } 179 180 for (i = 0; i < npaths; i++) { 181 Char *val, *name; 182 183 name = str2short(cpaths[i]); 184 for (val = str2short(cpaths[i]); val && *val && *val != '='; val++); 185 if (val && *val == '=') { 186 *val++ = '\0'; 187 188 tsetenv(name, val); 189 if (Strcmp(name, STRKPATH) == 0) { 190 importpath(val); 191 if (havhash) 192 dohash(NULL, NULL); 193 } 194 *--val = '='; 195 } 196 } 197 (void) sigsetmask(omask); 198 donefds(); 199 } 200 #endif /* MACH */ 201 202 /*** 203 *** AIX 204 ***/ 205 #ifdef TCF 206 /* ARGSUSED */ 207 void 208 dogetxvers(v, c) 209 Char **v; 210 struct command *c; 211 { 212 char xvers[MAXPATHLEN]; 213 214 if (getxvers(xvers, MAXPATHLEN) == -1) 215 stderror(ERR_SYSTEM, "getxvers", strerror(errno)); 216 xprintf("%s\n", xvers); 217 flush(); 218 } 219 220 /*ARGSUSED*/ 221 void 222 dosetxvers(v, c) 223 Char **v; 224 struct command *c; 225 { 226 char *xvers; 227 228 ++v; 229 if (!*v || *v[0] == '\0') 230 xvers = ""; 231 else 232 xvers = short2str(*v); 233 if (setxvers(xvers) == -1) 234 stderror(ERR_SYSTEM, "setxvers", strerror(errno)); 235 } 236 237 #include <sf.h> 238 #ifdef _AIXPS2 239 # define XC_PDP11 0x01 240 # define XC_23 0x02 241 # define XC_Z8K 0x03 242 # define XC_8086 0x04 243 # define XC_68K 0x05 244 # define XC_Z80 0x06 245 # define XC_VAX 0x07 246 # define XC_16032 0x08 247 # define XC_286 0x09 248 # define XC_386 0x0a 249 # define XC_S370 0x0b 250 #else 251 # include <sys/x.out.h> 252 #endif /* _AIXPS2 */ 253 254 static struct xc_cpu_t { 255 short xc_id; 256 char *xc_name; 257 } xcpu[] = 258 { 259 { XC_PDP11, "pdp11" }, 260 { XC_23, "i370" }, 261 { XC_Z8K, "z8000" }, 262 { XC_8086, "i86" }, 263 { XC_68K, "mc68000" }, 264 { XC_Z80, "x80" }, 265 { XC_VAX, "vax" }, 266 { XC_16032, "ns16032" }, 267 { XC_286, "i286" }, 268 { XC_386, "i386" }, 269 { XC_S370, "xa370" }, 270 { 0, NULL } 271 }; 272 273 /* 274 * our local hack table, stolen from x.out.h 275 */ 276 static char * 277 getxcode(xcid) 278 short xcid; 279 { 280 int i; 281 282 for (i = 0; xcpu[i].xc_name != NULL; i++) 283 if (xcpu[i].xc_id == xcid) 284 return (xcpu[i].xc_name); 285 return (NULL); 286 } 287 288 static short 289 getxid(xcname) 290 char *xcname; 291 { 292 int i; 293 294 for (i = 0; xcpu[i].xc_name != NULL; i++) 295 if (strcmp(xcpu[i].xc_name, xcname) == 0) 296 return (xcpu[i].xc_id); 297 return ((short) -1); 298 } 299 300 301 /*ARGSUSED*/ 302 void 303 dogetspath(v, c) 304 Char **v; 305 struct command *c; 306 { 307 int i, j; 308 sitepath_t p[MAXSITE]; 309 struct sf *st; 310 static char *local = "LOCAL "; 311 312 if ((j = getspath(p, MAXSITE)) == -1) 313 stderror(ERR_SYSTEM, "getspath", strerror(errno)); 314 for (i = 0; i < j && (p[i] & SPATH_CPU) != NOSITE; i++) { 315 if (p[i] & SPATH_CPU) { 316 if ((p[i] & SPATH_MASK) == NULLSITE) 317 xprintf(local); 318 else if ((st = sfxcode((short) (p[i] & SPATH_MASK))) != NULL) 319 xprintf("%s ", st->sf_ctype); 320 else { 321 char *xc = getxcode(p[i] & SPATH_MASK); 322 323 if (xc != NULL) 324 xprintf("%s ", xc); 325 else 326 xprintf("*cpu %d* ", (int) (p[i] & SPATH_MASK)); 327 /* 328 * BUG in the aix code... needs that cause if 329 * sfxcode fails once it fails for ever 330 */ 331 endsf(); 332 } 333 } 334 else { 335 if (p[i] == NULLSITE) 336 xprintf(local); 337 else if ((st = sfnum(p[i])) != NULL) 338 xprintf("%s ", st->sf_sname); 339 else 340 xprintf("*site %d* ", (int) (p[i] & SPATH_MASK)); 341 } 342 } 343 xputchar('\n'); 344 flush(); 345 } 346 347 /*ARGSUSED*/ 348 void 349 dosetspath(v, c) 350 Char **v; 351 struct command *c; 352 { 353 int i; 354 short j; 355 char *s; 356 sitepath_t p[MAXSITE]; 357 struct sf *st; 358 359 /* 360 * sfname() on AIX G9.9 at least, mallocs too pointers p, q 361 * then does the equivalent of while (*p++ == *q++) continue; 362 * and then tries to free(p,q) them! Congrats to the wizard who 363 * wrote that one. I bet he tested it really well too. 364 * Sooo, we set dont_free :-) 365 */ 366 dont_free = 1; 367 for (i = 0, v++; *v && *v[0] != '\0'; v++, i++) { 368 s = short2str(*v); 369 if (Isdigit(*s)) 370 p[i] = atoi(s); 371 else if (strcmp(s, "LOCAL") == 0) 372 p[i] = NULLSITE; 373 else if ((st = sfctype(s)) != NULL) 374 p[i] = SPATH_CPU | st->sf_ccode; 375 else if ((j = getxid(s)) != -1) 376 p[i] = SPATH_CPU | j; 377 else if ((st = sfname(s)) != NULL) 378 p[i] = st->sf_id; 379 else { 380 setname(s); 381 stderror(ERR_NAME | ERR_STRING, CGETS(23, 1, "Bad cpu/site name")); 382 } 383 if (i == MAXSITE - 1) 384 stderror(ERR_NAME | ERR_STRING, CGETS(23, 2, "Site path too long")); 385 } 386 if (setspath(p, i) == -1) 387 stderror(ERR_SYSTEM, "setspath", strerror(errno)); 388 dont_free = 0; 389 } 390 391 /* sitename(): 392 * Return the site name where the process is running 393 */ 394 char * 395 sitename(pid) 396 pid_t pid; 397 { 398 siteno_t ss; 399 struct sf *st; 400 401 if ((ss = site(pid)) == -1 || (st = sfnum(ss)) == NULL) 402 return CGETS(23, 3, "unknown"); 403 else 404 return st->sf_sname; 405 } 406 407 static int 408 migratepid(pid, new_site) 409 pid_t pid; 410 siteno_t new_site; 411 { 412 struct sf *st; 413 int need_local; 414 415 need_local = (pid == 0) || (pid == getpid()); 416 417 if (kill3((pid_t) pid, SIGMIGRATE, new_site) < 0) { 418 xprintf("%d: %s\n", pid, strerror(errno)); 419 return (-1); 420 } 421 422 if (need_local) { 423 if ((new_site = site(0)) == -1) { 424 xprintf(CGETS(23, 4, "site: %s\n"), strerror(errno)); 425 return (-1); 426 } 427 if ((st = sfnum(new_site)) == NULL) { 428 xprintf(CGETS(23, 5, "%d: Site not found\n"), new_site); 429 return (-1); 430 } 431 if (setlocal(st->sf_local, strlen(st->sf_local)) == -1) { 432 xprintf(CGETS(23, 6, "setlocal: %s: %s\n"), 433 st->sf_local, strerror(errno)); 434 return (-1); 435 } 436 } 437 return (0); 438 } 439 440 /*ARGSUSED*/ 441 void 442 domigrate(v, c) 443 Char **v; 444 struct command *c; 445 { 446 struct sf *st; 447 char *s; 448 Char *cp; 449 struct process *pp; 450 int err1 = 0; 451 int pid = 0; 452 siteno_t new_site = 0; 453 sigmask_t omask; 454 455 #ifdef BSDSIGS 456 omask = sigmask(SIGCHLD); 457 if (setintr) 458 omask |= sigmask(SIGINT); 459 omask = sigblock(omask) & ~omask; 460 #else 461 if (setintr) 462 (void) sighold(SIGINT); 463 (void) sighold(SIGCHLD); 464 #endif /* BSDSIGS */ 465 466 ++v; 467 if (*v[0] == '-') { 468 /* 469 * Do the -site. 470 */ 471 s = short2str(&v[0][1]); 472 /* 473 * see comment in setspath() 474 */ 475 dont_free = 1; 476 if ((st = sfname(s)) == NULL) { 477 setname(s); 478 stderror(ERR_NAME | ERR_STRING, CGETS(23, 7, "Site not found")); 479 } 480 dont_free = 0; 481 new_site = st->sf_id; 482 ++v; 483 } 484 485 if (!*v || *v[0] == '\0') { 486 if (migratepid(0, new_site) == -1) 487 err1++; 488 } 489 else { 490 gflag = 0, tglob(v); 491 if (gflag) { 492 v = globall(v); 493 if (v == 0) 494 stderror(ERR_NAME | ERR_NOMATCH); 495 } 496 else { 497 v = gargv = saveblk(v); 498 trim(v); 499 } 500 501 while (v && (cp = *v)) { 502 if (*cp == '%') { 503 pp = pfind(cp); 504 if (kill3((pid_t) - pp->p_jobid, SIGMIGRATE, new_site) < 0) { 505 xprintf("%S: %s\n", cp, strerror(errno)); 506 err1++; 507 } 508 } 509 else if (!(Isdigit(*cp) || *cp == '-')) 510 stderror(ERR_NAME | ERR_JOBARGS); 511 else { 512 pid = atoi(short2str(cp)); 513 if (migratepid(pid, new_site) == -1) 514 err1++; 515 } 516 v++; 517 } 518 if (gargv) 519 blkfree(gargv), gargv = 0; 520 } 521 522 done: 523 #ifdef BSDSIGS 524 (void) sigsetmask(omask); 525 #else 526 (void) sigrelse(SIGCHLD); 527 if (setintr) 528 (void) sigrelse(SIGINT); 529 #endif /* BSDSIGS */ 530 if (err1) 531 stderror(ERR_SILENT); 532 } 533 534 #endif /* TCF */ 535 536 /*** 537 *** CRAY ddmode <velo@sesun3.epfl.ch> (Martin Ouwehand EPFL-SIC/SE) 538 ***/ 539 #if defined(_CRAY) && !defined(_CRAYMPP) 540 void 541 dodmmode(v, c) 542 Char **v; 543 struct command *c; 544 { 545 Char *cp = v[1]; 546 547 USE(c); 548 549 if ( !cp ) { 550 int mode; 551 552 mode = dmmode(0); 553 dmmode(mode); 554 xprintf("%d\n",mode); 555 } 556 else { 557 if (cp[1] != '\0') 558 stderror(ERR_NAME | ERR_STRING, 559 CGETS(23, 30, "Too many arguments")); 560 else 561 switch(*cp) { 562 case '0': 563 dmmode(0); 564 break; 565 case '1': 566 dmmode(1); 567 break; 568 default: 569 stderror(ERR_NAME | ERR_STRING, 570 CGETS(23, 31, "Invalid argument")); 571 } 572 } 573 } 574 #endif /* _CRAY && !_CRAYMPP */ 575 576 577 /*** 578 *** CONVEX Warps. 579 ***/ 580 581 #ifdef WARP 582 /* 583 * handle the funky warping of symlinks 584 */ 585 #include <warpdb.h> 586 #include <sys/warp.h> 587 588 static jmp_buf sigsys_buf; 589 590 static sigret_t 591 catch_sigsys() 592 { 593 longjmp(sigsys_buf, 1); 594 } 595 596 597 /*ARGSUSED*/ 598 void 599 dowarp(v, c) 600 Char **v; 601 struct command *c; 602 { 603 int warp, oldwarp; 604 struct warpent *we; 605 void (*old_sigsys_handler) () = 0; 606 char *newwarp; 607 608 if (setjmp(sigsys_buf)) { 609 signal(SIGSYS, old_sigsys_handler); 610 stderror(ERR_NAME | ERR_STRING, 611 CGETS(23, 8, "You're trapped in a universe you never made")); 612 return; 613 } 614 old_sigsys_handler = signal(SIGSYS, catch_sigsys); 615 616 warp = getwarp(); 617 618 v++; 619 if (*v == 0) { /* display warp value */ 620 if (warp < 0) 621 stderror(ERR_NAME | ERR_STRING, CGETS(23, 9, "Getwarp failed")); 622 we = getwarpbyvalue(warp); 623 if (we) 624 printf("%s\n", we->w_name); 625 else 626 printf("%d\n", warp); 627 } 628 else { /* set warp value */ 629 oldwarp = warp; 630 newwarp = short2str(*v); 631 if (Isdigit(*v[0])) 632 warp = atoi(newwarp); 633 else { 634 we = getwarpbyname(newwarp); 635 if (we) 636 warp = we->w_value; 637 else 638 warp = -1; 639 } 640 if ((warp < 0) || (warp >= WARP_MAXLINK)) 641 stderror(ERR_NAME | ERR_STRING, CGETS(23, 10, "Invalid warp")); 642 if ((setwarp(warp) < 0) || (getwarp() != warp)) { 643 (void) setwarp(oldwarp); 644 stderror(ERR_NAME | ERR_STRING, CGETS(23, 11, "Setwarp failed")); 645 } 646 } 647 signal(SIGSYS, old_sigsys_handler); 648 return; 649 } 650 #endif /* WARP */ 651 652 /*** 653 *** Masscomp or HCX 654 ***/ 655 /* Added, DAS DEC-90. */ 656 #if defined(masscomp) || defined(_CX_UX) 657 /*ARGSUSED*/ 658 void 659 douniverse(v, c) 660 register Char **v; 661 struct command *c; 662 { 663 register Char *cp = v[1]; 664 register Char *cp2; /* dunno how many elements v comes in with */ 665 char ubuf[100]; 666 #ifdef BSDSIGS 667 register sigmask_t omask = 0; 668 #endif /* BSDSIGS */ 669 670 if (cp == 0) { 671 (void) getuniverse(ubuf); 672 xprintf("%s\n", ubuf); 673 } 674 else { 675 cp2 = v[2]; 676 if (cp2 == 0) { 677 if (*cp == '\0' || setuniverse(short2str(cp)) != 0) 678 stderror(ERR_NAME | ERR_STRING, CGETS(23, 12, "Illegal universe")); 679 } 680 else { 681 (void) getuniverse(ubuf); 682 if (*cp == '\0' || setuniverse(short2str(cp)) != 0) 683 stderror(ERR_NAME | ERR_STRING, CGETS(23, 12, "Illegal universe")); 684 if (setintr) 685 #ifdef BSDSIGS 686 omask = sigblock(sigmask(SIGINT)) & ~sigmask(SIGINT); 687 #else /* !BSDSIGS */ 688 (void) sighold(SIGINT); 689 #endif /* BSDSIGS */ 690 lshift(v, 2); 691 if (setintr) 692 #ifdef BSDSIGS 693 (void) sigsetmask(omask); 694 #else /* !BSDSIGS */ 695 (void) sigrelse (SIGINT); 696 #endif /* BSDSIGS */ 697 reexecute(c); 698 (void) setuniverse(ubuf); 699 } 700 } 701 } 702 #endif /* masscomp || _CX_UX */ 703 704 #if defined(_CX_UX) 705 /*ARGSUSED*/ 706 void 707 doatt(v, c) 708 register Char **v; 709 struct command *c; 710 { 711 register Char *cp = v[1]; 712 char ubuf[100]; 713 #ifdef BSDSIGS 714 register sigmask_t omask = 0; 715 #endif /* BSDSIGS */ 716 717 if (cp == 0) 718 (void) setuniverse("att"); 719 else { 720 (void) getuniverse(ubuf); 721 (void) setuniverse("att"); 722 if (setintr) 723 #ifdef BSDSIGS 724 omask = sigblock(sigmask(SIGINT)) & ~sigmask(SIGINT); 725 #else /* !BSDSIGS */ 726 (void) sighold(SIGINT); 727 #endif /* BSDSIGS */ 728 lshift(v, 1); 729 if (setintr) 730 #ifdef BSDSIGS 731 (void) sigsetmask(omask); 732 #else /* !BSDSIGS */ 733 (void) sigrelse (SIGINT); 734 #endif /* BSDSIGS */ 735 reexecute(c); 736 (void) setuniverse(ubuf); 737 } 738 } 739 740 /*ARGSUSED*/ 741 void 742 doucb(v, c) 743 register Char **v; 744 struct command *c; 745 { 746 register Char *cp = v[1]; 747 char ubuf[100]; 748 #ifdef BSDSIGS 749 register sigmask_t omask = 0; 750 #endif /* BSDSIGS */ 751 752 if (cp == 0) 753 (void) setuniverse("ucb"); 754 else { 755 (void) getuniverse(ubuf); 756 (void) setuniverse("ucb"); 757 if (setintr) 758 #ifdef BSDSIGS 759 omask = sigblock(sigmask(SIGINT)) & ~sigmask(SIGINT); 760 #else /* !BSDSIGS */ 761 (void) sighold(SIGINT); 762 #endif /* BSDSIGS */ 763 lshift(v, 1); 764 if (setintr) 765 #ifdef BSDSIGS 766 (void) sigsetmask(omask); 767 #else /* !BSDSIGS */ 768 (void) sigrelse (SIGINT); 769 #endif /* BSDSIGS */ 770 reexecute(c); 771 (void) setuniverse(ubuf); 772 } 773 } 774 #endif /* _CX_UX */ 775 776 #ifdef _SEQUENT_ 777 /* 778 * Compute the difference in process stats. 779 */ 780 void 781 pr_stat_sub(p2, p1, pr) 782 struct process_stats *p2, *p1, *pr; 783 { 784 pr->ps_utime.tv_sec = p2->ps_utime.tv_sec - p1->ps_utime.tv_sec; 785 pr->ps_utime.tv_usec = p2->ps_utime.tv_usec - p1->ps_utime.tv_usec; 786 if (pr->ps_utime.tv_usec < 0) { 787 pr->ps_utime.tv_sec -= 1; 788 pr->ps_utime.tv_usec += 1000000; 789 } 790 pr->ps_stime.tv_sec = p2->ps_stime.tv_sec - p1->ps_stime.tv_sec; 791 pr->ps_stime.tv_usec = p2->ps_stime.tv_usec - p1->ps_stime.tv_usec; 792 if (pr->ps_stime.tv_usec < 0) { 793 pr->ps_stime.tv_sec -= 1; 794 pr->ps_stime.tv_usec += 1000000; 795 } 796 797 pr->ps_maxrss = p2->ps_maxrss - p1->ps_maxrss; 798 pr->ps_pagein = p2->ps_pagein - p1->ps_pagein; 799 pr->ps_reclaim = p2->ps_reclaim - p1->ps_reclaim; 800 pr->ps_zerofill = p2->ps_zerofill - p1->ps_zerofill; 801 pr->ps_pffincr = p2->ps_pffincr - p1->ps_pffincr; 802 pr->ps_pffdecr = p2->ps_pffdecr - p1->ps_pffdecr; 803 pr->ps_swap = p2->ps_swap - p1->ps_swap; 804 pr->ps_syscall = p2->ps_syscall - p1->ps_syscall; 805 pr->ps_volcsw = p2->ps_volcsw - p1->ps_volcsw; 806 pr->ps_involcsw = p2->ps_involcsw - p1->ps_involcsw; 807 pr->ps_signal = p2->ps_signal - p1->ps_signal; 808 pr->ps_lread = p2->ps_lread - p1->ps_lread; 809 pr->ps_lwrite = p2->ps_lwrite - p1->ps_lwrite; 810 pr->ps_bread = p2->ps_bread - p1->ps_bread; 811 pr->ps_bwrite = p2->ps_bwrite - p1->ps_bwrite; 812 pr->ps_phread = p2->ps_phread - p1->ps_phread; 813 pr->ps_phwrite = p2->ps_phwrite - p1->ps_phwrite; 814 } 815 816 #endif /* _SEQUENT_ */ 817 818 819 #ifdef NEEDmemset 820 /* This is a replacement for a missing memset function */ 821 ptr_t xmemset(loc, value, len) 822 ptr_t loc; 823 int len; 824 size_t value; 825 { 826 char *ptr = (char *) loc; 827 828 while (len--) 829 *ptr++ = value; 830 return loc; 831 } 832 #endif /* NEEDmemset */ 833 834 835 #ifdef NEEDmemmove 836 /* memmove(): 837 * This is the ANSI form of bcopy() with the arguments backwards... 838 * Unlike memcpy(), it handles overlaps between source and 839 * destination memory 840 */ 841 ptr_t 842 xmemmove(vdst, vsrc, len) 843 ptr_t vdst; 844 const ptr_t vsrc; 845 size_t len; 846 { 847 const char *src = (const char *) vsrc; 848 char *dst = (char *) vdst; 849 850 if (src == dst) 851 return vdst; 852 853 if (src > dst) { 854 while (len--) 855 *dst++ = *src++; 856 } 857 else { 858 src += len; 859 dst += len; 860 while (len--) 861 *--dst = *--src; 862 } 863 return vdst; 864 } 865 #endif /* NEEDmemmove */ 866 867 868 #ifndef WINNT 869 #ifdef tcgetpgrp 870 int 871 xtcgetpgrp(fd) 872 int fd; 873 { 874 int pgrp; 875 876 /* ioctl will handle setting errno correctly. */ 877 if (ioctl(fd, TIOCGPGRP, (ioctl_t) & pgrp) < 0) 878 return (-1); 879 return (pgrp); 880 } 881 882 /* 883 * XXX: tcsetpgrp is not a macro any more cause on some systems, 884 * pid_t is a short, but the ioctl() takes a pointer to int (pyr) 885 * Thanks to Simon Day (simon@pharaoh.cyborg.bt.co.uk) for pointing 886 * this out. 887 */ 888 int 889 xtcsetpgrp(fd, pgrp) 890 int fd, pgrp; 891 { 892 return ioctl(fd, TIOCSPGRP, (ioctl_t) &pgrp); 893 } 894 895 #endif /* tcgetpgrp */ 896 #endif /* WINNT */ 897 898 899 #ifdef YPBUGS 900 void 901 fix_yp_bugs() 902 { 903 char *mydomain; 904 905 extern int yp_get_default_domain __P((char **)); 906 /* 907 * PWP: The previous version assumed that yp domain was the same as the 908 * internet name domain. This isn't allways true. (Thanks to Mat Landau 909 * <mlandau@bbn.com> for the original version of this.) 910 */ 911 if (yp_get_default_domain(&mydomain) == 0) { /* if we got a name */ 912 extern void yp_unbind __P((const char *)); 913 914 yp_unbind(mydomain); 915 } 916 } 917 918 #endif /* YPBUGS */ 919 920 #ifdef STRCOLLBUG 921 void 922 fix_strcoll_bug() 923 { 924 #if defined(NLS) && !defined(NOSTRCOLL) 925 /* 926 * SunOS4 checks the file descriptor from openlocale() for <= 0 927 * instead of == -1. Someone should tell sun that file descriptor 0 928 * is valid! Our portable hack: open one so we call it with 0 used... 929 * We have to call this routine every time the locale changes... 930 * 931 * Of course it also tries to free the constant locale "C" it initially 932 * had allocated, with the sequence 933 * > setenv LANG "fr" 934 * > ls^D 935 * > unsetenv LANG 936 * But we are smarter than that and just print a warning message. 937 */ 938 int fd = -1; 939 static char *root = "/"; 940 941 if (!didfds) 942 fd = open(root, O_RDONLY); 943 944 (void) strcoll(root, root); 945 946 if (fd != -1) 947 (void) close(fd); 948 #endif 949 } 950 #endif /* STRCOLLBUG */ 951 952 953 #ifdef OREO 954 #include <compat.h> 955 #endif /* OREO */ 956 957 void 958 osinit() 959 { 960 #ifdef OREO 961 set42sig(); 962 setcompat(getcompat() & ~COMPAT_EXEC); 963 sigignore(SIGIO); /* ignore SIGIO */ 964 #endif /* OREO */ 965 966 #ifdef aiws 967 { 968 struct sigstack inst; 969 inst.ss_sp = (char *) xmalloc((size_t) 4192) + 4192; 970 inst.ss_onstack = 0; 971 sigstack(&inst, NULL); 972 } 973 #endif /* aiws */ 974 975 #ifdef apollo 976 (void) isapad(); 977 #endif 978 979 #ifdef _SX 980 /* 981 * kill(SIGCONT) problems, don't know what this syscall does 982 * [schott@rzg.mpg.de] 983 */ 984 syscall(151, getpid(), getpid()); 985 #endif /* _SX */ 986 } 987 988 #ifdef strerror 989 char * 990 xstrerror(i) 991 int i; 992 { 993 static char errbuf[128]; 994 995 if (i >= 0 && i < sys_nerr) { 996 return sys_errlist[i]; 997 } else { 998 (void) xsnprintf(errbuf, sizeof(errbuf), 999 CGETS(23, 13, "Unknown Error: %d"), i); 1000 return errbuf; 1001 } 1002 } 1003 #endif /* strerror */ 1004 1005 #ifdef gethostname 1006 # if !defined(_MINIX) && !defined(__EMX__) && !defined(WINNT) 1007 # include <sys/utsname.h> 1008 # endif /* !_MINIX && !__EMX__ && !WINNT */ 1009 1010 int 1011 xgethostname(name, namlen) 1012 char *name; 1013 int namlen; 1014 { 1015 # if !defined(_MINIX) && !defined(__EMX__) && !defined(WINNT) 1016 int i, retval; 1017 struct utsname uts; 1018 1019 retval = uname(&uts); 1020 1021 # ifdef DEBUG 1022 xprintf(CGETS(23, 14, "sysname: %s\n"), uts.sysname); 1023 xprintf(CGETS(23, 15, "nodename: %s\n"), uts.nodename); 1024 xprintf(CGETS(23, 16, "release: %s\n"), uts.release); 1025 xprintf(CGETS(23, 17, "version: %s\n"), uts.version); 1026 xprintf(CGETS(23, 18, "machine: %s\n"), uts.machine); 1027 # endif /* DEBUG */ 1028 i = strlen(uts.nodename) + 1; 1029 (void) strncpy(name, uts.nodename, i < namlen ? i : namlen); 1030 1031 return retval; 1032 # else /* !_MINIX && !__EMX__ */ 1033 if (namlen > 0) { 1034 # ifdef __EMX__ 1035 (void) strncpy(name, "OS/2", namlen); 1036 # else /* _MINIX */ 1037 (void) strncpy(name, "minix", namlen); 1038 # endif /* __EMX__ */ 1039 name[namlen-1] = '\0'; 1040 } 1041 return(0); 1042 #endif /* _MINIX && !__EMX__ */ 1043 } /* end xgethostname */ 1044 #endif /* gethostname */ 1045 1046 #ifdef nice 1047 # if defined(_MINIX) && defined(NICE) 1048 # undef _POSIX_SOURCE /* redefined in <lib.h> */ 1049 # undef _MINIX /* redefined in <lib.h> */ 1050 # undef HZ /* redefined in <minix/const.h> */ 1051 # include <lib.h> 1052 # endif /* _MINIX && NICE */ 1053 int 1054 xnice(incr) 1055 int incr; 1056 { 1057 #if defined(_MINIX) && defined(NICE) 1058 return callm1(MM, NICE, incr, 0, 0, NIL_PTR, NIL_PTR, NIL_PTR); 1059 #else 1060 return /* incr ? 0 : */ 0; 1061 #endif /* _MINIX && NICE */ 1062 } /* end xnice */ 1063 #endif /* nice */ 1064 1065 #ifdef NEEDgetcwd 1066 static char *strnrcpy __P((char *, char *, size_t)); 1067 1068 /* xgetcwd(): 1069 * Return the pathname of the current directory, or return 1070 * an error message in pathname. 1071 */ 1072 1073 # ifdef hp9000s500 1074 /* 1075 * From: Bernd Mohr <mohr@faui77.informatik.uni-erlangen.de> 1076 * I also ported the tcsh to the HP9000 Series 500. This computer 1077 * is a little bit different than the other HP 9000 computer. It has 1078 * a HP Chip instead of a Motorola CPU and it is no "real" UNIX. It runs 1079 * HP-UX which is emulated in top of a HP operating system. So, the last 1080 * supported version of HP-UX is 5.2 on the HP9000s500. This has two 1081 * consequences: it supports no job control and it has a filesystem 1082 * without "." and ".." !!! 1083 */ 1084 char * 1085 xgetcwd(pathname, pathlen) 1086 char *pathname; 1087 size_t pathlen; 1088 { 1089 char pathbuf[MAXNAMLEN]; /* temporary pathname buffer */ 1090 char *pnptr = &pathbuf[(sizeof pathbuf)-1]; /* pathname pointer */ 1091 dev_t rdev; /* root device number */ 1092 DIR *dirp = NULL; /* directory stream */ 1093 ino_t rino; /* root inode number */ 1094 off_t rsize; /* root size */ 1095 struct direct *dir; /* directory entry struct */ 1096 struct stat d, dd; /* file status struct */ 1097 int serrno; 1098 1099 *pnptr = '\0'; 1100 (void) stat("/.", &d); 1101 rdev = d.st_dev; 1102 rino = d.st_ino; 1103 rsize = d.st_size; 1104 for (;;) { 1105 if (stat(".", &d) == -1) { 1106 (void) xsnprintf(pathname, pathlen, CGETS(23, 24, 1107 "getcwd: Cannot stat \".\" (%s)"), strerror(errno)); 1108 goto fail; 1109 } 1110 if (d.st_ino == rino && d.st_dev == rdev && d.st_size == rsize) 1111 break; /* reached root directory */ 1112 if ((dirp = opendir("..")) == NULL) { 1113 (void) xsnprintf(pathname, pathlen, CGETS(23, 19, 1114 "getcwd: Cannot open \"..\" (%s)"), strerror(errno)); 1115 goto fail; 1116 } 1117 if (chdir("..") == -1) { 1118 (void) xsnprintf(pathname, pathlen, CGETS(23, 20, 1119 "getcwd: Cannot chdir to \"..\" (%s)"), strerror(errno)); 1120 goto fail; 1121 } 1122 do { 1123 if ((dir = readdir(dirp)) == NULL) { 1124 (void) xsnprintf(pathname, pathlen, 1125 CGETS(23, 21, "getcwd: Read error in \"..\" (%s)"), 1126 strerror(errno)); 1127 goto fail; 1128 } 1129 if (stat(dir->d_name, &dd) == -1) { 1130 (void) xsnprintf(pathname, pathlen, 1131 CGETS(23, 25, "getcwd: Cannot stat directory \"%s\" (%s)"), 1132 dir->d_name, strerror(errno)); 1133 goto fail; 1134 } 1135 } while (dd.st_ino != d.st_ino || 1136 dd.st_dev != d.st_dev || 1137 dd.st_size != d.st_size); 1138 (void) closedir(dirp); 1139 dirp = NULL; 1140 pnptr = strnrcpy(dirp->d_name, pnptr, pnptr - pathbuf); 1141 pnptr = strnrcpy("/", pnptr, pnptr - pathbuf); 1142 } 1143 1144 if (*pnptr == '\0') /* current dir == root dir */ 1145 (void) strncpy(pathname, "/", pathlen); 1146 else { 1147 (void) strncpy(pathname, pnptr, pathlen); 1148 pathname[pathlen - 1] = '\0'; 1149 if (chdir(pnptr) == -1) { 1150 (void) xsnprintf(pathname, MAXPATHLEN, CGETS(23, 22, 1151 "getcwd: Cannot change back to \".\" (%s)"), 1152 strerror(errno)); 1153 return NULL; 1154 } 1155 } 1156 return pathname; 1157 1158 fail: 1159 serrno = errno; 1160 (void) chdir(strnrcpy(".", pnptr, pnptr - pathbuf)); 1161 errno = serrno; 1162 return NULL; 1163 } 1164 1165 # else /* ! hp9000s500 */ 1166 1167 # if (SYSVREL != 0 && !defined(d_fileno)) || defined(_VMS_POSIX) || defined(WINNT) 1168 # define d_fileno d_ino 1169 # endif 1170 1171 char * 1172 xgetcwd(pathname, pathlen) 1173 char *pathname; 1174 size_t pathlen; 1175 { 1176 DIR *dp; 1177 struct dirent *d; 1178 1179 struct stat st_root, st_cur, st_next, st_dotdot; 1180 char pathbuf[MAXPATHLEN], nextpathbuf[MAXPATHLEN * 2]; 1181 char *pathptr, *nextpathptr, *cur_name_add; 1182 int save_errno = 0; 1183 1184 /* find the inode of root */ 1185 if (stat("/", &st_root) == -1) { 1186 (void) xsnprintf(pathname, pathlen, CGETS(23, 23, 1187 "getcwd: Cannot stat \"/\" (%s)"), 1188 strerror(errno)); 1189 return NULL; 1190 } 1191 pathbuf[MAXPATHLEN - 1] = '\0'; 1192 pathptr = &pathbuf[MAXPATHLEN - 1]; 1193 nextpathbuf[MAXPATHLEN - 1] = '\0'; 1194 cur_name_add = nextpathptr = &nextpathbuf[MAXPATHLEN - 1]; 1195 1196 /* find the inode of the current directory */ 1197 if (lstat(".", &st_cur) == -1) { 1198 (void) xsnprintf(pathname, pathlen, CGETS(23, 24, 1199 "getcwd: Cannot stat \".\" (%s)"), 1200 strerror(errno)); 1201 return NULL; 1202 } 1203 nextpathptr = strnrcpy(nextpathptr, "../", nextpathptr - nextpathbuf); 1204 1205 /* Descend to root */ 1206 for (;;) { 1207 1208 /* look if we found root yet */ 1209 if (st_cur.st_ino == st_root.st_ino && 1210 DEV_DEV_COMPARE(st_cur.st_dev, st_root.st_dev)) { 1211 (void) strncpy(pathname, *pathptr != '/' ? "/" : pathptr, pathlen); 1212 pathname[pathlen - 1] = '\0'; 1213 return pathname; 1214 } 1215 1216 /* open the parent directory */ 1217 if (stat(nextpathptr, &st_dotdot) == -1) { 1218 (void) xsnprintf(pathname, pathlen, CGETS(23, 25, 1219 "getcwd: Cannot stat directory \"%s\" (%s)"), 1220 nextpathptr, strerror(errno)); 1221 return NULL; 1222 } 1223 if ((dp = opendir(nextpathptr)) == NULL) { 1224 (void) xsnprintf(pathname, pathlen, CGETS(23, 26, 1225 "getcwd: Cannot open directory \"%s\" (%s)"), 1226 nextpathptr, strerror(errno)); 1227 return NULL; 1228 } 1229 1230 /* look in the parent for the entry with the same inode */ 1231 if (DEV_DEV_COMPARE(st_dotdot.st_dev, st_cur.st_dev)) { 1232 /* Parent has same device. No need to stat every member */ 1233 for (d = readdir(dp); d != NULL; d = readdir(dp)) { 1234 #ifdef __clipper__ 1235 if (((unsigned long)d->d_fileno & 0xffff) == st_cur.st_ino) 1236 break; 1237 #else 1238 if (d->d_fileno == st_cur.st_ino) 1239 break; 1240 #endif 1241 } 1242 } 1243 else { 1244 /* 1245 * Parent has a different device. This is a mount point so we 1246 * need to stat every member 1247 */ 1248 for (d = readdir(dp); d != NULL; d = readdir(dp)) { 1249 if (ISDOT(d->d_name) || ISDOTDOT(d->d_name)) 1250 continue; 1251 (void)strncpy(cur_name_add, d->d_name, 1252 (size_t) (&nextpathbuf[sizeof(nextpathbuf) - 1] - cur_name_add)); 1253 if (lstat(nextpathptr, &st_next) == -1) { 1254 /* 1255 * We might not be able to stat() some path components 1256 * if we are using afs, but this is not an error as 1257 * long as we find the one we need; we also save the 1258 * first error to report it if we don't finally succeed. 1259 */ 1260 if (save_errno == 0) 1261 save_errno = errno; 1262 continue; 1263 } 1264 /* check if we found it yet */ 1265 if (st_next.st_ino == st_cur.st_ino && 1266 DEV_DEV_COMPARE(st_next.st_dev, st_cur.st_dev)) 1267 break; 1268 } 1269 } 1270 if (d == NULL) { 1271 (void) xsnprintf(pathname, pathlen, CGETS(23, 27, 1272 "getcwd: Cannot find \".\" in \"..\" (%s)"), 1273 strerror(save_errno ? save_errno : ENOENT)); 1274 (void) closedir(dp); 1275 return NULL; 1276 } 1277 else 1278 save_errno = 0; 1279 st_cur = st_dotdot; 1280 pathptr = strnrcpy(pathptr, d->d_name, pathptr - pathbuf); 1281 pathptr = strnrcpy(pathptr, "/", pathptr - pathbuf); 1282 nextpathptr = strnrcpy(nextpathptr, "../", nextpathptr - nextpathbuf); 1283 *cur_name_add = '\0'; 1284 (void) closedir(dp); 1285 } 1286 } /* end getcwd */ 1287 # endif /* hp9000s500 */ 1288 1289 /* strnrcpy(): 1290 * Like strncpy, going backwards and returning the new pointer 1291 */ 1292 static char * 1293 strnrcpy(ptr, str, siz) 1294 register char *ptr, *str; 1295 size_t siz; 1296 { 1297 register int len = strlen(str); 1298 if (siz == 0) 1299 return ptr; 1300 1301 while (len && siz--) 1302 *--ptr = str[--len]; 1303 1304 return (ptr); 1305 } /* end strnrcpy */ 1306 #endif /* getcwd */ 1307 1308 #ifdef apollo 1309 /*** 1310 *** Domain/OS 1311 ***/ 1312 #include <apollo/base.h> 1313 #include <apollo/loader.h> 1314 #include <apollo/error.h> 1315 1316 1317 static char * 1318 apperr(st) 1319 status_$t *st; 1320 { 1321 static char buf[BUFSIZE]; 1322 short e_subl, e_modl, e_codel; 1323 error_$string_t e_sub, e_mod, e_code; 1324 1325 error_$get_text(*st, e_sub, &e_subl, e_mod, &e_modl, e_code, &e_codel); 1326 e_sub[e_subl] = '\0'; 1327 e_code[e_codel] = '\0'; 1328 e_mod[e_modl] = '\0'; 1329 (void) xsnprintf(buf, sizeof(buf), "%s (%s/%s)", e_code, e_sub, e_mod); 1330 1331 return(buf); 1332 } 1333 1334 static int 1335 llib(s) 1336 Char *s; 1337 { 1338 short len = Strlen(s); 1339 status_$t st; 1340 char *t; 1341 1342 loader_$inlib(t = short2str(s), len, &st); 1343 if (st.all != status_$ok) 1344 stderror(ERR_SYSTEM, t, apperr(&st)); 1345 } 1346 1347 /*ARGSUSED*/ 1348 void 1349 doinlib(v, c) 1350 Char **v; 1351 struct command *c; 1352 { 1353 setname(short2str(*v++)); 1354 gflag = 0, tglob(v); 1355 if (gflag) { 1356 v = globall(v); 1357 if (v == 0) 1358 stderror(ERR_NAME | ERR_NOMATCH); 1359 } 1360 else { 1361 v = gargv = saveblk(v); 1362 trim(v); 1363 } 1364 1365 while (v && *v) 1366 llib(*v++); 1367 if (gargv) 1368 blkfree(gargv), gargv = 0; 1369 } 1370 1371 int 1372 getv(v) 1373 Char *v; 1374 { 1375 if (eq(v, STRbsd43)) 1376 return(1); 1377 else if (eq(v, STRsys53)) 1378 return(0); 1379 else 1380 stderror(ERR_NAME | ERR_SYSTEM, short2str(v), 1381 CGETS(23, 28, "Invalid system type")); 1382 /*NOTREACHED*/ 1383 return(0); 1384 } 1385 1386 /*ARGSUSED*/ 1387 void 1388 dover(v, c) 1389 Char **v; 1390 struct command *c; 1391 { 1392 Char *p; 1393 1394 setname(short2str(*v++)); 1395 if (!*v) { 1396 if (!(p = tgetenv(STRSYSTYPE))) 1397 stderror(ERR_NAME | ERR_STRING, 1398 CGETS(23, 29, "System type is not set")); 1399 xprintf("%S\n", p); 1400 } 1401 else { 1402 tsetenv(STRSYSTYPE, getv(*v) ? STRbsd43 : STRsys53); 1403 dohash(NULL, NULL); 1404 } 1405 } 1406 1407 /* 1408 * Many thanks to rees@citi.umich.edu (Jim Rees) and 1409 * mathys@ssdt-tempe.sps.mot.com (Yves Mathys) 1410 * For figuring out how to do this... I could have never done 1411 * it without their help. 1412 */ 1413 typedef short enum { 1414 name_$wdir_type, 1415 name_$ndir_type, 1416 name_$node_dir_type, 1417 } name_$dir_type_t; 1418 1419 /*ARGSUSED*/ 1420 void 1421 dorootnode(v, c) 1422 Char **v; 1423 struct command *c; 1424 { 1425 name_$dir_type_t dirtype = name_$node_dir_type; 1426 uid_$t uid; 1427 status_$t st; 1428 char *name; 1429 short namelen; 1430 1431 setname(short2str(*v++)); 1432 1433 name = short2str(*v); 1434 namelen = strlen(name); 1435 1436 name_$resolve(name, &namelen, &uid, &st); 1437 if (st.all != status_$ok) 1438 stderror(ERR_SYSTEM, name, apperr(&st)); 1439 namelen = 0; 1440 name_$set_diru(&uid, "", &namelen, &dirtype, &st); 1441 if (st.all != status_$ok) 1442 stderror(ERR_SYSTEM, name, apperr(&st)); 1443 dohash(NULL, NULL); 1444 } 1445 1446 int 1447 isapad() 1448 { 1449 static int res = -1; 1450 static status_$t st; 1451 1452 if (res == -1) { 1453 int strm; 1454 if (isatty(0)) 1455 strm = 0; 1456 if (isatty(1)) 1457 strm = 1; 1458 if (isatty(2)) 1459 strm = 2; 1460 else { 1461 res = 0; 1462 st.all = status_$ok; 1463 return(res); 1464 } 1465 res = stream_$isavt(&strm, &st); 1466 res = res ? 1 : 0; 1467 } 1468 else { 1469 if (st.all != status_$ok) 1470 stderror(ERR_SYSTEM, "stream_$isavt", apperr(&st)); 1471 } 1472 return(res); 1473 } 1474 #endif 1475