1 /* $RCSfile: doarg.c,v $$Revision: 4.0.1.1 $$Date: 91/04/11 17:40:14 $
3 * Copyright (c) 1989, Larry Wall
5 * You may distribute under the terms of the GNU General Public License
6 * as specified in the README file that comes with the perl 3.0 kit.
9 * Revision 4.0.1.1 91/04/11 17:40:14 lwall
10 * patch1: fixed undefined environ problem
11 * patch1: fixed debugger coredump on subroutines
13 * Revision 4.0 91/03/20 01:06:42 lwall
21 #if !defined(NSIG) || defined(M_UNIX) || defined(M_XENIX)
25 extern unsigned char fold[];
28 #pragma function(memcmp)
29 #endif /* BUGGY_MSC */
40 register char *s = str_get(str);
41 char *strend = s + str->str_cur;
47 int maxiters = (strend - s) + 10;
53 rspat = spat = arg[2].arg_ptr.arg_spat;
55 fatal("panic: do_subst");
56 else if (spat->spat_runtime) {
58 (void)eval(spat->spat_runtime,G_SCALAR,sp);
59 m = str_get(dstr = stack->ary_array[sp+1]);
61 if (spat->spat_regexp) {
62 regfree(spat->spat_regexp);
63 spat->spat_regexp = Null(REGEXP*); /* required if regcomp pukes */
65 spat->spat_regexp = regcomp(m,m+dstr->str_cur,
66 spat->spat_flags & SPAT_FOLD);
67 if (spat->spat_flags & SPAT_KEEP) {
68 arg_free(spat->spat_runtime); /* it won't change, so */
69 spat->spat_runtime = Nullarg; /* no point compiling again */
74 deb("2.SPAT /%s/\n",spat->spat_regexp->precomp);
77 safebase = ((!spat->spat_regexp || !spat->spat_regexp->nparens) &&
79 if (!*spat->spat_regexp->precomp && lastspat)
83 if (hint < s || hint > strend)
84 fatal("panic: hint in do_match");
87 if (spat->spat_regexp->regback >= 0) {
88 s -= spat->spat_regexp->regback;
95 else if (spat->spat_short) {
96 if (spat->spat_flags & SPAT_SCANFIRST) {
97 if (str->str_pok & SP_STUDIED) {
98 if (screamfirst[spat->spat_short->str_rare] < 0)
100 else if (!(s = screaminstr(str,spat->spat_short)))
104 else if (!(s = fbminstr((unsigned char*)s, (unsigned char*)strend,
108 if (s && spat->spat_regexp->regback >= 0) {
109 ++spat->spat_short->str_u.str_useful;
110 s -= spat->spat_regexp->regback;
117 else if (!multiline && (*spat->spat_short->str_ptr != *s ||
118 bcmp(spat->spat_short->str_ptr, s, spat->spat_slen) ))
120 if (--spat->spat_short->str_u.str_useful < 0) {
121 str_free(spat->spat_short);
122 spat->spat_short = Nullstr; /* opt is being useless */
125 once = ((rspat->spat_flags & SPAT_ONCE) != 0);
126 if (rspat->spat_flags & SPAT_CONST) { /* known replacement string? */
127 if ((rspat->spat_repl[1].arg_type & A_MASK) == A_SINGLE)
128 dstr = rspat->spat_repl[1].arg_ptr.arg_str;
129 else { /* constant over loop, anyway */
130 (void)eval(rspat->spat_repl,G_SCALAR,sp);
131 dstr = stack->ary_array[sp+1];
134 clen = dstr->str_cur;
135 if (clen <= spat->spat_slen + (int)spat->spat_regexp->regback) {
136 /* can do inplace substitution */
137 if (regexec(spat->spat_regexp, s, strend, orig, 0,
138 str->str_pok & SP_STUDIED ? str : Nullstr, safebase)) {
139 if (spat->spat_regexp->subbase) /* oops, no we can't */
143 str->str_pok = SP_VALID; /* disable possible screamer */
145 m = spat->spat_regexp->startp[0];
146 d = spat->spat_regexp->endp[0];
148 if (m - s > strend - d) { /* faster to shorten from end */
150 (void)bcopy(c, m, clen);
155 (void)bcopy(d, m, i);
159 str->str_cur = m - s;
161 str_numset(arg->arg_ptr.arg_str, 1.0);
162 stack->ary_array[++sp] = arg->arg_ptr.arg_str;
165 else if (i = m - s) { /* faster from front */
173 (void)bcopy(c, m, clen);
175 str_numset(arg->arg_ptr.arg_str, 1.0);
176 stack->ary_array[++sp] = arg->arg_ptr.arg_str;
182 (void)bcopy(c,d,clen);
184 str_numset(arg->arg_ptr.arg_str, 1.0);
185 stack->ary_array[++sp] = arg->arg_ptr.arg_str;
191 str_numset(arg->arg_ptr.arg_str, 1.0);
192 stack->ary_array[++sp] = arg->arg_ptr.arg_str;
198 if (iters++ > maxiters)
199 fatal("Substitution loop");
200 m = spat->spat_regexp->startp[0];
207 (void)bcopy(c,d,clen);
210 s = spat->spat_regexp->endp[0];
211 } while (regexec(spat->spat_regexp, s, strend, orig, s == m,
212 Nullstr, TRUE)); /* (don't match same null twice) */
215 str->str_cur = d - str->str_ptr + i;
216 (void)bcopy(s,d,i+1); /* include the Null */
219 str_numset(arg->arg_ptr.arg_str, (double)iters);
220 stack->ary_array[++sp] = arg->arg_ptr.arg_str;
223 str_numset(arg->arg_ptr.arg_str, 0.0);
224 stack->ary_array[++sp] = arg->arg_ptr.arg_str;
230 if (regexec(spat->spat_regexp, s, strend, orig, 0,
231 str->str_pok & SP_STUDIED ? str : Nullstr, safebase)) {
233 dstr = Str_new(25,str_len(str));
234 str_nset(dstr,m,s-m);
235 if (spat->spat_regexp->subbase)
239 if (iters++ > maxiters)
240 fatal("Substitution loop");
241 if (spat->spat_regexp->subbase
242 && spat->spat_regexp->subbase != orig) {
245 orig = spat->spat_regexp->subbase;
247 strend = s + (strend - m);
249 m = spat->spat_regexp->startp[0];
250 str_ncat(dstr,s,m-s);
251 s = spat->spat_regexp->endp[0];
254 str_ncat(dstr,c,clen);
257 char *mysubbase = spat->spat_regexp->subbase;
259 spat->spat_regexp->subbase = Nullch; /* so recursion works */
260 (void)eval(rspat->spat_repl,G_SCALAR,sp);
261 str_scat(dstr,stack->ary_array[sp+1]);
262 if (spat->spat_regexp->subbase)
263 Safefree(spat->spat_regexp->subbase);
264 spat->spat_regexp->subbase = mysubbase;
268 } while (regexec(spat->spat_regexp, s, strend, orig, s == m, Nullstr,
270 str_ncat(dstr,s,strend - s);
271 str_replace(str,dstr);
273 str_numset(arg->arg_ptr.arg_str, (double)iters);
274 stack->ary_array[++sp] = arg->arg_ptr.arg_str;
277 str_numset(arg->arg_ptr.arg_str, 0.0);
278 stack->ary_array[++sp] = arg->arg_ptr.arg_str;
282 ++spat->spat_short->str_u.str_useful;
283 str_numset(arg->arg_ptr.arg_str, 0.0);
284 stack->ary_array[++sp] = arg->arg_ptr.arg_str;
288 #pragma intrinsic(memcmp)
289 #endif /* BUGGY_MSC */
298 register int matches = 0;
302 register int squash = arg[2].arg_len & 1;
304 tbl = (short*) arg[2].arg_ptr.arg_cval;
306 send = s + str->str_cur;
308 fatal("panic: do_trans");
314 if (!arg[2].arg_len) {
316 if ((ch = tbl[*s & 0377]) >= 0) {
326 if ((ch = tbl[*s & 0377]) >= 0) {
328 if (matches++ && squash) {
337 else if (ch == -1) /* -1 is unmapped character */
338 *d++ = *s; /* -2 is delete character */
341 matches += send - d; /* account for disappeared chars */
343 str->str_cur = d - str->str_ptr;
354 register STR **st = stack->ary_array;
355 register int sp = arglast[1];
356 register int items = arglast[2] - sp;
357 register char *delim = str_get(st[sp]);
358 int delimlen = st[sp]->str_cur;
362 str_sset(str, *st++);
366 for (; items > 0; items--,st++) {
367 str_ncat(str,delim,delimlen);
372 for (; items > 0; items--,st++)
383 register STR **st = stack->ary_array;
384 register int sp = arglast[1];
386 register char *pat = str_get(st[sp]);
387 register char *patend = pat + st[sp]->str_cur;
391 static char *null10 = "\0\0\0\0\0\0\0\0\0\0";
392 static char *space10 = " ";
394 /* These must not be in registers: */
400 unsigned long aulong;
405 items = arglast[2] - sp;
408 while (pat < patend) {
409 #define NEXTFROM (items-- > 0 ? *st++ : &str_no)
412 len = index("@Xxu",datumtype) ? 0 : items;
415 else if (isdigit(*pat)) {
417 while (isdigit(*pat))
418 len = (len * 10) + (*pat++ - '0');
426 fatal("% may only be used in unpack");
437 if (str->str_cur < len)
438 fatal("X outside of string");
440 str->str_ptr[str->str_cur] = '\0';
445 str_ncat(str,null10,10);
448 str_ncat(str,null10,len);
453 aptr = str_get(fromstr);
455 len = fromstr->str_cur;
456 if (fromstr->str_cur > len)
457 str_ncat(str,aptr,len);
459 str_ncat(str,aptr,fromstr->str_cur);
460 len -= fromstr->str_cur;
461 if (datumtype == 'A') {
463 str_ncat(str,space10,10);
466 str_ncat(str,space10,len);
470 str_ncat(str,null10,10);
473 str_ncat(str,null10,len);
481 int saveitems = items;
484 aptr = str_get(fromstr);
486 len = fromstr->str_cur;
489 str->str_cur += (len+7)/8;
490 STR_GROW(str, str->str_cur + 1);
491 aptr = str->str_ptr + aint;
492 if (len > fromstr->str_cur)
493 len = fromstr->str_cur;
496 if (datumtype == 'B') {
497 for (len = 0; len++ < aint;) {
502 *aptr++ = items & 0xff;
508 for (len = 0; len++ < aint;) {
514 *aptr++ = items & 0xff;
520 if (datumtype == 'B')
521 items <<= 7 - (aint & 7);
523 items >>= 7 - (aint & 7);
524 *aptr++ = items & 0xff;
526 pat = str->str_ptr + str->str_cur;
538 int saveitems = items;
541 aptr = str_get(fromstr);
543 len = fromstr->str_cur;
546 str->str_cur += (len+1)/2;
547 STR_GROW(str, str->str_cur + 1);
548 aptr = str->str_ptr + aint;
549 if (len > fromstr->str_cur)
550 len = fromstr->str_cur;
553 if (datumtype == 'H') {
554 for (len = 0; len++ < aint;) {
556 items |= ((*pat++ & 15) + 9) & 15;
558 items |= *pat++ & 15;
562 *aptr++ = items & 0xff;
568 for (len = 0; len++ < aint;) {
570 items |= (((*pat++ & 15) + 9) & 15) << 4;
572 items |= (*pat++ & 15) << 4;
576 *aptr++ = items & 0xff;
582 *aptr++ = items & 0xff;
583 pat = str->str_ptr + str->str_cur;
595 aint = (int)str_gnum(fromstr);
597 str_ncat(str,&achar,sizeof(char));
600 /* Float and double added by gnb@melba.bby.oz.au 22/11/89 */
605 afloat = (float)str_gnum(fromstr);
606 str_ncat(str, (char *)&afloat, sizeof (float));
613 adouble = (double)str_gnum(fromstr);
614 str_ncat(str, (char *)&adouble, sizeof (double));
620 ashort = (short)str_gnum(fromstr);
622 ashort = htons(ashort);
624 str_ncat(str,(char*)&ashort,sizeof(short));
631 ashort = (short)str_gnum(fromstr);
632 str_ncat(str,(char*)&ashort,sizeof(short));
638 auint = U_I(str_gnum(fromstr));
639 str_ncat(str,(char*)&auint,sizeof(unsigned int));
645 aint = (int)str_gnum(fromstr);
646 str_ncat(str,(char*)&aint,sizeof(int));
652 aulong = U_L(str_gnum(fromstr));
654 aulong = htonl(aulong);
656 str_ncat(str,(char*)&aulong,sizeof(unsigned long));
662 aulong = U_L(str_gnum(fromstr));
663 str_ncat(str,(char*)&aulong,sizeof(unsigned long));
669 along = (long)str_gnum(fromstr);
670 str_ncat(str,(char*)&along,sizeof(long));
676 aptr = str_get(fromstr);
677 str_ncat(str,(char*)&aptr,sizeof(char*));
682 aptr = str_get(fromstr);
683 aint = fromstr->str_cur;
684 STR_GROW(str,aint * 4 / 3);
696 doencodes(str, aptr, todo);
707 doencodes(str, s, len)
715 str_ncat(str, hunk, 1);
718 hunk[0] = ' ' + (077 & (*s >> 2));
719 hunk[1] = ' ' + (077 & ((*s << 4) & 060 | (s[1] >> 4) & 017));
720 hunk[2] = ' ' + (077 & ((s[1] << 2) & 074 | (s[2] >> 6) & 03));
721 hunk[3] = ' ' + (077 & (s[2] & 077));
722 str_ncat(str, hunk, 4);
726 for (s = str->str_ptr; *s; s++) {
730 str_ncat(str, "\n", 1);
734 do_sprintf(str,len,sarg)
744 static STR *sargnull = &str_no;
752 len--; /* don't count pattern string */
753 origs = t = s = str_get(*sarg);
754 send = s + (*sarg)->str_cur;
757 if (len <= 0 || !*sarg) {
761 for ( ; t < send && *t != '%'; t++) ;
763 break; /* end of format string, ignore extra args */
768 for (t++; t < send; t++) {
777 case '0': case '1': case '2': case '3': case '4':
778 case '5': case '6': case '7': case '8': case '9':
779 case '.': case '#': case '-': case '+': case ' ':
787 xlen = (int)str_gnum(*(sarg++));
788 if (strEQ(f,"%c")) { /* some printfs fail on null chars */
794 (void)sprintf(xs,f,xlen);
805 (void)sprintf(xs,f,(long)str_gnum(*(sarg++)));
807 (void)sprintf(xs,f,(int)str_gnum(*(sarg++)));
813 case 'x': case 'o': case 'u':
816 value = str_gnum(*(sarg++));
818 (void)sprintf(xs,f,U_L(value));
820 (void)sprintf(xs,f,U_I(value));
823 case 'E': case 'e': case 'f': case 'G': case 'g':
826 (void)sprintf(xs,f,str_gnum(*(sarg++)));
833 xlen = (*sarg)->str_cur;
834 if (*xs == 'S' && xs[1] == 't' && xs[2] == 'B' && xs[3] == '\0'
835 && xlen == sizeof(STBP)) {
836 STR *tmpstr = Str_new(24,0);
838 stab_fullname(tmpstr, ((STAB*)(*sarg))); /* a stab value! */
839 sprintf(tokenbuf,"*%s",tmpstr->str_ptr);
840 /* reformat to non-binary */
842 xlen = strlen(tokenbuf);
846 if (strEQ(f,"%s")) { /* some printfs fail on >128 chars */
847 break; /* so handle simple case */
849 strcpy(tokenbuf+64,f); /* sprintf($s,...$s...) */
851 (void)sprintf(buf,tokenbuf+64,xs);
856 /* end of switch, copy results */
858 STR_GROW(str, str->str_cur + (f - s) + len + 1);
859 str_ncat(str, s, f - s);
860 str_ncat(str, xs, xlen);
862 break; /* break from for loop */
865 str_ncat(str, s, t - s);
874 register STR **st = stack->ary_array;
875 register int sp = arglast[1];
876 register int items = arglast[2] - sp;
877 register STR *str = &str_undef;
879 for (st += ++sp; items > 0; items--,st++) {
883 (void)apush(ary,str);
889 do_unshift(ary,arglast)
893 register STR **st = stack->ary_array;
894 register int sp = arglast[1];
895 register int items = arglast[2] - sp;
901 for (st += ++sp; i < items; i++,st++) {
904 (void)astore(ary,i,str);
909 do_subr(arg,gimme,arglast)
914 register STR **st = stack->ary_array;
915 register int sp = arglast[1];
916 register int items = arglast[2] - sp;
920 int oldsave = savestack->ary_fill;
921 int oldtmps_base = tmps_base;
922 int hasargs = ((arg[2].arg_type & A_MASK) != A_NULL);
925 if ((arg[1].arg_type & A_MASK) == A_WORD)
926 stab = arg[1].arg_ptr.arg_stab;
928 STR *tmpstr = stab_val(arg[1].arg_ptr.arg_stab);
931 stab = stabent(str_get(tmpstr),TRUE);
936 fatal("Undefined subroutine called");
937 if (!(sub = stab_sub(stab))) {
938 STR *tmpstr = arg[0].arg_ptr.arg_str;
940 stab_fullname(tmpstr, stab);
941 fatal("Undefined subroutine \"%s\" called",tmpstr->str_ptr);
943 if (arg->arg_type == O_DBSUBR && !sub->usersub) {
944 str = stab_val(DBsub);
946 stab_fullname(str,stab);
947 sub = stab_sub(DBsub);
949 fatal("No DBsub routine");
951 str = Str_new(15, sizeof(CSV));
952 str->str_state = SS_SCSV;
953 (void)apush(savestack,str);
954 csv = (CSV*)str->str_ptr;
957 csv->curcsv = curcsv;
958 csv->curcmd = curcmd;
959 csv->depth = sub->depth;
960 csv->wantarray = gimme;
961 csv->hasargs = hasargs;
965 csv->savearray = Null(ARRAY*);;
966 csv->argarray = Null(ARRAY*);
967 st[sp] = arg->arg_ptr.arg_str;
970 return (*sub->usersub)(sub->userindex,sp,items);
973 csv->savearray = stab_xarray(defstab);
974 csv->argarray = afake(defstab, items, &st[sp+1]);
975 stab_xarray(defstab) = csv->argarray;
978 if (sub->depth >= 2) { /* save temporaries on recursion? */
979 if (sub->depth == 100 && dowarn)
980 warn("Deep recursion on subroutine \"%s\"",stab_name(stab));
981 savelist(sub->tosave->ary_array,sub->tosave->ary_fill);
983 tmps_base = tmps_max;
984 sp = cmd_exec(sub->cmd,gimme, --sp); /* so do it already */
985 st = stack->ary_array;
987 tmps_base = oldtmps_base;
988 for (items = arglast[0] + 1; items <= sp; items++)
989 st[items] = str_mortal(st[items]);
990 /* in case restore wipes old str */
991 restorelist(oldsave);
996 do_assign(arg,gimme,arglast)
1002 register STR **st = stack->ary_array;
1003 STR **firstrelem = st + arglast[1] + 1;
1004 STR **firstlelem = st + arglast[0] + 1;
1005 STR **lastrelem = st + arglast[2];
1006 STR **lastlelem = st + arglast[1];
1007 register STR **relem;
1008 register STR **lelem;
1011 register ARRAY *ary;
1012 register int makelocal;
1016 makelocal = (arg->arg_flags & AF_LOCAL);
1017 localizing = makelocal;
1018 delaymagic = DM_DELAY; /* catch simultaneous items */
1020 /* If there's a common identifier on both sides we have to take
1021 * special care that assigning the identifier on the left doesn't
1022 * clobber a value on the right that's used later in the list.
1024 if (arg->arg_flags & AF_COMMON) {
1025 for (relem = firstrelem; relem <= lastrelem; relem++) {
1027 *relem = str_mortal(str);
1034 while (lelem <= lastlelem) {
1036 if (str->str_state >= SS_HASH) {
1037 if (str->str_state == SS_ARY) {
1039 ary = saveary(str->str_u.str_stab);
1041 ary = stab_array(str->str_u.str_stab);
1045 while (relem <= lastrelem) { /* gobble up all the rest */
1046 str = Str_new(28,0);
1048 str_sset(str,*relem);
1050 (void)astore(ary,i++,str);
1053 else if (str->str_state == SS_HASH) {
1057 STAB *tmpstab = str->str_u.str_stab;
1060 hash = savehash(str->str_u.str_stab);
1062 hash = stab_hash(str->str_u.str_stab);
1063 if (tmpstab == envstab) {
1065 environ[0] = Nullch;
1067 else if (tmpstab == sigstab) {
1072 for (i = 1; i < NSIG; i++)
1073 signal(i, SIG_DFL); /* crunch, crunch, crunch */
1076 else if (hash->tbl_dbm)
1079 hclear(hash, magic == 'D'); /* wipe any dbm file too */
1082 while (relem < lastrelem) { /* gobble up all the rest */
1086 str = &str_no, relem++;
1087 tmps = str_get(str);
1088 tmpstr = Str_new(29,0);
1090 str_sset(tmpstr,*relem); /* value */
1091 *(relem++) = tmpstr;
1092 (void)hstore(hash,tmps,str->str_cur,tmpstr,0);
1094 str_magic(tmpstr, tmpstab, magic, tmps, str->str_cur);
1095 stabset(tmpstr->str_magic, tmpstr);
1100 fatal("panic: do_assign");
1105 if (relem <= lastrelem) {
1106 str_sset(str, *relem);
1110 str_sset(str, &str_undef);
1111 if (gimme == G_ARRAY) {
1112 i = ++lastrelem - firstrelem;
1113 relem++; /* tacky, I suppose */
1114 astore(stack,i,str);
1115 if (st != stack->ary_array) {
1116 st = stack->ary_array;
1117 firstrelem = st + arglast[1] + 1;
1118 firstlelem = st + arglast[0] + 1;
1119 lastlelem = st + arglast[1];
1121 relem = lastrelem + 1;
1128 if (delaymagic > 1) {
1129 if (delaymagic & DM_REUID) {
1133 if (uid != euid || setuid(uid) < 0)
1134 fatal("No setreuid available");
1137 if (delaymagic & DM_REGID) {
1141 if (gid != egid || setgid(gid) < 0)
1142 fatal("No setregid available");
1148 if (gimme == G_ARRAY) {
1149 i = lastrelem - firstrelem + 1;
1151 Copy(firstrelem, firstlelem, i, STR*);
1152 return arglast[0] + i;
1155 str_numset(arg->arg_ptr.arg_str,(double)(arglast[2] - arglast[1]));
1156 *firstlelem = arg->arg_ptr.arg_str;
1157 return arglast[0] + 1;
1162 do_study(str,arg,gimme,arglast)
1168 register unsigned char *s;
1169 register int pos = str->str_cur;
1171 register int *sfirst;
1172 register int *snext;
1173 static int maxscream = -1;
1174 static STR *lastscream = Nullstr;
1176 int retarg = arglast[0] + 1;
1179 s = (unsigned char*)(str_get(str));
1181 s = Null(unsigned char*);
1184 lastscream->str_pok &= ~SP_STUDIED;
1190 if (pos > maxscream) {
1191 if (maxscream < 0) {
1192 maxscream = pos + 80;
1193 New(301,screamfirst, 256, int);
1194 New(302,screamnext, maxscream, int);
1197 maxscream = pos + pos / 4;
1198 Renew(screamnext, maxscream, int);
1202 sfirst = screamfirst;
1205 if (!sfirst || !snext)
1206 fatal("do_study: out of memory");
1208 for (ch = 256; ch; --ch)
1212 while (--pos >= 0) {
1214 if (sfirst[ch] >= 0)
1215 snext[pos] = sfirst[ch] - pos;
1220 /* If there were any case insensitive searches, we must assume they
1221 * all are. This speeds up insensitive searches much more than
1222 * it slows down sensitive ones.
1225 sfirst[fold[ch]] = pos;
1228 str->str_pok |= SP_STUDIED;
1231 str_numset(arg->arg_ptr.arg_str,(double)retval);
1232 stack->ary_array[retarg] = arg->arg_ptr.arg_str;
1237 do_defined(str,arg,gimme,arglast)
1244 register int retarg = arglast[0] + 1;
1249 if ((arg[1].arg_type & A_MASK) != A_LEXPR)
1250 fatal("Illegal argument to defined()");
1251 arg = arg[1].arg_ptr.arg_arg;
1252 type = arg->arg_type;
1254 if (type == O_SUBR || type == O_DBSUBR)
1255 retval = stab_sub(arg[1].arg_ptr.arg_stab) != 0;
1256 else if (type == O_ARRAY || type == O_LARRAY ||
1257 type == O_ASLICE || type == O_LASLICE )
1258 retval = ((ary = stab_xarray(arg[1].arg_ptr.arg_stab)) != 0
1259 && ary->ary_max >= 0 );
1260 else if (type == O_HASH || type == O_LHASH ||
1261 type == O_HSLICE || type == O_LHSLICE )
1262 retval = ((hash = stab_xhash(arg[1].arg_ptr.arg_stab)) != 0
1263 && hash->tbl_array);
1266 str_numset(str,(double)retval);
1267 stack->ary_array[retarg] = str;
1272 do_undef(str,arg,gimme,arglast)
1279 register STAB *stab;
1280 int retarg = arglast[0] + 1;
1282 if ((arg[1].arg_type & A_MASK) != A_LEXPR)
1283 fatal("Illegal argument to undef()");
1284 arg = arg[1].arg_ptr.arg_arg;
1285 type = arg->arg_type;
1287 if (type == O_ARRAY || type == O_LARRAY) {
1288 stab = arg[1].arg_ptr.arg_stab;
1289 afree(stab_xarray(stab));
1290 stab_xarray(stab) = Null(ARRAY*);
1292 else if (type == O_HASH || type == O_LHASH) {
1293 stab = arg[1].arg_ptr.arg_stab;
1294 if (stab == envstab)
1295 environ[0] = Nullch;
1296 else if (stab == sigstab) {
1299 for (i = 1; i < NSIG; i++)
1300 signal(i, SIG_DFL); /* munch, munch, munch */
1302 (void)hfree(stab_xhash(stab), TRUE);
1303 stab_xhash(stab) = Null(HASH*);
1305 else if (type == O_SUBR || type == O_DBSUBR) {
1306 stab = arg[1].arg_ptr.arg_stab;
1307 if (stab_sub(stab)) {
1308 cmd_free(stab_sub(stab)->cmd);
1309 stab_sub(stab)->cmd = Nullcmd;
1310 afree(stab_sub(stab)->tosave);
1311 Safefree(stab_sub(stab));
1312 stab_sub(stab) = Null(SUBR*);
1316 fatal("Can't undefine that kind of object");
1317 str_numset(str,0.0);
1318 stack->ary_array[retarg] = str;
1323 do_vec(lvalue,astr,arglast)
1328 STR **st = stack->ary_array;
1329 int sp = arglast[0];
1330 register STR *str = st[++sp];
1331 register int offset = (int)str_gnum(st[++sp]);
1332 register int size = (int)str_gnum(st[++sp]);
1333 unsigned char *s = (unsigned char*)str_get(str);
1334 unsigned long retnum;
1338 offset *= size; /* turn into bit offset */
1339 len = (offset + size + 7) / 8;
1340 if (offset < 0 || size < 1)
1342 else if (!lvalue && len > str->str_cur)
1345 if (len > str->str_cur) {
1347 (void)bzero(str->str_ptr + str->str_cur, len - str->str_cur);
1350 s = (unsigned char*)str_get(str);
1352 retnum = (s[offset >> 3] >> (offset & 7)) & ((1 << size) - 1);
1357 else if (size == 16)
1358 retnum = (s[offset] << 8) + s[offset+1];
1359 else if (size == 32)
1360 retnum = (s[offset] << 24) + (s[offset + 1] << 16) +
1361 (s[offset + 2] << 8) + s[offset+3];
1364 if (lvalue) { /* it's an lvalue! */
1365 struct lstring *lstr = (struct lstring*)astr;
1367 astr->str_magic = str;
1368 st[sp]->str_rare = 'v';
1369 lstr->lstr_offset = offset;
1370 lstr->lstr_len = size;
1374 str_numset(astr,(double)retnum);
1384 struct lstring *lstr = (struct lstring*)str;
1385 register int offset;
1387 register unsigned char *s = (unsigned char*)mstr->str_ptr;
1388 register unsigned long lval = U_L(str_gnum(str));
1392 str->str_magic = Nullstr;
1393 offset = lstr->lstr_offset;
1394 size = lstr->lstr_len;
1396 mask = (1 << size) - 1;
1400 s[offset] &= ~(mask << size);
1401 s[offset] |= lval << size;
1405 s[offset] = lval & 255;
1406 else if (size == 16) {
1407 s[offset] = (lval >> 8) & 255;
1408 s[offset+1] = lval & 255;
1410 else if (size == 32) {
1411 s[offset] = (lval >> 24) & 255;
1412 s[offset+1] = (lval >> 16) & 255;
1413 s[offset+2] = (lval >> 8) & 255;
1414 s[offset+3] = lval & 255;
1423 register char *tmps;
1431 if (str->str_state == SS_ARY) {
1432 ary = stab_array(str->str_u.str_stab);
1433 for (i = 0; i <= ary->ary_fill; i++)
1434 do_chop(astr,ary->ary_array[i]);
1437 if (str->str_state == SS_HASH) {
1438 hash = stab_hash(str->str_u.str_stab);
1439 (void)hiterinit(hash);
1440 while (entry = hiternext(hash))
1441 do_chop(astr,hiterval(hash,entry));
1444 tmps = str_get(str);
1447 tmps += str->str_cur - (str->str_cur != 0);
1448 str_nset(astr,tmps,1); /* remember last char */
1449 *tmps = '\0'; /* wipe it out */
1450 str->str_cur = tmps - str->str_ptr;
1455 do_vop(optype,str,left,right)
1461 register char *l = str_get(left);
1462 register char *r = str_get(right);
1465 len = left->str_cur;
1466 if (len > right->str_cur)
1467 len = right->str_cur;
1468 if (str->str_cur > len)
1470 else if (str->str_cur < len) {
1472 (void)bzero(str->str_ptr + str->str_cur, len - str->str_cur);
1494 if (right->str_cur > len)
1495 str_ncat(str,right->str_ptr+len,right->str_cur - len);
1496 else if (left->str_cur > len)
1497 str_ncat(str,left->str_ptr+len,left->str_cur - len);
1506 register STR **st = stack->ary_array;
1507 register int sp = arglast[1];
1508 register int items = arglast[2] - sp;
1515 for (st += ++sp; items--; st++)
1516 tainted |= (*st)->str_tainted;
1517 st = stack->ary_array;
1519 items = arglast[2] - sp;
1522 taintproper("Insecure dependency in syscall");
1524 /* This probably won't work on machines where sizeof(long) != sizeof(int)
1525 * or where sizeof(long) != sizeof(char*). But such machines will
1526 * not likely have syscall implemented either, so who cares?
1529 if (st[++sp]->str_nok || !i)
1530 arg[i++] = (long)str_gnum(st[sp]);
1533 arg[i++] = (long)st[sp]->str_ptr;
1537 items = arglast[2] - sp;
1540 fatal("Too few args to syscall");
1542 retval = syscall(arg[0]);
1545 retval = syscall(arg[0],arg[1]);
1548 retval = syscall(arg[0],arg[1],arg[2]);
1551 retval = syscall(arg[0],arg[1],arg[2],arg[3]);
1554 retval = syscall(arg[0],arg[1],arg[2],arg[3],arg[4]);
1557 retval = syscall(arg[0],arg[1],arg[2],arg[3],arg[4],arg[5]);
1560 retval = syscall(arg[0],arg[1],arg[2],arg[3],arg[4],arg[5],arg[6]);
1563 retval = syscall(arg[0],arg[1],arg[2],arg[3],arg[4],arg[5],arg[6],
1569 fatal("syscall() unimplemented");