Home | History | Annotate | Line # | Download | only in ksh
sh.h revision 1.11
      1  1.11     kamil /*	$NetBSD: sh.h,v 1.11 2017/06/22 14:11:27 kamil Exp $	*/
      2   1.2       tls 
      3   1.1       jtc /*
      4   1.1       jtc  * Public Domain Bourne/Korn shell
      5   1.1       jtc  */
      6   1.1       jtc 
      7  1.11     kamil /* $Id: sh.h,v 1.11 2017/06/22 14:11:27 kamil Exp $ */
      8   1.1       jtc 
      9   1.1       jtc #include "config.h"	/* system and option configuration info */
     10   1.1       jtc 
     11   1.1       jtc #ifdef HAVE_PROTOTYPES
     12   1.1       jtc # define	ARGS(args)	args	/* prototype declaration */
     13   1.1       jtc #else
     14   1.1       jtc # define	ARGS(args)	()	/* K&R declaration */
     15   1.1       jtc #endif
     16   1.1       jtc 
     17   1.1       jtc /* Start of common headers */
     18   1.1       jtc 
     19   1.1       jtc #include <stdio.h>
     20   1.1       jtc #include <sys/types.h>
     21   1.1       jtc #include <setjmp.h>
     22   1.1       jtc #ifdef HAVE_STDDEF_H
     23   1.1       jtc # include <stddef.h>
     24   1.1       jtc #endif
     25   1.1       jtc 
     26   1.1       jtc #ifdef HAVE_STDLIB_H
     27   1.1       jtc # include <stdlib.h>
     28   1.1       jtc #else
     29   1.1       jtc /* just a useful subset of what stdlib.h would have */
     30   1.1       jtc extern char * getenv  ARGS((const char *));
     31   1.1       jtc extern void * malloc  ARGS((size_t));
     32   1.3   hubertf extern void * realloc ARGS((void *, size_t));
     33   1.1       jtc extern int    free    ARGS((void *));
     34   1.1       jtc extern int    exit    ARGS((int));
     35   1.1       jtc extern int    rand    ARGS((void));
     36   1.1       jtc extern void   srand   ARGS((unsigned int));
     37   1.1       jtc extern int    atoi    ARGS((const char *));
     38   1.1       jtc #endif /* HAVE_STDLIB_H */
     39   1.1       jtc 
     40   1.1       jtc #ifdef HAVE_UNISTD_H
     41   1.1       jtc # include <unistd.h>
     42   1.1       jtc #else
     43   1.1       jtc /* just a useful subset of what unistd.h would have */
     44   1.1       jtc extern int access ARGS((const char *, int));
     45   1.1       jtc extern int open ARGS((const char *, int, ...));
     46   1.1       jtc extern int creat ARGS((const char *, mode_t));
     47   1.1       jtc extern int read ARGS((int, char *, unsigned));
     48   1.1       jtc extern int write ARGS((int, const char *, unsigned));
     49   1.1       jtc extern off_t lseek ARGS((int, off_t, int));
     50   1.1       jtc extern int close ARGS((int));
     51   1.1       jtc extern int pipe ARGS((int []));
     52   1.1       jtc extern int dup2 ARGS((int, int));
     53   1.1       jtc extern int unlink ARGS((const char *));
     54   1.1       jtc extern int fork ARGS((void));
     55   1.1       jtc extern int execve ARGS((const char *, char * const[], char * const[]));
     56   1.1       jtc extern int chdir ARGS((const char *));
     57   1.1       jtc extern int kill ARGS((pid_t, int));
     58   1.1       jtc extern char *getcwd();	/* no ARGS here - differs on different machines */
     59   1.1       jtc extern int geteuid ARGS((void));
     60   1.1       jtc extern int readlink ARGS((const char *, char *, int));
     61   1.1       jtc extern int getegid ARGS((void));
     62   1.1       jtc extern int getpid ARGS((void));
     63   1.1       jtc extern int getppid ARGS((void));
     64   1.1       jtc extern unsigned int sleep ARGS((unsigned int));
     65   1.1       jtc extern int isatty ARGS((int));
     66   1.1       jtc # ifdef POSIX_PGRP
     67   1.1       jtc extern int getpgrp ARGS((void));
     68   1.1       jtc extern int setpgid ARGS((pid_t, pid_t));
     69   1.1       jtc # endif /* POSIX_PGRP */
     70   1.1       jtc # ifdef BSD_PGRP
     71   1.1       jtc extern int getpgrp ARGS((pid_t));
     72   1.1       jtc extern int setpgrp ARGS((pid_t, pid_t));
     73   1.1       jtc # endif /* BSD_PGRP */
     74   1.1       jtc # ifdef SYSV_PGRP
     75   1.1       jtc extern int getpgrp ARGS((void));
     76   1.1       jtc extern int setpgrp ARGS((void));
     77   1.1       jtc # endif /* SYSV_PGRP */
     78   1.1       jtc #endif /* HAVE_UNISTD_H */
     79   1.1       jtc 
     80   1.1       jtc #ifdef HAVE_STRING_H
     81   1.1       jtc # include <string.h>
     82   1.1       jtc #else
     83   1.1       jtc # include <strings.h>
     84   1.1       jtc # define strchr index
     85   1.1       jtc # define strrchr rindex
     86   1.1       jtc #endif /* HAVE_STRING_H */
     87   1.1       jtc #ifndef HAVE_STRSTR
     88   1.1       jtc char *strstr ARGS((const char *s, const char *p));
     89   1.1       jtc #endif /* HAVE_STRSTR */
     90   1.1       jtc #ifndef HAVE_STRCASECMP
     91   1.1       jtc int strcasecmp ARGS((const char *s1, const char *s2));
     92   1.1       jtc int strncasecmp ARGS((const char *s1, const char *s2, int n));
     93   1.1       jtc #endif /* HAVE_STRCASECMP */
     94   1.1       jtc 
     95   1.1       jtc #ifdef HAVE_MEMORY_H
     96   1.1       jtc # include <memory.h>
     97   1.1       jtc #endif
     98   1.1       jtc #ifndef HAVE_MEMSET
     99   1.1       jtc # define memcpy(d, s, n)	bcopy(s, d, n)
    100   1.1       jtc # define memcmp(s1, s2, n)	bcmp(s1, s2, n)
    101   1.1       jtc void *memset ARGS((void *d, int c, size_t n));
    102   1.1       jtc #endif /* HAVE_MEMSET */
    103   1.1       jtc #ifndef HAVE_MEMMOVE
    104   1.1       jtc # ifdef HAVE_BCOPY
    105   1.1       jtc #  define memmove(d, s, n)	bcopy(s, d, n)
    106   1.1       jtc # else
    107   1.1       jtc void *memmove ARGS((void *d, const void *s, size_t n));
    108   1.1       jtc # endif
    109   1.1       jtc #endif /* HAVE_MEMMOVE */
    110   1.1       jtc 
    111   1.1       jtc #ifdef HAVE_PROTOTYPES
    112   1.1       jtc # include <stdarg.h>
    113   1.1       jtc # define SH_VA_START(va, argn) va_start(va, argn)
    114   1.1       jtc #else
    115   1.1       jtc # include <varargs.h>
    116   1.1       jtc # define SH_VA_START(va, argn) va_start(va)
    117   1.1       jtc #endif /* HAVE_PROTOTYPES */
    118   1.1       jtc 
    119   1.1       jtc #include <errno.h>
    120   1.1       jtc 
    121   1.1       jtc #ifdef HAVE_FCNTL_H
    122   1.1       jtc # include <fcntl.h>
    123   1.1       jtc #else
    124   1.1       jtc # include <sys/file.h>
    125   1.1       jtc #endif /* HAVE_FCNTL_H */
    126   1.1       jtc #ifndef O_ACCMODE
    127   1.1       jtc # define O_ACCMODE	(O_RDONLY|O_WRONLY|O_RDWR)
    128   1.1       jtc #endif /* !O_ACCMODE */
    129   1.1       jtc 
    130   1.1       jtc #ifndef F_OK 	/* access() arguments */
    131   1.1       jtc # define F_OK 0
    132   1.1       jtc # define X_OK 1
    133   1.1       jtc # define W_OK 2
    134   1.1       jtc # define R_OK 4
    135   1.1       jtc #endif /* !F_OK */
    136   1.1       jtc 
    137   1.1       jtc #ifndef SEEK_SET
    138   1.1       jtc # ifdef L_SET
    139   1.1       jtc #  define SEEK_SET L_SET
    140   1.1       jtc #  define SEEK_CUR L_INCR
    141   1.1       jtc #  define SEEK_END L_XTND
    142   1.1       jtc # else /* L_SET */
    143   1.1       jtc #  define SEEK_SET 0
    144   1.1       jtc #  define SEEK_CUR 1
    145   1.1       jtc #  define SEEK_END 2
    146   1.1       jtc # endif /* L_SET */
    147   1.1       jtc #endif /* !SEEK_SET */
    148   1.1       jtc 
    149   1.1       jtc /* Some machines (eg, FreeBSD 1.1.5) define CLK_TCK in limits.h
    150   1.1       jtc  * (ksh_limval.h assumes limits has been included, if available)
    151   1.1       jtc  */
    152   1.1       jtc #ifdef HAVE_LIMITS_H
    153   1.1       jtc # include <limits.h>
    154   1.1       jtc #endif /* HAVE_LIMITS_H */
    155   1.1       jtc 
    156   1.1       jtc #include <signal.h>
    157   1.1       jtc #ifdef	NSIG
    158   1.1       jtc # define SIGNALS	NSIG
    159   1.1       jtc #else
    160   1.1       jtc # ifdef	_MINIX
    161   1.1       jtc #  define SIGNALS	(_NSIG+1) /* _NSIG is # of signals used, excluding 0. */
    162   1.1       jtc # else
    163   1.1       jtc #  ifdef _SIGMAX	/* QNX */
    164   1.1       jtc #   define SIGNALS	_SIGMAX
    165   1.1       jtc #  else /* _SIGMAX */
    166   1.1       jtc #   define SIGNALS	32
    167   1.1       jtc #  endif /* _SIGMAX */
    168   1.1       jtc # endif	/* _MINIX */
    169   1.1       jtc #endif	/* NSIG */
    170   1.1       jtc #ifndef SIGCHLD
    171   1.1       jtc # define SIGCHLD SIGCLD
    172   1.1       jtc #endif
    173   1.1       jtc /* struct sigaction.sa_flags is set to KSH_SA_FLAGS.  Used to ensure
    174   1.1       jtc  * system calls are interrupted
    175   1.1       jtc  */
    176   1.1       jtc #ifdef SA_INTERRUPT
    177   1.1       jtc # define KSH_SA_FLAGS	SA_INTERRUPT
    178   1.1       jtc #else /* SA_INTERRUPT */
    179   1.1       jtc # define KSH_SA_FLAGS	0
    180   1.1       jtc #endif /* SA_INTERRUPT */
    181   1.1       jtc 
    182   1.1       jtc typedef	RETSIGTYPE (*handler_t) ARGS((int));	/* signal handler */
    183   1.1       jtc 
    184   1.1       jtc #ifdef USE_FAKE_SIGACT
    185   1.1       jtc # include "sigact.h"			/* use sjg's fake sigaction() */
    186   1.1       jtc #endif
    187   1.1       jtc 
    188   1.1       jtc #ifdef HAVE_PATHS_H
    189   1.1       jtc # include <paths.h>
    190   1.1       jtc #endif /* HAVE_PATHS_H */
    191   1.1       jtc #ifdef _PATH_DEFPATH
    192   1.1       jtc # define DEFAULT__PATH _PATH_DEFPATH
    193   1.1       jtc #else /* _PATH_DEFPATH */
    194   1.1       jtc # define DEFAULT__PATH DEFAULT_PATH
    195   1.1       jtc #endif /* _PATH_DEFPATH */
    196   1.1       jtc 
    197   1.1       jtc #ifndef offsetof
    198   1.1       jtc # define offsetof(type,id) ((size_t)&((type*)NULL)->id)
    199   1.1       jtc #endif
    200   1.1       jtc 
    201   1.1       jtc #ifndef HAVE_KILLPG
    202   1.1       jtc # define killpg(p, s)	kill(-(p), (s))
    203   1.1       jtc #endif /* !HAVE_KILLPG */
    204   1.1       jtc 
    205   1.1       jtc /* Special cases for execve(2) */
    206  1.11     kamil #ifdef OS2
    207  1.11     kamil extern int ksh_execve(char *cmd, char **args, char **env, int flags);
    208  1.11     kamil #else /* OS2 */
    209   1.1       jtc # if defined(OS_ISC) && defined(_POSIX_SOURCE)
    210   1.1       jtc /* Kludge for ISC 3.2 (and other versions?) so programs will run correctly.  */
    211   1.3   hubertf #  define ksh_execve(p, av, ev, flags) \
    212   1.3   hubertf 				do { \
    213   1.1       jtc 					__setostype(0); \
    214   1.1       jtc 					execve(p, av, ev); \
    215   1.1       jtc 					__setostype(1); \
    216   1.1       jtc 				} while (0)
    217   1.1       jtc # else /* OS_ISC && _POSIX */
    218   1.3   hubertf #  define ksh_execve(p, av, ev, flags)	execve(p, av, ev)
    219   1.1       jtc # endif /* OS_ISC && _POSIX */
    220  1.11     kamil #endif /* OS2 */
    221   1.1       jtc 
    222   1.1       jtc /* this is a hang-over from older versions of the os2 port */
    223   1.1       jtc #define ksh_dupbase(fd, base) fcntl(fd, F_DUPFD, base)
    224   1.1       jtc 
    225   1.1       jtc #ifdef HAVE_SIGSETJMP
    226   1.1       jtc # define ksh_sigsetjmp(env,sm)	sigsetjmp((env), (sm))
    227   1.1       jtc # define ksh_siglongjmp(env,v)	siglongjmp((env), (v))
    228   1.1       jtc # define ksh_jmp_buf		sigjmp_buf
    229   1.1       jtc #else /* HAVE_SIGSETJMP */
    230   1.1       jtc # ifdef HAVE__SETJMP
    231   1.1       jtc #  define ksh_sigsetjmp(env,sm)	_setjmp(env)
    232   1.1       jtc #  define ksh_siglongjmp(env,v)	_longjmp((env), (v))
    233   1.1       jtc # else /* HAVE__SETJMP */
    234   1.1       jtc #  define ksh_sigsetjmp(env,sm)	setjmp(env)
    235   1.1       jtc #  define ksh_siglongjmp(env,v)	longjmp((env), (v))
    236   1.1       jtc # endif /* HAVE__SETJMP */
    237   1.1       jtc # define ksh_jmp_buf		jmp_buf
    238   1.1       jtc #endif /* HAVE_SIGSETJMP */
    239   1.1       jtc 
    240   1.3   hubertf #ifndef HAVE_DUP2
    241   1.3   hubertf extern int dup2 ARGS((int, int));
    242   1.3   hubertf #endif /* !HAVE_DUP2 */
    243   1.3   hubertf 
    244   1.1       jtc /* Find a integer type that is at least 32 bits (or die) - SIZEOF_* defined
    245   1.3   hubertf  * by autoconf (assumes an 8 bit byte, but I'm not concerned).
    246   1.3   hubertf  * NOTE: INT32 may end up being more than 32 bits.
    247   1.1       jtc  */
    248   1.1       jtc #if SIZEOF_INT >= 4
    249   1.1       jtc # define INT32	int
    250   1.1       jtc #else /* SIZEOF_INT */
    251   1.1       jtc # if SIZEOF_LONG >= 4
    252   1.1       jtc #  define INT32	long
    253   1.1       jtc # else /* SIZEOF_LONG */
    254   1.1       jtc    #error cannot find 32 bit type...
    255   1.1       jtc # endif /* SIZEOF_LONG */
    256   1.1       jtc #endif /* SIZEOF_INT */
    257   1.1       jtc 
    258   1.1       jtc /* end of common headers */
    259   1.1       jtc 
    260   1.1       jtc /* Stop gcc and lint from complaining about possibly uninitialized variables */
    261   1.1       jtc #if defined(__GNUC__) || defined(lint)
    262   1.1       jtc # define UNINITIALIZED(var)	var = 0
    263   1.1       jtc #else
    264   1.1       jtc # define UNINITIALIZED(var)	var
    265   1.1       jtc #endif /* GNUC || lint */
    266   1.1       jtc 
    267   1.1       jtc /* some useful #defines */
    268   1.1       jtc #ifdef EXTERN
    269   1.1       jtc # define I__(i) = i
    270   1.1       jtc #else
    271   1.1       jtc # define I__(i)
    272   1.1       jtc # define EXTERN extern
    273   1.1       jtc # define EXTERN_DEFINED
    274   1.1       jtc #endif
    275   1.1       jtc 
    276  1.11     kamil #ifdef OS2
    277  1.11     kamil # define inDOS() (!(_emx_env & 0x200))
    278  1.11     kamil #endif
    279  1.11     kamil 
    280   1.1       jtc #ifndef EXECSHELL
    281   1.1       jtc /* shell to exec scripts (see also $SHELL initialization in main.c) */
    282  1.11     kamil # ifdef OS2
    283  1.11     kamil #  define EXECSHELL	(inDOS() ? "c:\\command.com" : "c:\\os2\\cmd.exe")
    284  1.11     kamil #  define EXECSHELL_STR	(inDOS() ? "COMSPEC" : "OS2_SHELL")
    285  1.11     kamil # else /* OS2 */
    286   1.1       jtc #  define EXECSHELL	"/bin/sh"
    287   1.1       jtc #  define EXECSHELL_STR	"EXECSHELL"
    288  1.11     kamil # endif /* OS2 */
    289   1.1       jtc #endif
    290   1.1       jtc 
    291   1.1       jtc /* ISABSPATH() means path is fully and completely specified,
    292   1.1       jtc  * ISROOTEDPATH() means a .. as the first component is a no-op,
    293   1.1       jtc  * ISRELPATH() means $PWD can be tacked on to get an absolute path.
    294   1.1       jtc  *
    295   1.3   hubertf  * OS		Path		ISABSPATH	ISROOTEDPATH	ISRELPATH
    296   1.3   hubertf  * unix		/foo		yes		yes		no
    297   1.3   hubertf  * unix		foo		no		no		yes
    298   1.3   hubertf  * unix		../foo		no		no		yes
    299  1.11     kamil  * os2+cyg	a:/foo		yes		yes		no
    300  1.11     kamil  * os2+cyg	a:foo		no		no		no
    301  1.11     kamil  * os2+cyg	/foo		no		yes		no
    302  1.11     kamil  * os2+cyg	foo		no		no		yes
    303  1.11     kamil  * os2+cyg	../foo		no		no		yes
    304  1.11     kamil  * cyg 		//foo		yes		yes		no
    305  1.11     kamil  */
    306  1.11     kamil #ifdef OS2
    307  1.11     kamil # define PATHSEP        ';'
    308  1.11     kamil # define DIRSEP         '/'	/* even though \ is native */
    309  1.11     kamil # define DIRSEPSTR      "\\"
    310  1.11     kamil # define ISDIRSEP(c)    ((c) == '\\' || (c) == '/')
    311  1.11     kamil # define ISABSPATH(s)	(((s)[0] && (s)[1] == ':' && ISDIRSEP((s)[2])))
    312  1.11     kamil # define ISROOTEDPATH(s) (ISDIRSEP((s)[0]) || ISABSPATH(s))
    313  1.11     kamil # define ISRELPATH(s)	(!(s)[0] || ((s)[1] != ':' && !ISDIRSEP((s)[0])))
    314  1.11     kamil # define FILECHCONV(c)	(isascii(c) && isupper(c) ? tolower(c) : c)
    315  1.11     kamil # define FILECMP(s1, s2) stricmp(s1, s2)
    316  1.11     kamil # define FILENCMP(s1, s2, n) strnicmp(s1, s2, n)
    317  1.11     kamil extern char *ksh_strchr_dirsep(const char *path);
    318  1.11     kamil extern char *ksh_strrchr_dirsep(const char *path);
    319  1.11     kamil # define chdir          _chdir2
    320  1.11     kamil # define getcwd         _getcwd2
    321  1.11     kamil #else
    322   1.1       jtc # define PATHSEP        ':'
    323   1.1       jtc # define DIRSEP         '/'
    324   1.1       jtc # define DIRSEPSTR      "/"
    325   1.1       jtc # define ISDIRSEP(c)    ((c) == '/')
    326  1.11     kamil #ifdef __CYGWIN__
    327  1.11     kamil #  define ISABSPATH(s) \
    328  1.11     kamil        (((s)[0] && (s)[1] == ':' && ISDIRSEP((s)[2])) || ISDIRSEP((s)[0]))
    329  1.11     kamil #  define ISRELPATH(s) (!(s)[0] || ((s)[1] != ':' && !ISDIRSEP((s)[0])))
    330  1.11     kamil #else /* __CYGWIN__ */
    331   1.1       jtc # define ISABSPATH(s)	ISDIRSEP((s)[0])
    332   1.3   hubertf # define ISRELPATH(s)	(!ISABSPATH(s))
    333  1.11     kamil #endif /* __CYGWIN__ */
    334   1.1       jtc # define ISROOTEDPATH(s) ISABSPATH(s)
    335   1.1       jtc # define FILECHCONV(c)	c
    336   1.1       jtc # define FILECMP(s1, s2) strcmp(s1, s2)
    337   1.1       jtc # define FILENCMP(s1, s2, n) strncmp(s1, s2, n)
    338   1.1       jtc # define ksh_strchr_dirsep(p)   strchr(p, DIRSEP)
    339   1.1       jtc # define ksh_strrchr_dirsep(p)  strrchr(p, DIRSEP)
    340  1.11     kamil #endif
    341   1.1       jtc 
    342   1.1       jtc typedef int bool_t;
    343   1.1       jtc #define	FALSE	0
    344   1.1       jtc #define	TRUE	1
    345   1.1       jtc 
    346   1.1       jtc #define	NELEM(a) (sizeof(a) / sizeof((a)[0]))
    347   1.1       jtc #define	sizeofN(type, n) (sizeof(type) * (n))
    348   1.1       jtc #define	BIT(i)	(1<<(i))	/* define bit in flag */
    349   1.1       jtc 
    350   1.1       jtc /* Table flag type - needs > 16 and < 32 bits */
    351   1.1       jtc typedef INT32 Tflag;
    352   1.1       jtc 
    353   1.6   mycroft #define	NUFILE	32		/* Number of user-accessible files */
    354   1.1       jtc #define	FDBASE	10		/* First file usable by Shell */
    355   1.1       jtc 
    356   1.1       jtc /* you're not going to run setuid shell scripts, are you? */
    357   1.1       jtc #define	eaccess(path, mode)	access(path, mode)
    358   1.1       jtc 
    359   1.1       jtc /* Make MAGIC a char that might be printed to make bugs more obvious, but
    360   1.1       jtc  * not a char that is used often.  Also, can't use the high bit as it causes
    361   1.1       jtc  * portability problems (calling strchr(x, 0x80|'x') is error prone).
    362   1.1       jtc  */
    363   1.6   mycroft #define	MAGIC		(7)	/* prefix for *?[!{,} during expand */
    364   1.1       jtc #define ISMAGIC(c)	((unsigned char)(c) == MAGIC)
    365   1.1       jtc #define	NOT		'!'	/* might use ^ (ie, [!...] vs [^..]) */
    366   1.1       jtc 
    367   1.1       jtc #define	LINE	1024		/* input line size */
    368   1.1       jtc #define	PATH	1024		/* pathname size (todo: PATH_MAX/pathconf()) */
    369   1.1       jtc #define ARRAYMAX 1023		/* max array index */
    370   1.1       jtc 
    371   1.1       jtc EXTERN	const char *kshname;	/* $0 */
    372   1.1       jtc EXTERN	pid_t	kshpid;		/* $$, shell pid */
    373   1.1       jtc EXTERN	pid_t	procpid;	/* pid of executing process */
    374   1.6   mycroft EXTERN	uid_t	ksheuid;	/* effective uid of shell */
    375   1.1       jtc EXTERN	int	exstat;		/* exit status */
    376   1.1       jtc EXTERN	int	subst_exstat;	/* exit status of last $(..)/`..` */
    377   1.1       jtc EXTERN	const char *safe_prompt; /* safe prompt if PS1 substitution fails */
    378   1.1       jtc 
    379   1.1       jtc /*
    380   1.1       jtc  * Area-based allocation built on malloc/free
    381   1.1       jtc  */
    382   1.1       jtc typedef struct Area {
    383   1.6   mycroft 	struct link *freelist;	/* free list */
    384   1.1       jtc } Area;
    385   1.1       jtc 
    386   1.1       jtc EXTERN	Area	aperm;		/* permanent object space */
    387   1.1       jtc #define	APERM	&aperm
    388   1.1       jtc #define	ATEMP	&e->area
    389   1.1       jtc 
    390   1.1       jtc #ifdef MEM_DEBUG
    391   1.1       jtc # include "chmem.h" /* a debugging front end for malloc et. al. */
    392   1.1       jtc #endif /* MEM_DEBUG */
    393   1.1       jtc 
    394   1.3   hubertf #ifdef KSH_DEBUG
    395   1.3   hubertf # define kshdebug_init()	kshdebug_init_()
    396   1.3   hubertf # define kshdebug_printf(a)	kshdebug_printf_ a
    397   1.3   hubertf # define kshdebug_dump(a)	kshdebug_dump_ a
    398   1.3   hubertf #else /* KSH_DEBUG */
    399   1.3   hubertf # define kshdebug_init()
    400   1.3   hubertf # define kshdebug_printf(a)
    401   1.3   hubertf # define kshdebug_dump(a)
    402   1.3   hubertf #endif /* KSH_DEBUG */
    403   1.3   hubertf 
    404   1.1       jtc /*
    405   1.1       jtc  * parsing & execution environment
    406   1.1       jtc  */
    407   1.1       jtc EXTERN	struct env {
    408   1.5       wiz 	short	type;			/* environment type - see below */
    409   1.1       jtc 	short	flags;			/* EF_* */
    410   1.1       jtc 	Area	area;			/* temporary allocation area */
    411   1.1       jtc 	struct	block *loc;		/* local variables and functions */
    412   1.1       jtc 	short  *savefd;			/* original redirected fd's */
    413   1.5       wiz 	struct	env *oenv;		/* link to previous environment */
    414   1.1       jtc 	ksh_jmp_buf jbuf;		/* long jump back to env creator */
    415   1.1       jtc 	struct temp *temps;		/* temp files */
    416   1.1       jtc } *e;
    417   1.1       jtc 
    418   1.1       jtc /* struct env.type values */
    419   1.5       wiz #define	E_NONE	0		/* dummy environment */
    420   1.1       jtc #define	E_PARSE	1		/* parsing command # */
    421   1.1       jtc #define	E_FUNC	2		/* executing function # */
    422   1.1       jtc #define	E_INCL	3		/* including a file via . # */
    423   1.1       jtc #define	E_EXEC	4		/* executing command tree */
    424   1.1       jtc #define	E_LOOP	5		/* executing for/while # */
    425   1.1       jtc #define	E_ERRH	6		/* general error handler # */
    426   1.1       jtc /* # indicates env has valid jbuf (see unwind()) */
    427   1.1       jtc 
    428   1.1       jtc /* struct env.flag values */
    429   1.1       jtc #define EF_FUNC_PARSE	BIT(0)	/* function being parsed */
    430   1.1       jtc #define EF_BRKCONT_PASS	BIT(1)	/* set if E_LOOP must pass break/continue on */
    431   1.3   hubertf #define EF_FAKE_SIGDIE	BIT(2)	/* hack to get info from unwind to quitenv */
    432   1.1       jtc 
    433   1.1       jtc /* Do breaks/continues stop at env type e? */
    434   1.1       jtc #define STOP_BRKCONT(t)	((t) == E_NONE || (t) == E_PARSE \
    435   1.1       jtc 			 || (t) == E_FUNC || (t) == E_INCL)
    436   1.1       jtc /* Do returns stop at env type e? */
    437   1.1       jtc #define STOP_RETURN(t)	((t) == E_FUNC || (t) == E_INCL)
    438   1.1       jtc 
    439   1.1       jtc /* values for ksh_siglongjmp(e->jbuf, 0) */
    440   1.1       jtc #define LRETURN	1		/* return statement */
    441   1.1       jtc #define	LEXIT	2		/* exit statement */
    442   1.1       jtc #define LERROR	3		/* errorf() called */
    443   1.1       jtc #define LLEAVE	4		/* untrappable exit/error */
    444   1.1       jtc #define LINTR	5		/* ^C noticed */
    445   1.1       jtc #define	LBREAK	6		/* break statement */
    446   1.1       jtc #define	LCONTIN	7		/* continue statement */
    447   1.1       jtc #define LSHELL	8		/* return to interactive shell() */
    448   1.1       jtc #define LAEXPR	9		/* error in arithmetic expression */
    449   1.1       jtc 
    450   1.1       jtc /* option processing */
    451   1.1       jtc #define OF_CMDLINE	0x01	/* command line */
    452   1.1       jtc #define OF_SET		0x02	/* set builtin */
    453   1.1       jtc #define OF_SPECIAL	0x04	/* a special variable changing */
    454   1.3   hubertf #define OF_INTERNAL	0x08	/* set internally by shell */
    455   1.3   hubertf #define OF_ANY		(OF_CMDLINE | OF_SET | OF_SPECIAL | OF_INTERNAL)
    456   1.1       jtc 
    457   1.1       jtc struct option {
    458   1.1       jtc     const char	*name;	/* long name of option */
    459   1.1       jtc     char	c;	/* character flag (if any) */
    460   1.1       jtc     short	flags;	/* OF_* */
    461   1.1       jtc };
    462   1.7  christos extern const struct option goptions[];
    463   1.1       jtc 
    464   1.1       jtc /*
    465   1.1       jtc  * flags (the order of these enums MUST match the order in misc.c(options[]))
    466   1.1       jtc  */
    467   1.1       jtc enum sh_flag {
    468   1.1       jtc 	FEXPORT = 0,	/* -a: export all */
    469   1.1       jtc #ifdef BRACE_EXPAND
    470   1.1       jtc 	FBRACEEXPAND,	/* enable {} globbing */
    471   1.1       jtc #endif
    472   1.1       jtc 	FBGNICE,	/* bgnice */
    473   1.1       jtc 	FCOMMAND,	/* -c: (invocation) execute specified command */
    474   1.1       jtc #ifdef EMACS
    475   1.1       jtc 	FEMACS,		/* emacs command editing */
    476   1.6   mycroft 	FEMACSUSEMETA,	/* use 8th bit as meta */
    477   1.1       jtc #endif
    478   1.1       jtc 	FERREXIT,	/* -e: quit on error */
    479   1.1       jtc #ifdef EMACS
    480   1.1       jtc 	FGMACS,		/* gmacs command editing */
    481   1.1       jtc #endif
    482   1.1       jtc 	FIGNOREEOF,	/* eof does not exit */
    483   1.1       jtc 	FTALKING,	/* -i: interactive */
    484   1.6   mycroft 	FKEYWORD,	/* -k: name=value anywhere */
    485   1.1       jtc 	FLOGIN,		/* -l: a login shell */
    486   1.1       jtc 	FMARKDIRS,	/* mark dirs with / in file name completion */
    487   1.1       jtc 	FMONITOR,	/* -m: job control monitoring */
    488   1.1       jtc 	FNOCLOBBER,	/* -C: don't overwrite existing files */
    489   1.1       jtc 	FNOEXEC,	/* -n: don't execute any commands */
    490   1.1       jtc 	FNOGLOB,	/* -f: don't do file globbing */
    491   1.1       jtc 	FNOHUP,		/* -H: don't kill running jobs when login shell exits */
    492   1.1       jtc 	FNOLOG,		/* don't save functions in history (ignored) */
    493   1.1       jtc #ifdef	JOBS
    494   1.1       jtc 	FNOTIFY,	/* -b: asynchronous job completion notification */
    495   1.1       jtc #endif
    496   1.1       jtc 	FNOUNSET,	/* -u: using an unset var is an error */
    497   1.1       jtc 	FPHYSICAL,	/* -o physical: don't do logical cd's/pwd's */
    498   1.1       jtc 	FPOSIX,		/* -o posix: be posixly correct */
    499   1.1       jtc 	FPRIVILEGED,	/* -p: use suid_profile */
    500   1.1       jtc 	FRESTRICTED,	/* -r: restricted shell */
    501   1.1       jtc 	FSTDIN,		/* -s: (invocation) parse stdin */
    502   1.1       jtc 	FTRACKALL,	/* -h: create tracked aliases for all commands */
    503   1.1       jtc 	FVERBOSE,	/* -v: echo input */
    504   1.1       jtc #ifdef VI
    505   1.1       jtc 	FVI,		/* vi command editing */
    506   1.1       jtc 	FVIRAW,		/* always read in raw mode (ignored) */
    507   1.1       jtc 	FVISHOW8,	/* display chars with 8th bit set as is (versus M-) */
    508   1.1       jtc 	FVITABCOMPLETE,	/* enable tab as file name completion char */
    509   1.1       jtc 	FVIESCCOMPLETE,	/* enable ESC as file name completion in command mode */
    510   1.1       jtc #endif
    511   1.1       jtc 	FXTRACE,	/* -x: execution trace */
    512   1.3   hubertf 	FTALKING_I,	/* (internal): initial shell was interactive */
    513   1.1       jtc 	FNFLAGS /* (place holder: how many flags are there) */
    514   1.1       jtc };
    515   1.1       jtc 
    516   1.1       jtc #define Flag(f)	(shell_flags[(int) (f)])
    517   1.1       jtc 
    518   1.1       jtc EXTERN	char shell_flags [FNFLAGS];
    519   1.1       jtc 
    520   1.1       jtc EXTERN	char	null [] I__("");	/* null value for variable */
    521   1.1       jtc EXTERN	char	space [] I__(" ");
    522   1.1       jtc EXTERN	char	newline [] I__("\n");
    523   1.1       jtc EXTERN	char	slash [] I__("/");
    524   1.1       jtc 
    525   1.3   hubertf enum temp_type {
    526   1.3   hubertf     TT_HEREDOC_EXP,	/* expanded heredoc */
    527   1.3   hubertf     TT_HIST_EDIT	/* temp file used for history editing (fc -e) */
    528   1.3   hubertf };
    529   1.3   hubertf typedef enum temp_type Temp_type;
    530   1.3   hubertf /* temp/heredoc files.  The file is removed when the struct is freed. */
    531   1.1       jtc struct temp {
    532   1.1       jtc 	struct temp	*next;
    533   1.1       jtc 	struct shf	*shf;
    534   1.1       jtc 	int		pid;		/* pid of process parsed here-doc */
    535   1.3   hubertf 	Temp_type	type;
    536   1.1       jtc 	char		*name;
    537   1.1       jtc };
    538   1.1       jtc 
    539   1.1       jtc /*
    540   1.1       jtc  * stdio and our IO routines
    541   1.1       jtc  */
    542   1.1       jtc 
    543   1.1       jtc #define shl_spare	(&shf_iob[0])	/* for c_read()/c_print() */
    544   1.1       jtc #define shl_stdout	(&shf_iob[1])
    545   1.1       jtc #define shl_out		(&shf_iob[2])
    546   1.1       jtc EXTERN int shl_stdout_ok;
    547   1.1       jtc 
    548   1.1       jtc /*
    549   1.1       jtc  * trap handlers
    550   1.1       jtc  */
    551   1.1       jtc typedef struct trap {
    552   1.1       jtc 	int	signal;		/* signal number */
    553   1.1       jtc 	const char *name;	/* short name */
    554   1.1       jtc 	const char *mess;	/* descriptive name */
    555   1.1       jtc 	char   *trap;		/* trap command */
    556   1.1       jtc 	int	volatile set;	/* trap pending */
    557   1.1       jtc 	int	flags;		/* TF_* */
    558   1.1       jtc 	handler_t cursig;	/* current handler (valid if TF_ORIG_* set) */
    559   1.1       jtc 	handler_t shtrap;	/* shell signal handler */
    560   1.1       jtc } Trap;
    561   1.1       jtc 
    562   1.1       jtc /* values for Trap.flags */
    563   1.1       jtc #define TF_SHELL_USES	BIT(0)	/* shell uses signal, user can't change */
    564   1.1       jtc #define TF_USER_SET	BIT(1)	/* user has (tried to) set trap */
    565   1.1       jtc #define TF_ORIG_IGN	BIT(2)	/* original action was SIG_IGN */
    566   1.1       jtc #define TF_ORIG_DFL	BIT(3)	/* original action was SIG_DFL */
    567   1.1       jtc #define TF_EXEC_IGN	BIT(4)	/* restore SIG_IGN just before exec */
    568   1.1       jtc #define TF_EXEC_DFL	BIT(5)	/* restore SIG_DFL just before exec */
    569   1.1       jtc #define TF_DFL_INTR	BIT(6)	/* when received, default action is LINTR */
    570   1.1       jtc #define TF_TTY_INTR	BIT(7)	/* tty generated signal (see j_waitj) */
    571   1.1       jtc #define TF_CHANGED	BIT(8)	/* used by runtrap() to detect trap changes */
    572   1.1       jtc #define TF_FATAL	BIT(9)	/* causes termination if not trapped */
    573   1.1       jtc 
    574   1.1       jtc /* values for setsig()/setexecsig() flags argument */
    575   1.1       jtc #define SS_RESTORE_MASK	0x3	/* how to restore a signal before an exec() */
    576   1.1       jtc #define SS_RESTORE_CURR	0	/* leave current handler in place */
    577   1.1       jtc #define SS_RESTORE_ORIG	1	/* restore original handler */
    578   1.1       jtc #define SS_RESTORE_DFL	2	/* restore to SIG_DFL */
    579   1.1       jtc #define SS_RESTORE_IGN	3	/* restore to SIG_IGN */
    580   1.1       jtc #define SS_FORCE	BIT(3)	/* set signal even if original signal ignored */
    581   1.1       jtc #define SS_USER		BIT(4)	/* user is doing the set (ie, trap command) */
    582   1.1       jtc #define SS_SHTRAP	BIT(5)	/* trap for internal use (CHLD,ALRM,WINCH) */
    583   1.1       jtc 
    584   1.1       jtc #define SIGEXIT_	0	/* for trap EXIT */
    585   1.1       jtc #define SIGERR_		SIGNALS	/* for trap ERR */
    586   1.1       jtc 
    587   1.1       jtc EXTERN	int volatile trap;	/* traps pending? */
    588   1.1       jtc EXTERN	int volatile intrsig;	/* pending trap interrupts executing command */
    589   1.1       jtc EXTERN	int volatile fatal_trap;/* received a fatal signal */
    590  1.11     kamil #ifndef FROM_TRAP_C
    591  1.11     kamil /* Kludge to avoid bogus re-declaration of sigtraps[] error on AIX 3.2.5 */
    592   1.1       jtc extern	Trap	sigtraps[SIGNALS+1];
    593  1.11     kamil #endif /* !FROM_TRAP_C */
    594   1.1       jtc 
    595   1.1       jtc #ifdef KSH
    596   1.1       jtc /*
    597   1.1       jtc  * TMOUT support
    598   1.1       jtc  */
    599   1.1       jtc /* values for ksh_tmout_state */
    600   1.1       jtc enum tmout_enum {
    601   1.1       jtc 		TMOUT_EXECUTING	= 0,	/* executing commands */
    602   1.1       jtc 		TMOUT_READING,		/* waiting for input */
    603   1.1       jtc 		TMOUT_LEAVING		/* have timed out */
    604   1.1       jtc 	};
    605   1.1       jtc EXTERN unsigned int ksh_tmout;
    606   1.1       jtc EXTERN enum tmout_enum ksh_tmout_state I__(TMOUT_EXECUTING);
    607   1.1       jtc #endif /* KSH */
    608   1.1       jtc 
    609   1.1       jtc /* For "You have stopped jobs" message */
    610   1.1       jtc EXTERN int really_exit;
    611   1.1       jtc 
    612   1.1       jtc /*
    613   1.1       jtc  * fast character classes
    614   1.1       jtc  */
    615   1.1       jtc #define	C_ALPHA	 BIT(0)		/* a-z_A-Z */
    616   1.1       jtc #define	C_DIGIT	 BIT(1)		/* 0-9 */
    617   1.1       jtc #define	C_LEX1	 BIT(2)		/* \0 \t\n|&;<>() */
    618   1.1       jtc #define	C_VAR1	 BIT(3)		/* *@#!$-? */
    619   1.1       jtc #define	C_IFSWS	 BIT(4)		/* \t \n (IFS white space) */
    620   1.1       jtc #define	C_SUBOP1 BIT(5)		/* "=-+?" */
    621   1.1       jtc #define	C_SUBOP2 BIT(6)		/* "#%" */
    622   1.1       jtc #define	C_IFS	 BIT(7)		/* $IFS */
    623   1.1       jtc #define	C_QUOTE	 BIT(8)		/*  \n\t"#$&'()*;<>?[\`| (needing quoting) */
    624   1.1       jtc 
    625   1.1       jtc extern	short ctypes [];
    626   1.1       jtc 
    627   1.1       jtc #define	ctype(c, t)	!!(ctypes[(unsigned char)(c)]&(t))
    628   1.1       jtc #define	letter(c)	ctype(c, C_ALPHA)
    629   1.1       jtc #define	digit(c)	ctype(c, C_DIGIT)
    630   1.1       jtc #define	letnum(c)	ctype(c, C_ALPHA|C_DIGIT)
    631   1.1       jtc 
    632   1.1       jtc EXTERN int ifs0 I__(' ');	/* for "$*" */
    633   1.1       jtc 
    634   1.1       jtc /* Argument parsing for built-in commands and getopts command */
    635   1.1       jtc 
    636   1.1       jtc /* Values for Getopt.flags */
    637   1.1       jtc #define GF_ERROR	BIT(0)	/* call errorf() if there is an error */
    638   1.1       jtc #define GF_PLUSOPT	BIT(1)	/* allow +c as an option */
    639   1.1       jtc #define GF_NONAME	BIT(2)	/* don't print argv[0] in errors */
    640   1.1       jtc 
    641   1.1       jtc /* Values for Getopt.info */
    642   1.1       jtc #define GI_MINUS	BIT(0)	/* an option started with -... */
    643   1.1       jtc #define GI_PLUS		BIT(1)	/* an option started with +... */
    644   1.1       jtc #define GI_MINUSMINUS	BIT(2)	/* arguments were ended with -- */
    645   1.1       jtc 
    646   1.1       jtc typedef struct {
    647   1.1       jtc 	int		optind;
    648   1.3   hubertf 	int		uoptind;/* what user sees in $OPTIND */
    649   1.1       jtc 	char		*optarg;
    650   1.1       jtc 	int		flags;	/* see GF_* */
    651   1.1       jtc 	int		info;	/* see GI_* */
    652   1.1       jtc 	unsigned int	p;	/* 0 or index into argv[optind - 1] */
    653   1.1       jtc 	char		buf[2];	/* for bad option OPTARG value */
    654   1.1       jtc } Getopt;
    655   1.1       jtc 
    656   1.1       jtc EXTERN Getopt builtin_opt;	/* for shell builtin commands */
    657   1.3   hubertf EXTERN Getopt user_opt;		/* parsing state for getopts builtin command */
    658   1.1       jtc 
    659   1.1       jtc #ifdef KSH
    660   1.1       jtc /* This for co-processes */
    661   1.1       jtc 
    662   1.1       jtc typedef INT32 Coproc_id; /* something that won't (realisticly) wrap */
    663   1.1       jtc struct coproc {
    664   1.1       jtc 	int	read;		/* pipe from co-process's stdout */
    665   1.1       jtc 	int	readw;		/* other side of read (saved temporarily) */
    666   1.1       jtc 	int	write;		/* pipe to co-process's stdin */
    667   1.1       jtc 	Coproc_id id;		/* id of current output pipe */
    668   1.1       jtc 	int	njobs;		/* number of live jobs using output pipe */
    669   1.1       jtc 	void    *job;           /* 0 or job of co-process using input pipe */
    670   1.1       jtc };
    671   1.1       jtc EXTERN struct coproc coproc;
    672   1.1       jtc #endif /* KSH */
    673   1.1       jtc 
    674   1.1       jtc /* Used in jobs.c and by coprocess stuff in exec.c */
    675   1.1       jtc #ifdef JOB_SIGS
    676   1.1       jtc EXTERN sigset_t		sm_default, sm_sigchld;
    677   1.1       jtc #endif /* JOB_SIGS */
    678   1.1       jtc 
    679   1.7  christos extern char ksh_version[];
    680   1.1       jtc 
    681   1.1       jtc /* name of called builtin function (used by error functions) */
    682   1.1       jtc EXTERN char	*builtin_argv0;
    683   1.1       jtc EXTERN Tflag	builtin_flag;	/* flags of called builtin (SPEC_BI, etc.) */
    684   1.1       jtc 
    685   1.1       jtc /* current working directory, and size of memory allocated for same */
    686   1.1       jtc EXTERN char	*current_wd;
    687   1.1       jtc EXTERN int	current_wd_size;
    688   1.1       jtc 
    689   1.1       jtc #ifdef EDIT
    690   1.6   mycroft /* Minimum required space to work with on a line - if the prompt leaves less
    691   1.1       jtc  * space than this on a line, the prompt is truncated.
    692   1.1       jtc  */
    693   1.1       jtc # define MIN_EDIT_SPACE	7
    694   1.6   mycroft /* Minimum allowed value for x_cols: 2 for prompt, 3 for " < " at end of line
    695   1.1       jtc  */
    696   1.1       jtc # define MIN_COLS	(2 + MIN_EDIT_SPACE + 3)
    697   1.1       jtc EXTERN	int	x_cols I__(80);	/* tty columns */
    698   1.1       jtc #else
    699   1.1       jtc # define x_cols 80		/* for pr_menu(exec.c) */
    700   1.1       jtc #endif
    701   1.1       jtc 
    702   1.1       jtc /* These to avoid bracket matching problems */
    703   1.1       jtc #define OPAREN	'('
    704   1.1       jtc #define CPAREN	')'
    705   1.1       jtc #define OBRACK	'['
    706   1.1       jtc #define CBRACK	']'
    707   1.1       jtc #define OBRACE	'{'
    708   1.1       jtc #define CBRACE	'}'
    709   1.1       jtc 
    710   1.2       tls /* Determine the location of the system (common) profile */
    711   1.2       tls #ifndef KSH_SYSTEM_PROFILE
    712   1.2       tls # ifdef __NeXT
    713   1.2       tls #  define KSH_SYSTEM_PROFILE "/etc/profile.std"
    714   1.2       tls # else /* __NeXT */
    715   1.2       tls #  define KSH_SYSTEM_PROFILE "/etc/profile"
    716   1.2       tls # endif /* __NeXT */
    717   1.2       tls #endif /* KSH_SYSTEM_PROFILE */
    718   1.3   hubertf 
    719   1.3   hubertf /* Used by v_evaluate() and setstr() to control action when error occurs */
    720   1.3   hubertf #define KSH_UNWIND_ERROR	0	/* unwind the stack (longjmp) */
    721   1.3   hubertf #define KSH_RETURN_ERROR	1	/* return 1/0 for success/failure */
    722   1.1       jtc 
    723   1.1       jtc #include "shf.h"
    724   1.1       jtc #include "table.h"
    725   1.1       jtc #include "tree.h"
    726   1.1       jtc #include "expand.h"
    727   1.1       jtc #include "lex.h"
    728   1.1       jtc #include "proto.h"
    729   1.1       jtc 
    730   1.1       jtc /* be sure not to interfere with anyone else's idea about EXTERN */
    731   1.1       jtc #ifdef EXTERN_DEFINED
    732   1.1       jtc # undef EXTERN_DEFINED
    733   1.1       jtc # undef EXTERN
    734   1.1       jtc #endif
    735   1.1       jtc #undef I__
    736