mirror of
git://git.code.sf.net/p/zsh/code
synced 2025-01-19 11:31:26 +01:00
c4dba4f2e6
Unless ary1 is made from a scalar, semantics of (@) in double quotes is lost.
4429 lines
111 KiB
C
4429 lines
111 KiB
C
/*
|
|
* subst.c - various substitutions
|
|
*
|
|
* This file is part of zsh, the Z shell.
|
|
*
|
|
* Copyright (c) 1992-1997 Paul Falstad
|
|
* All rights reserved.
|
|
*
|
|
* Permission is hereby granted, without written agreement and without
|
|
* license or royalty fees, to use, copy, modify, and distribute this
|
|
* software and to distribute modified versions of this software for any
|
|
* purpose, provided that the above copyright notice and the following
|
|
* two paragraphs appear in all copies of this software.
|
|
*
|
|
* In no event shall Paul Falstad or the Zsh Development Group be liable
|
|
* to any party for direct, indirect, special, incidental, or consequential
|
|
* damages arising out of the use of this software and its documentation,
|
|
* even if Paul Falstad and the Zsh Development Group have been advised of
|
|
* the possibility of such damage.
|
|
*
|
|
* Paul Falstad and the Zsh Development Group specifically disclaim any
|
|
* warranties, including, but not limited to, the implied warranties of
|
|
* merchantability and fitness for a particular purpose. The software
|
|
* provided hereunder is on an "as is" basis, and Paul Falstad and the
|
|
* Zsh Development Group have no obligation to provide maintenance,
|
|
* support, updates, enhancements, or modifications.
|
|
*
|
|
*/
|
|
|
|
#include "zsh.mdh"
|
|
#include "subst.pro"
|
|
|
|
#define LF_ARRAY 1
|
|
|
|
/**/
|
|
char nulstring[] = {Nularg, '\0'};
|
|
|
|
/* Do substitutions before fork. These are:
|
|
* - Process substitution: <(...), >(...), =(...)
|
|
* - Parameter substitution
|
|
* - Command substitution
|
|
* Followed by
|
|
* - Quote removal
|
|
* - Brace expansion
|
|
* - Tilde and equals substitution
|
|
*
|
|
* "flag"s contains PREFORK_* flags, defined in zsh.h.
|
|
*
|
|
* "ret_flags" is used to return values from nested parameter
|
|
* substitions. It may be NULL in which case PREFORK_SUBEXP
|
|
* must not appear in flags; any return value from below
|
|
* will be discarded.
|
|
*/
|
|
|
|
/**/
|
|
mod_export void
|
|
prefork(LinkList list, int flags, int *ret_flags)
|
|
{
|
|
LinkNode node, stop = 0;
|
|
int keep = 0, asssub = (flags & PREFORK_TYPESET) && isset(KSHTYPESET);
|
|
int ret_flags_local = 0;
|
|
if (!ret_flags)
|
|
ret_flags = &ret_flags_local; /* will be discarded */
|
|
|
|
queue_signals();
|
|
for (node = firstnode(list); node; incnode(node)) {
|
|
if (isset(SHFILEEXPANSION)) {
|
|
/*
|
|
* Here and below we avoid taking the address
|
|
* of a void * and then pretending it's a char **
|
|
* instead of a void ** by a little inefficiency.
|
|
* This could be avoided with some extra linked list
|
|
* machinery, but that would need quite a lot of work
|
|
* to ensure consistency. What we really need is
|
|
* templates...
|
|
*/
|
|
char *cptr = (char *)getdata(node);
|
|
filesub(&cptr, flags & (PREFORK_TYPESET|PREFORK_ASSIGN));
|
|
/*
|
|
* The assignment is so simple it's not worth
|
|
* testing if cptr changed...
|
|
*/
|
|
setdata(node, cptr);
|
|
}
|
|
if (!(node = stringsubst(list, node,
|
|
flags & ~(PREFORK_TYPESET|PREFORK_ASSIGN),
|
|
ret_flags, asssub))) {
|
|
unqueue_signals();
|
|
return;
|
|
}
|
|
}
|
|
for (node = firstnode(list); node; incnode(node)) {
|
|
if (node == stop)
|
|
keep = 0;
|
|
if (*(char *)getdata(node)) {
|
|
remnulargs(getdata(node));
|
|
if (unset(IGNOREBRACES) && !(flags & PREFORK_SINGLE)) {
|
|
if (!keep)
|
|
stop = nextnode(node);
|
|
while (hasbraces(getdata(node))) {
|
|
keep = 1;
|
|
xpandbraces(list, &node);
|
|
}
|
|
}
|
|
if (unset(SHFILEEXPANSION)) {
|
|
char *cptr = (char *)getdata(node);
|
|
filesub(&cptr, flags & (PREFORK_TYPESET|PREFORK_ASSIGN));
|
|
setdata(node, cptr);
|
|
}
|
|
} else if (!(flags & PREFORK_SINGLE) && !keep)
|
|
uremnode(list, node);
|
|
if (errflag) {
|
|
unqueue_signals();
|
|
return;
|
|
}
|
|
}
|
|
unqueue_signals();
|
|
}
|
|
|
|
/*
|
|
* Perform $'...' quoting. The arguments are
|
|
* strstart The start of the string
|
|
* pstrdpos Initially, *pstrdpos is the position where the $ of the $'
|
|
* occurs. It will be updated to the next character after the
|
|
* last ' of the $'...'.
|
|
* The return value is the entire allocated string from strstart on the heap.
|
|
* Note the original string may be modified in the process.
|
|
*/
|
|
/**/
|
|
static char *
|
|
stringsubstquote(char *strstart, char **pstrdpos)
|
|
{
|
|
int len;
|
|
char *strdpos = *pstrdpos, *strsub, *strret;
|
|
|
|
strsub = getkeystring(strdpos+2, &len,
|
|
GETKEYS_DOLLARS_QUOTE, NULL);
|
|
len += 2; /* measured from strdpos */
|
|
|
|
if (strstart != strdpos) {
|
|
*strdpos = '\0';
|
|
if (strdpos[len])
|
|
strret = zhtricat(strstart, strsub, strdpos + len);
|
|
else
|
|
strret = dyncat(strstart, strsub);
|
|
} else if (strdpos[len])
|
|
strret = dyncat(strsub, strdpos + len);
|
|
else if (*strsub)
|
|
strret = strsub;
|
|
else {
|
|
/* This ensures a $'' doesn't get elided. */
|
|
strret = dupstring(nulstring);
|
|
}
|
|
|
|
*pstrdpos = strret + (strdpos - strstart) + strlen(strsub);
|
|
|
|
return strret;
|
|
}
|
|
|
|
/**/
|
|
static LinkNode
|
|
stringsubst(LinkList list, LinkNode node, int pf_flags, int *ret_flags,
|
|
int asssub)
|
|
{
|
|
int qt;
|
|
char *str3 = (char *)getdata(node);
|
|
char *str = str3, c;
|
|
|
|
while (!errflag && (c = *str)) {
|
|
if (((c = *str) == Inang || c == OutangProc ||
|
|
(str == str3 && c == Equals))
|
|
&& str[1] == Inpar) {
|
|
char *subst, *rest, *snew, *sptr;
|
|
int str3len = str - str3, sublen, restlen;
|
|
|
|
if (c == Inang || c == OutangProc)
|
|
subst = getproc(str, &rest); /* <(...) or >(...) */
|
|
else
|
|
subst = getoutputfile(str, &rest); /* =(...) */
|
|
if (errflag)
|
|
return NULL;
|
|
if (!subst)
|
|
rest = subst = "";
|
|
|
|
sublen = strlen(subst);
|
|
restlen = strlen(rest);
|
|
sptr = snew = hcalloc(str3len + sublen + restlen + 1);
|
|
if (str3len) {
|
|
memcpy(sptr, str3, str3len);
|
|
sptr += str3len;
|
|
}
|
|
if (sublen) {
|
|
memcpy(sptr, subst, sublen);
|
|
sptr += sublen;
|
|
}
|
|
if (restlen)
|
|
memcpy(sptr, rest, restlen);
|
|
sptr[restlen] = '\0';
|
|
str3 = snew;
|
|
str = snew + str3len + sublen;
|
|
setdata(node, str3);
|
|
} else
|
|
str++;
|
|
}
|
|
str = str3;
|
|
|
|
while (!errflag && (c = *str)) {
|
|
if ((qt = c == Qstring) || c == String) {
|
|
if ((c = str[1]) == Inpar || c == Inparmath) {
|
|
if (!qt)
|
|
list->list.flags |= LF_ARRAY;
|
|
str++;
|
|
goto comsub;
|
|
} else if (c == Inbrack) {
|
|
/* $[...] */
|
|
char *str2 = str;
|
|
str2++;
|
|
if (skipparens(Inbrack, Outbrack, &str2)) {
|
|
zerr("closing bracket missing");
|
|
return NULL;
|
|
}
|
|
str2[-1] = *str = '\0';
|
|
str = arithsubst(str + 2, &str3, str2);
|
|
setdata(node, (void *) str3);
|
|
continue;
|
|
} else if (c == Snull) {
|
|
str3 = stringsubstquote(str3, &str);
|
|
setdata(node, (void *) str3);
|
|
continue;
|
|
} else {
|
|
/*
|
|
* To avoid setting and unsetting the SHWORDSPLIT
|
|
* option, we pass flags if we need to control it for
|
|
* recursive expansion via multsub()
|
|
* If PREFORK_NOSHWORDSPLIT is set, the option is
|
|
* disregarded; otherwise, use it if set.
|
|
* If PREFORK_SPLIT is set, splitting is forced,
|
|
* regardless of the option
|
|
* If PREFORK_SHWORDSPLIT is already set, or used by the
|
|
* previous two to signal paramsubst(), we'll do
|
|
* sh-style wordsplitting on parameters.
|
|
*/
|
|
if ((isset(SHWORDSPLIT) &&
|
|
!(pf_flags & PREFORK_NOSHWORDSPLIT)) ||
|
|
(pf_flags & PREFORK_SPLIT))
|
|
pf_flags |= PREFORK_SHWORDSPLIT;
|
|
node = paramsubst(
|
|
list, node, &str, qt,
|
|
pf_flags & (PREFORK_SINGLE|PREFORK_SHWORDSPLIT|
|
|
PREFORK_SUBEXP), ret_flags);
|
|
if (errflag || !node)
|
|
return NULL;
|
|
str3 = (char *)getdata(node);
|
|
continue;
|
|
}
|
|
} else if ((qt = c == Qtick) || (c == Tick ? (list->list.flags |= LF_ARRAY) : 0))
|
|
comsub: {
|
|
LinkList pl;
|
|
char *s, *str2 = str;
|
|
char endchar;
|
|
int l1, l2;
|
|
|
|
if (c == Inpar) {
|
|
endchar = Outpar;
|
|
str[-1] = '\0';
|
|
#ifdef DEBUG
|
|
if (skipparens(Inpar, Outpar, &str))
|
|
dputs("BUG: parse error in command substitution");
|
|
#else
|
|
skipparens(Inpar, Outpar, &str);
|
|
#endif
|
|
str--;
|
|
} else if (c == Inparmath) {
|
|
/*
|
|
* Math substitution of the form $((...)).
|
|
* These can be nested, for goodness sake...
|
|
*/
|
|
int mathpar = 1;
|
|
str[-1] = '\0';
|
|
while (mathpar && *str) {
|
|
str++;
|
|
if (*str == Outparmath)
|
|
mathpar--;
|
|
else if (*str == Inparmath)
|
|
mathpar++;
|
|
}
|
|
if (*str != Outparmath) {
|
|
zerr("failed to find end of math substitution");
|
|
return NULL;
|
|
}
|
|
str[-1] = '\0';
|
|
if (isset(EXECOPT))
|
|
str = arithsubst(str2 + 2, &str3, str+1);
|
|
else
|
|
strncpy(str3, str2, 1);
|
|
setdata(node, (void *) str3);
|
|
continue;
|
|
} else {
|
|
endchar = c;
|
|
*str = '\0';
|
|
|
|
while (*++str != endchar) {
|
|
if (!*str) {
|
|
zerr("failed to find end of command substitution");
|
|
return NULL;
|
|
}
|
|
}
|
|
}
|
|
*str++ = '\0';
|
|
|
|
/* It is a command substitution, which will be parsed again *
|
|
* by the lexer, so we untokenize it first, but we cannot use *
|
|
* untokenize() since in the case of `...` some Bnulls should *
|
|
* be left unchanged. Note that the lexer doesn't tokenize *
|
|
* the body of a command substitution so if there are some *
|
|
* tokens here they are from a ${(e)~...} substitution. */
|
|
for (str = str2; (c = *++str); )
|
|
if (itok(c) && c != Nularg &&
|
|
!(endchar != Outpar && c == Bnull &&
|
|
(str[1] == '$' || str[1] == '\\' || str[1] == '`' ||
|
|
(qt && str[1] == '"'))))
|
|
*str = ztokens[c - Pound];
|
|
str++;
|
|
if (!(pl = getoutput(str2 + 1, qt ||
|
|
(pf_flags & PREFORK_SINGLE)))) {
|
|
zerr("parse error in command substitution");
|
|
return NULL;
|
|
}
|
|
if (endchar == Outpar)
|
|
str2--;
|
|
if (!(s = (char *) ugetnode(pl))) {
|
|
str = (char *)memmove(str2, str, strlen(str)+1);
|
|
continue;
|
|
}
|
|
if (!qt && (pf_flags & PREFORK_SINGLE) && isset(GLOBSUBST))
|
|
shtokenize(s);
|
|
l1 = str2 - str3;
|
|
l2 = strlen(s);
|
|
if (nonempty(pl)) {
|
|
LinkNode n = lastnode(pl);
|
|
str2 = (char *) hcalloc(l1 + l2 + 1);
|
|
strcpy(str2, str3);
|
|
strcpy(str2 + l1, s);
|
|
setdata(node, str2);
|
|
insertlinklist(pl, node, list);
|
|
s = (char *) getdata(node = n);
|
|
l1 = 0;
|
|
l2 = strlen(s);
|
|
}
|
|
str2 = (char *) hcalloc(l1 + l2 + strlen(str) + 1);
|
|
if (l1)
|
|
strcpy(str2, str3);
|
|
strcpy(str2 + l1, s);
|
|
str = strcpy(str2 + l1 + l2, str);
|
|
str3 = str2;
|
|
setdata(node, str3);
|
|
continue;
|
|
} else if (asssub && ((c == '=') || c == Equals) && str != str3) {
|
|
/*
|
|
* We are in a normal argument which looks like an assignment
|
|
* and is to be treated like one, with no word splitting.
|
|
*/
|
|
pf_flags |= PREFORK_SINGLE;
|
|
}
|
|
str++;
|
|
}
|
|
return errflag ? NULL : node;
|
|
}
|
|
|
|
/*
|
|
* Simplified version of the prefork/singsub processing where
|
|
* we only do substitutions appropriate to quoting. Currently
|
|
* this means only the expansions in $'....'. This is used
|
|
* for the end tag for here documents. As we are not doing
|
|
* `...` expansions, we just use those for quoting. However,
|
|
* they stay in the text. This is weird, but that's not
|
|
* my fault.
|
|
*
|
|
* The remnulargs() makes this consistent with the other forms
|
|
* of substitution, indicating that quotes have been fully
|
|
* processed.
|
|
*
|
|
* The fully processed string is returned.
|
|
*/
|
|
|
|
/**/
|
|
char *
|
|
quotesubst(char *str)
|
|
{
|
|
char *s = str;
|
|
|
|
while (*s) {
|
|
if (*s == String && s[1] == Snull) {
|
|
str = stringsubstquote(str, &s);
|
|
} else {
|
|
s++;
|
|
}
|
|
}
|
|
remnulargs(str);
|
|
return str;
|
|
}
|
|
|
|
/**/
|
|
mod_export void
|
|
globlist(LinkList list, int nountok)
|
|
{
|
|
LinkNode node, next;
|
|
|
|
badcshglob = 0;
|
|
for (node = firstnode(list); !errflag && node; node = next) {
|
|
next = nextnode(node);
|
|
zglob(list, node, nountok);
|
|
}
|
|
if (noerrs)
|
|
badcshglob = 0;
|
|
else if (badcshglob == 1)
|
|
zerr("no match");
|
|
}
|
|
|
|
/* perform substitution on a single word */
|
|
|
|
/**/
|
|
mod_export void
|
|
singsub(char **s)
|
|
{
|
|
local_list1(foo);
|
|
|
|
init_list1(foo, *s);
|
|
|
|
prefork(&foo, PREFORK_SINGLE, NULL);
|
|
if (errflag)
|
|
return;
|
|
*s = (char *) ugetnode(&foo);
|
|
DPUTS(nonempty(&foo), "BUG: singsub() produced more than one word!");
|
|
}
|
|
|
|
/* Perform substitution on a single word, *s. Unlike with singsub(), the
|
|
* result can be more than one word. If split is non-zero, the string is
|
|
* first word-split using IFS, but only for non-quoted "whitespace" (as
|
|
* indicated by Dnull, Snull, Tick, Bnull, Inpar, and Outpar).
|
|
*
|
|
* If arg "a" was non-NULL and we got an array as a result of the parsing,
|
|
* the strings are stored in *a (even for a 1-element array) and *isarr is
|
|
* set to 1. Otherwise, *isarr is set to 0, and the result is put into *s,
|
|
* with any necessary joining of multiple elements using sep (which can be
|
|
* NULL to use IFS). The return value is true iff the expansion resulted
|
|
* in an empty list.
|
|
*
|
|
* *ms_flags is set to bits in the enum above as neeed.
|
|
*/
|
|
|
|
/**/
|
|
static int
|
|
multsub(char **s, int pf_flags, char ***a, int *isarr, char *sep,
|
|
int *ms_flags)
|
|
{
|
|
int l;
|
|
char **r, **p, *x = *s;
|
|
local_list1(foo);
|
|
|
|
if (pf_flags & PREFORK_SPLIT) {
|
|
/*
|
|
* This doesn't handle multibyte characters, but we're
|
|
* looking for whitespace separators which must be ASCII.
|
|
*/
|
|
for ( ; *x; x += l) {
|
|
char c = (l = *x == Meta) ? x[1] ^ 32 : *x;
|
|
l++;
|
|
if (!iwsep(STOUC(c)))
|
|
break;
|
|
*ms_flags |= MULTSUB_WS_AT_START;
|
|
}
|
|
}
|
|
|
|
init_list1(foo, x);
|
|
|
|
if (pf_flags & PREFORK_SPLIT) {
|
|
LinkNode n = firstnode(&foo);
|
|
int inq = 0, inp = 0;
|
|
MB_METACHARINIT();
|
|
for ( ; *x; x += l) {
|
|
int rawc = -1;
|
|
convchar_t c;
|
|
if (itok(STOUC(*x))) {
|
|
/* token, can't be separator, must be single byte */
|
|
rawc = *x;
|
|
l = 1;
|
|
} else {
|
|
l = MB_METACHARLENCONV(x, &c);
|
|
if (!inq && !inp && WC_ZISTYPE(c, ISEP)) {
|
|
*x = '\0';
|
|
for (x += l; *x; x += l) {
|
|
if (itok(STOUC(*x))) {
|
|
/* as above */
|
|
rawc = *x;
|
|
l = 1;
|
|
break;
|
|
}
|
|
l = MB_METACHARLENCONV(x, &c);
|
|
if (!WC_ZISTYPE(c, ISEP))
|
|
break;
|
|
}
|
|
if (!*x) {
|
|
*ms_flags |= MULTSUB_WS_AT_END;
|
|
break;
|
|
}
|
|
insertlinknode(&foo, n, (void *)x), incnode(n);
|
|
}
|
|
}
|
|
switch (rawc) {
|
|
case Dnull: /* " */
|
|
case Snull: /* ' */
|
|
case Tick: /* ` (note: no Qtick!) */
|
|
/* These always occur in unnested pairs. */
|
|
inq = !inq;
|
|
break;
|
|
case Inpar: /* ( */
|
|
inp++;
|
|
break;
|
|
case Outpar: /* ) */
|
|
inp--;
|
|
break;
|
|
case Bnull: /* \ */
|
|
case Bnullkeep:
|
|
/* The parser verified the following char's existence. */
|
|
x += l;
|
|
l = MB_METACHARLEN(x);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
prefork(&foo, pf_flags, ms_flags);
|
|
if (errflag) {
|
|
if (isarr)
|
|
*isarr = 0;
|
|
return 0;
|
|
}
|
|
|
|
if ((l = countlinknodes(&foo)) > 1 || (foo.list.flags & LF_ARRAY && a)) {
|
|
p = r = hcalloc((l + 1) * sizeof(char*));
|
|
while (nonempty(&foo))
|
|
*p++ = (char *)ugetnode(&foo);
|
|
*p = NULL;
|
|
/* We need a way to figure out if a one-item result was a scalar
|
|
* or a single-item array. The parser will have set LF_ARRAY
|
|
* in the latter case, allowing us to return it as an array to
|
|
* our caller (if they provided for that result). */
|
|
if (a && (l > 1 || foo.list.flags & LF_ARRAY)) {
|
|
*a = r;
|
|
*isarr = SCANPM_MATCHMANY;
|
|
return 0;
|
|
}
|
|
*s = sepjoin(r, sep, 1);
|
|
if (isarr)
|
|
*isarr = 0;
|
|
return 0;
|
|
}
|
|
if (l)
|
|
*s = (char *) ugetnode(&foo);
|
|
else
|
|
*s = dupstring("");
|
|
if (isarr)
|
|
*isarr = 0;
|
|
return !l;
|
|
}
|
|
|
|
/*
|
|
* ~, = subs: assign & PREFORK_TYPESET => typeset or magic equals
|
|
* assign & PREFORK_ASSIGN => normal assignment
|
|
*/
|
|
|
|
/**/
|
|
mod_export void
|
|
filesub(char **namptr, int assign)
|
|
{
|
|
char *eql = NULL, *sub = NULL, *str, *ptr;
|
|
int len;
|
|
|
|
filesubstr(namptr, assign);
|
|
|
|
if (!assign)
|
|
return;
|
|
|
|
if (assign & PREFORK_TYPESET) {
|
|
if ((*namptr)[1] && (eql = sub = strchr(*namptr + 1, Equals))) {
|
|
str = sub + 1;
|
|
if ((sub[1] == Tilde || sub[1] == Equals) && filesubstr(&str, assign)) {
|
|
sub[1] = '\0';
|
|
*namptr = dyncat(*namptr, str);
|
|
}
|
|
} else
|
|
return;
|
|
}
|
|
|
|
ptr = *namptr;
|
|
while ((sub = strchr(ptr, ':'))) {
|
|
str = sub + 1;
|
|
len = sub - *namptr;
|
|
if (sub > eql &&
|
|
(sub[1] == Tilde || sub[1] == Equals) &&
|
|
filesubstr(&str, assign)) {
|
|
sub[1] = '\0';
|
|
*namptr = dyncat(*namptr, str);
|
|
}
|
|
ptr = *namptr + len + 1;
|
|
}
|
|
}
|
|
|
|
#define isend(c) ( !(c) || (c)=='/' || (c)==Inpar || (assign && (c)==':') )
|
|
#define isend2(c) ( !(c) || (c)==Inpar || (assign && (c)==':') )
|
|
|
|
/*
|
|
* do =foo substitution, or equivalent.
|
|
* on entry, str should point to the "foo".
|
|
* if assign, this is in an assignment
|
|
* if nomatch, report hard error on failure.
|
|
* if successful, returns the expansion, else NULL.
|
|
*/
|
|
|
|
/**/
|
|
char *
|
|
equalsubstr(char *str, int assign, int nomatch)
|
|
{
|
|
char *pp, *cnam, *cmdstr, *ret;
|
|
|
|
for (pp = str; !isend2(*pp); pp++)
|
|
;
|
|
cmdstr = dupstrpfx(str, pp-str);
|
|
untokenize(cmdstr);
|
|
remnulargs(cmdstr);
|
|
if (!(cnam = findcmd(cmdstr, 1, 0))) {
|
|
if (nomatch)
|
|
zerr("%s not found", cmdstr);
|
|
return NULL;
|
|
}
|
|
ret = dupstring(cnam);
|
|
if (*pp)
|
|
ret = dyncat(ret, pp);
|
|
return ret;
|
|
}
|
|
|
|
/**/
|
|
mod_export int
|
|
filesubstr(char **namptr, int assign)
|
|
{
|
|
char *str = *namptr;
|
|
|
|
if (*str == Tilde && str[1] != '=' && str[1] != Equals) {
|
|
char *ptr, *tmp, *res, *ptr2;
|
|
int val;
|
|
|
|
if (str[1] == Dash)
|
|
str[1] = '-';
|
|
val = zstrtol(str + 1, &ptr, 10);
|
|
if (isend(str[1])) { /* ~ */
|
|
*namptr = dyncat(home ? home : "", str + 1);
|
|
return 1;
|
|
} else if (str[1] == '+' && isend(str[2])) { /* ~+ */
|
|
*namptr = dyncat(pwd, str + 2);
|
|
return 1;
|
|
} else if (str[1] == '-' && isend(str[2])) { /* ~- */
|
|
*namptr = dyncat((tmp = oldpwd) ? tmp : pwd, str + 2);
|
|
return 1;
|
|
} else if (str[1] == Inbrack &&
|
|
(ptr2 = strchr(str+2, Outbrack))) {
|
|
char **arr;
|
|
untokenize(tmp = dupstrpfx(str+2, ptr2 - (str+2)));
|
|
remnulargs(tmp);
|
|
arr = subst_string_by_hook("zsh_directory_name", "n", tmp);
|
|
res = arr ? *arr : NULL;
|
|
if (res) {
|
|
*namptr = dyncat(res, ptr2+1);
|
|
return 1;
|
|
}
|
|
if (isset(NOMATCH))
|
|
zerr("no directory expansion: ~[%s]", tmp);
|
|
return 0;
|
|
} else if (!inblank(str[1]) && isend(*ptr) &&
|
|
(!idigit(str[1]) || (ptr - str < 4))) {
|
|
char *ds;
|
|
|
|
if (val < 0)
|
|
val = -val;
|
|
ds = dstackent(str[1], val);
|
|
if (!ds)
|
|
return 0;
|
|
*namptr = dyncat(ds, ptr);
|
|
return 1;
|
|
} else if ((ptr = itype_end(str+1, IUSER, 0)) != str+1) { /* ~foo */
|
|
char *untok, *hom;
|
|
|
|
if (!isend(*ptr))
|
|
return 0;
|
|
untok = dupstring(++str);
|
|
untok[ptr-str] = 0;
|
|
untokenize(untok);
|
|
|
|
if (!(hom = getnameddir(untok))) {
|
|
if (isset(NOMATCH) && isset(EXECOPT))
|
|
zerr("no such user or named directory: %s", untok);
|
|
return 0;
|
|
}
|
|
*namptr = dyncat(hom, ptr);
|
|
return 1;
|
|
}
|
|
} else if (*str == Equals && isset(EQUALS) && str[1]) { /* =foo */
|
|
char *expn = equalsubstr(str+1, assign, isset(NOMATCH));
|
|
if (expn) {
|
|
*namptr = expn;
|
|
return 1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
#undef isend
|
|
#undef isend2
|
|
|
|
/**/
|
|
static char *
|
|
strcatsub(char **d, char *pb, char *pe, char *src, int l, char *s, int glbsub,
|
|
int copied)
|
|
{
|
|
char *dest;
|
|
int pl = pe - pb;
|
|
|
|
if (!pl && (!s || !*s)) {
|
|
*d = dest = (copied ? src : dupstring(src));
|
|
if (glbsub)
|
|
shtokenize(dest);
|
|
} else {
|
|
*d = dest = hcalloc(pl + l + (s ? strlen(s) : 0) + 1);
|
|
strncpy(dest, pb, pl);
|
|
dest += pl;
|
|
strcpy(dest, src);
|
|
if (glbsub)
|
|
shtokenize(dest);
|
|
dest += l;
|
|
if (s)
|
|
strcpy(dest, s);
|
|
}
|
|
return dest;
|
|
}
|
|
|
|
#ifdef MULTIBYTE_SUPPORT
|
|
#define WCPADWIDTH(cchar, mw) wcpadwidth(cchar, mw)
|
|
|
|
/*
|
|
* Width of character for padding purposes.
|
|
* 0: all characters count 1.
|
|
* 1: use width of multibyte character.
|
|
* 2: non-zero width characters count 1, zero width 0.
|
|
*/
|
|
static int
|
|
wcpadwidth(wchar_t wc, int multi_width)
|
|
{
|
|
int width;
|
|
|
|
switch (multi_width)
|
|
{
|
|
case 0:
|
|
return 1;
|
|
|
|
case 1:
|
|
width = WCWIDTH(wc);
|
|
if (width >= 0)
|
|
return width;
|
|
return 0;
|
|
|
|
default:
|
|
return WCWIDTH(wc) > 0 ? 1 : 0;
|
|
}
|
|
}
|
|
|
|
#else
|
|
#define WCPADWIDTH(cchar, mw) (1)
|
|
#endif
|
|
|
|
/*
|
|
* Pad the string str, returning a result from the heap (or str itself,
|
|
* if it didn't need padding). If str is too large, it will be truncated.
|
|
* Calculations are in terms of width if MULTIBYTE is in effect and
|
|
* multi_width is non-zero, else characters.
|
|
*
|
|
* prenum and postnum are the width to which the string needs padding
|
|
* on the left and right.
|
|
*
|
|
* preone and postone are string to insert once only before and after
|
|
* str. They will be truncated on the left or right, respectively,
|
|
* if necessary to fit the width. Either or both may be NULL in which
|
|
* case they will not be used.
|
|
*
|
|
* premul and postmul are the padding strings to be repeated before
|
|
* on the left (if prenum is non-zero) and right (if postnum is non-zero). If
|
|
* NULL the first character of IFS (typically but not necessarily a space)
|
|
* will be used.
|
|
*/
|
|
|
|
static char *
|
|
dopadding(char *str, int prenum, int postnum, char *preone, char *postone,
|
|
char *premul, char *postmul
|
|
#ifdef MULTIBYTE_SUPPORT
|
|
, int multi_width
|
|
#endif
|
|
)
|
|
{
|
|
char *def, *ret, *t, *r;
|
|
int ls, ls2, lpreone, lpostone, lpremul, lpostmul, lr, f, m, c, cc, cl;
|
|
convchar_t cchar;
|
|
|
|
MB_METACHARINIT();
|
|
if (!ifs || *ifs) {
|
|
char *tmpifs = ifs ? ifs : DEFAULT_IFS;
|
|
def = dupstrpfx(tmpifs, MB_METACHARLEN(tmpifs));
|
|
} else
|
|
def = "";
|
|
if (preone && !*preone)
|
|
preone = def;
|
|
if (postone && !*postone)
|
|
postone = def;
|
|
if (!premul || !*premul)
|
|
premul = def;
|
|
if (!postmul || !*postmul)
|
|
postmul = def;
|
|
|
|
ls = MB_METASTRLEN2(str, multi_width);
|
|
lpreone = preone ? MB_METASTRLEN2(preone, multi_width) : 0;
|
|
lpostone = postone ? MB_METASTRLEN2(postone, multi_width) : 0;
|
|
lpremul = MB_METASTRLEN2(premul, multi_width);
|
|
lpostmul = MB_METASTRLEN2(postmul, multi_width);
|
|
|
|
if (prenum + postnum == ls)
|
|
return str;
|
|
|
|
/*
|
|
* Try to be careful with allocated lengths. The following
|
|
* is a maximum, in case we need the entire repeated string
|
|
* for each repetition. We probably don't, but in case the user
|
|
* has given us something pathological which doesn't convert
|
|
* easily into a width we'd better be safe.
|
|
*/
|
|
lr = strlen(str) + strlen(premul) * prenum + strlen(postmul) * postnum;
|
|
/*
|
|
* Same logic for preone and postone, except those may be NULL.
|
|
*/
|
|
if (preone)
|
|
lr += strlen(preone);
|
|
if (postone)
|
|
lr += strlen(postone);
|
|
r = ret = (char *)zhalloc(lr + 1);
|
|
|
|
if (prenum) {
|
|
/*
|
|
* Pad on the left.
|
|
*/
|
|
if (postnum) {
|
|
/*
|
|
* Pad on both right and left.
|
|
* The strategy is to divide the string into two halves.
|
|
* The first half is dealt with by the left hand padding
|
|
* code, the second by the right hand.
|
|
*/
|
|
ls2 = ls / 2;
|
|
|
|
/* The width left to pad for the first half. */
|
|
f = prenum - ls2;
|
|
if (f <= 0) {
|
|
/* First half doesn't fit. Skip the first -f width. */
|
|
f = -f;
|
|
MB_METACHARINIT();
|
|
while (f > 0) {
|
|
cl = MB_METACHARLENCONV(str, &cchar);
|
|
if (!cl)
|
|
break;
|
|
str += cl;
|
|
f -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
/* Now finish the first half. */
|
|
for (c = prenum; c > 0; ) {
|
|
cl = MB_METACHARLENCONV(str, &cchar);
|
|
if (!cl)
|
|
break;
|
|
while (cl--)
|
|
*r++ = *str++;
|
|
c -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
} else {
|
|
if (f <= lpreone) {
|
|
if (preone) {
|
|
/*
|
|
* The unrepeated string doesn't fit.
|
|
*/
|
|
MB_METACHARINIT();
|
|
/* The width we need to skip */
|
|
f = lpreone - f;
|
|
/* So skip. */
|
|
for (t = preone; f > 0; ) {
|
|
cl = MB_METACHARLENCONV(t, &cchar);
|
|
if (!cl)
|
|
break;
|
|
t += cl;
|
|
f -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
/* Then copy the entire remainder. */
|
|
while (*t)
|
|
*r++ = *t++;
|
|
}
|
|
} else {
|
|
f -= lpreone;
|
|
if (lpremul) {
|
|
if ((m = f % lpremul)) {
|
|
/*
|
|
* Left over fraction of repeated string.
|
|
*/
|
|
MB_METACHARINIT();
|
|
/* Skip this much. */
|
|
m = lpremul - m;
|
|
for (t = premul; m > 0; ) {
|
|
cl = MB_METACHARLENCONV(t, &cchar);
|
|
if (!cl)
|
|
break;
|
|
t += cl;
|
|
m -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
/* Output the rest. */
|
|
while (*t)
|
|
*r++ = *t++;
|
|
}
|
|
for (cc = f / lpremul; cc--;) {
|
|
/* Repeat the repeated string */
|
|
MB_METACHARINIT();
|
|
for (c = lpremul, t = premul; c > 0; ) {
|
|
cl = MB_METACHARLENCONV(t, &cchar);
|
|
if (!cl)
|
|
break;
|
|
while (cl--)
|
|
*r++ = *t++;
|
|
c -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
}
|
|
}
|
|
if (preone) {
|
|
/* Output the full unrepeated string */
|
|
while (*preone)
|
|
*r++ = *preone++;
|
|
}
|
|
}
|
|
/* Output the first half width of the original string. */
|
|
for (c = ls2; c > 0; ) {
|
|
cl = MB_METACHARLENCONV(str, &cchar);
|
|
if (!cl)
|
|
break;
|
|
c -= WCPADWIDTH(cchar, multi_width);
|
|
while (cl--)
|
|
*r++ = *str++;
|
|
}
|
|
}
|
|
/* Other half. In case the string had an odd length... */
|
|
ls2 = ls - ls2;
|
|
/* Width that needs padding... */
|
|
f = postnum - ls2;
|
|
if (f <= 0) {
|
|
/* ...is negative, truncate original string */
|
|
MB_METACHARINIT();
|
|
for (c = postnum; c > 0; ) {
|
|
cl = MB_METACHARLENCONV(str, &cchar);
|
|
if (!cl)
|
|
break;
|
|
c -= WCPADWIDTH(cchar, multi_width);
|
|
while (cl--)
|
|
*r++ = *str++;
|
|
}
|
|
} else {
|
|
/* Rest of original string fits, output it complete */
|
|
while (*str)
|
|
*r++ = *str++;
|
|
if (f <= lpostone) {
|
|
if (postone) {
|
|
/* Can't fit unrepeated string, truncate it */
|
|
for (c = f; c > 0; ) {
|
|
cl = MB_METACHARLENCONV(postone, &cchar);
|
|
if (!cl)
|
|
break;
|
|
c -= WCPADWIDTH(cchar, multi_width);
|
|
while (cl--)
|
|
*r++ = *postone++;
|
|
}
|
|
}
|
|
} else {
|
|
if (postone) {
|
|
f -= lpostone;
|
|
/* Output entire unrepeated string */
|
|
while (*postone)
|
|
*r++ = *postone++;
|
|
}
|
|
if (lpostmul) {
|
|
for (cc = f / lpostmul; cc--;) {
|
|
/* Begin the beguine */
|
|
for (t = postmul; *t; )
|
|
*r++ = *t++;
|
|
}
|
|
if ((m = f % lpostmul)) {
|
|
/* Fill leftovers with chunk of repeated string */
|
|
MB_METACHARINIT();
|
|
while (m > 0) {
|
|
cl = MB_METACHARLENCONV(postmul, &cchar);
|
|
if (!cl)
|
|
break;
|
|
m -= WCPADWIDTH(cchar, multi_width);
|
|
while (cl--)
|
|
*r++ = *postmul++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
/*
|
|
* Pad only on the left.
|
|
*/
|
|
f = prenum - ls;
|
|
if (f <= 0) {
|
|
/*
|
|
* Original string is at least as wide as padding.
|
|
* Truncate original string to width.
|
|
* Truncate on left, so skip the characters we
|
|
* don't need.
|
|
*/
|
|
f = -f;
|
|
MB_METACHARINIT();
|
|
while (f > 0) {
|
|
cl = MB_METACHARLENCONV(str, &cchar);
|
|
if (!cl)
|
|
break;
|
|
str += cl;
|
|
f -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
/* Copy the rest of the original string */
|
|
for (c = prenum; c > 0; ) {
|
|
cl = MB_METACHARLENCONV(str, &cchar);
|
|
if (!cl)
|
|
break;
|
|
while (cl--)
|
|
*r++ = *str++;
|
|
c -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
} else {
|
|
/*
|
|
* We can fit the entire string...
|
|
*/
|
|
if (f <= lpreone) {
|
|
if (preone) {
|
|
/*
|
|
* ...with some fraction of the unrepeated string.
|
|
*/
|
|
/* We need this width of characters. */
|
|
c = f;
|
|
/*
|
|
* We therefore need to skip this width of
|
|
* characters.
|
|
*/
|
|
f = lpreone - f;
|
|
MB_METACHARINIT();
|
|
for (t = preone; f > 0; ) {
|
|
cl = MB_METACHARLENCONV(t, &cchar);
|
|
if (!cl)
|
|
break;
|
|
t += cl;
|
|
f -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
/* Copy the rest of preone */
|
|
while (*t)
|
|
*r++ = *t++;
|
|
}
|
|
} else {
|
|
/*
|
|
* We can fit the whole of preone, needing this width
|
|
* first
|
|
*/
|
|
f -= lpreone;
|
|
if (lpremul) {
|
|
if ((m = f % lpremul)) {
|
|
/*
|
|
* Some fraction of the repeated string needed.
|
|
*/
|
|
/* Need this much... */
|
|
c = m;
|
|
/* ...skipping this much first. */
|
|
m = lpremul - m;
|
|
MB_METACHARINIT();
|
|
for (t = premul; m > 0; ) {
|
|
cl = MB_METACHARLENCONV(t, &cchar);
|
|
if (!cl)
|
|
break;
|
|
t += cl;
|
|
m -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
/* Now the rest of the repeated string. */
|
|
while (c > 0) {
|
|
cl = MB_METACHARLENCONV(t, &cchar);
|
|
if (!cl)
|
|
break;
|
|
while (cl--)
|
|
*r++ = *t++;
|
|
c -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
}
|
|
for (cc = f / lpremul; cc--;) {
|
|
/*
|
|
* Repeat the repeated string.
|
|
*/
|
|
MB_METACHARINIT();
|
|
for (c = lpremul, t = premul; c > 0; ) {
|
|
cl = MB_METACHARLENCONV(t, &cchar);
|
|
if (!cl)
|
|
break;
|
|
while (cl--)
|
|
*r++ = *t++;
|
|
c -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
}
|
|
}
|
|
if (preone) {
|
|
/*
|
|
* Now the entire unrepeated string. Don't
|
|
* count the width, just dump it. This is
|
|
* significant if there are special characters
|
|
* in this string. It's sort of a historical
|
|
* accident that this worked, but there's nothing
|
|
* to stop us just dumping the thing out and assuming
|
|
* the user knows what they're doing.
|
|
*/
|
|
while (*preone)
|
|
*r++ = *preone++;
|
|
}
|
|
}
|
|
/* Now the string being padded */
|
|
while (*str)
|
|
*r++ = *str++;
|
|
}
|
|
}
|
|
} else if (postnum) {
|
|
/*
|
|
* Pad on the right.
|
|
*/
|
|
f = postnum - ls;
|
|
MB_METACHARINIT();
|
|
if (f <= 0) {
|
|
/*
|
|
* Original string is at least as wide as padding.
|
|
* Truncate original string to width.
|
|
*/
|
|
for (c = postnum; c > 0; ) {
|
|
cl = MB_METACHARLENCONV(str, &cchar);
|
|
if (!cl)
|
|
break;
|
|
while (cl--)
|
|
*r++ = *str++;
|
|
c -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
} else {
|
|
/*
|
|
* There's some space to fill. First copy the original
|
|
* string, counting the width. Make sure we copy the
|
|
* entire string.
|
|
*/
|
|
for (c = ls; *str; ) {
|
|
cl = MB_METACHARLENCONV(str, &cchar);
|
|
if (!cl)
|
|
break;
|
|
while (cl--)
|
|
*r++ = *str++;
|
|
c -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
MB_METACHARINIT();
|
|
if (f <= lpostone) {
|
|
if (postone) {
|
|
/*
|
|
* Not enough or only just enough space to fit
|
|
* the unrepeated string. Truncate as necessary.
|
|
*/
|
|
for (c = f; c > 0; ) {
|
|
cl = MB_METACHARLENCONV(postone, &cchar);
|
|
if (!cl)
|
|
break;
|
|
while (cl--)
|
|
*r++ = *postone++;
|
|
c -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
}
|
|
} else {
|
|
if (postone) {
|
|
f -= lpostone;
|
|
/* Copy the entire unrepeated string */
|
|
for (c = lpostone; *postone; ) {
|
|
cl = MB_METACHARLENCONV(postone, &cchar);
|
|
if (!cl)
|
|
break;
|
|
while (cl--)
|
|
*r++ = *postone++;
|
|
c -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
}
|
|
if (lpostmul) {
|
|
/* Repeat the repeated string */
|
|
for (cc = f / lpostmul; cc--;) {
|
|
MB_METACHARINIT();
|
|
for (c = lpostmul, t = postmul; *t; ) {
|
|
cl = MB_METACHARLENCONV(t, &cchar);
|
|
if (!cl)
|
|
break;
|
|
while (cl--)
|
|
*r++ = *t++;
|
|
c -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
}
|
|
/*
|
|
* See if there's any fraction of the repeated
|
|
* string needed to fill up the remaining space.
|
|
*/
|
|
if ((m = f % lpostmul)) {
|
|
MB_METACHARINIT();
|
|
while (m > 0) {
|
|
cl = MB_METACHARLENCONV(postmul, &cchar);
|
|
if (!cl)
|
|
break;
|
|
while (cl--)
|
|
*r++ = *postmul++;
|
|
m -= WCPADWIDTH(cchar, multi_width);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
*r = '\0';
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
/*
|
|
* Look for a delimited portion of a string. The first (possibly
|
|
* multibyte) character at s is the delimiter. Various forms
|
|
* of brackets are treated separately, as documented.
|
|
*
|
|
* Returns a pointer to the final delimiter. Sets *len to the
|
|
* length of the final delimiter; a NULL causes *len to be set
|
|
* to zero since we shouldn't advance past it. (The string is
|
|
* tokenized, so a NULL is a real end of string.)
|
|
*/
|
|
|
|
/**/
|
|
char *
|
|
get_strarg(char *s, int *lenp)
|
|
{
|
|
convchar_t del;
|
|
int len;
|
|
char ctok = 0;
|
|
|
|
MB_METACHARINIT();
|
|
len = MB_METACHARLENCONV(s, &del);
|
|
if (!len) {
|
|
*lenp = 0;
|
|
return s;
|
|
}
|
|
|
|
#ifdef MULTIBYTE_SUPPORT
|
|
if (del == WEOF)
|
|
del = (wint_t)((*s == Meta) ? s[1] ^ 32 : *s);
|
|
#endif
|
|
s += len;
|
|
switch (del) {
|
|
case ZWC('('):
|
|
del = ZWC(')');
|
|
break;
|
|
case '[':
|
|
del = ZWC(']');
|
|
break;
|
|
case '{':
|
|
del = ZWC('}');
|
|
break;
|
|
case '<':
|
|
del = ZWC('>');
|
|
break;
|
|
case Inpar:
|
|
ctok = Outpar;
|
|
break;
|
|
case Inang:
|
|
ctok = Outang;
|
|
break;
|
|
case Inbrace:
|
|
ctok = Outbrace;
|
|
break;
|
|
case Inbrack:
|
|
ctok = Outbrack;
|
|
break;
|
|
}
|
|
|
|
if (ctok) {
|
|
/*
|
|
* Looking for a matching token; we want the literal byte,
|
|
* not a decoded multibyte character, so search specially.
|
|
*/
|
|
while (*s && *s != ctok)
|
|
s++;
|
|
} else {
|
|
convchar_t del2;
|
|
len = 0;
|
|
while (*s) {
|
|
len = MB_METACHARLENCONV(s, &del2);
|
|
#ifdef MULTIBYTE_SUPPORT
|
|
if (del2 == WEOF)
|
|
del2 = (wint_t)((*s == Meta) ? s[1] ^ 32 : *s);
|
|
#endif
|
|
if (del == del2)
|
|
break;
|
|
s += len;
|
|
}
|
|
}
|
|
|
|
*lenp = len;
|
|
return s;
|
|
}
|
|
|
|
/*
|
|
* Get an integer argument; update *s to the end of the
|
|
* final delimiter. *delmatchp is set to the length of the
|
|
* matched delimiter if we have matching, delimiters and there was no error in
|
|
* the evaluation, else 0.
|
|
*/
|
|
|
|
/**/
|
|
static int
|
|
get_intarg(char **s, int *delmatchp)
|
|
{
|
|
int arglen;
|
|
char *t = get_strarg(*s, &arglen);
|
|
char *p, sav;
|
|
zlong ret;
|
|
|
|
*delmatchp = 0;
|
|
if (!*t)
|
|
return -1;
|
|
sav = *t;
|
|
*t = '\0';
|
|
p = dupstring(*s + arglen);
|
|
*s = t + arglen;
|
|
*t = sav;
|
|
if (parsestr(&p))
|
|
return -1;
|
|
singsub(&p);
|
|
if (errflag)
|
|
return -1;
|
|
ret = mathevali(p);
|
|
if (errflag)
|
|
return -1;
|
|
if (ret < 0)
|
|
ret = -ret;
|
|
*delmatchp = arglen;
|
|
return ret;
|
|
}
|
|
|
|
/* Parsing for the (e) flag. */
|
|
|
|
static int
|
|
subst_parse_str(char **sp, int single, int err)
|
|
{
|
|
char *s;
|
|
|
|
*sp = s = dupstring(*sp);
|
|
|
|
if (!(err ? parsestr(&s) : parsestrnoerr(&s))) {
|
|
*sp = s;
|
|
if (!single) {
|
|
int qt = 0;
|
|
|
|
for (; *s; s++) {
|
|
if (!qt) {
|
|
if (*s == Qstring)
|
|
*s = String;
|
|
else if (*s == Qtick)
|
|
*s = Tick;
|
|
}
|
|
if (*s == Dnull)
|
|
qt = !qt;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/* Evaluation for (#) flag */
|
|
|
|
static char *
|
|
substevalchar(char *ptr)
|
|
{
|
|
zlong ires = mathevali(ptr);
|
|
int len = 0;
|
|
|
|
if (errflag)
|
|
return NULL;
|
|
#ifdef MULTIBYTE_SUPPORT
|
|
if (isset(MULTIBYTE) && ires > 127) {
|
|
/* '\\' + 'U' + 8 bytes of character + '\0' */
|
|
char buf[11];
|
|
|
|
/* inefficient: should separate out \U handling from getkeystring */
|
|
sprintf(buf, "\\U%.8x", (unsigned int)ires & 0xFFFFFFFFu);
|
|
ptr = getkeystring(buf, &len, GETKEYS_BINDKEY, NULL);
|
|
}
|
|
if (len == 0)
|
|
#endif
|
|
{
|
|
ptr = zhalloc(2);
|
|
len = 1;
|
|
sprintf(ptr, "%c", (int)ires);
|
|
}
|
|
return metafy(ptr, len, META_USEHEAP);
|
|
}
|
|
|
|
/*
|
|
* Helper function for arguments to parameter flags which
|
|
* handles the (p) and (~) flags as escapes and tok_arg respectively.
|
|
*/
|
|
|
|
static char *
|
|
untok_and_escape(char *s, int escapes, int tok_arg)
|
|
{
|
|
int klen;
|
|
char *dst = NULL;
|
|
|
|
if (escapes && (*s == String || *s == Qstring) && s[1]) {
|
|
char *pstart = s+1, *pend;
|
|
for (pend = pstart; *pend; pend++)
|
|
if (!iident(*pend))
|
|
break;
|
|
if (!*pend) {
|
|
dst = dupstring(getsparam(pstart));
|
|
}
|
|
}
|
|
if (dst == NULL) {
|
|
untokenize(dst = dupstring(s));
|
|
if (escapes) {
|
|
dst = getkeystring(dst, &klen, GETKEYS_SEP, NULL);
|
|
dst = metafy(dst, klen, META_HREALLOC);
|
|
}
|
|
}
|
|
if (tok_arg)
|
|
shtokenize(dst);
|
|
return dst;
|
|
}
|
|
|
|
/*
|
|
* See if an argument str looks like a subscript or length following
|
|
* a colon and parse it. It must be followed by a ':' or nothing.
|
|
* If this succeeds, expand and return the evaulated expression if
|
|
* found, else return NULL.
|
|
*
|
|
* We assume this is what is meant if the first character is not
|
|
* an alphabetic character or '&', which signify modifiers.
|
|
*
|
|
* Set *endp to point to the next character following.
|
|
*/
|
|
static char *
|
|
check_colon_subscript(char *str, char **endp)
|
|
{
|
|
int sav;
|
|
|
|
/* Could this be a modifier (or empty)? */
|
|
if (!*str || ialpha(*str) || *str == '&')
|
|
return NULL;
|
|
|
|
*endp = parse_subscript(str, 0, ':');
|
|
if (!*endp) {
|
|
/* No trailing colon? */
|
|
*endp = parse_subscript(str, 0, '\0');
|
|
if (!*endp)
|
|
return NULL;
|
|
}
|
|
sav = **endp;
|
|
**endp = '\0';
|
|
str = dupstring(str);
|
|
if (parsestr(&str))
|
|
return NULL;
|
|
singsub(&str);
|
|
remnulargs(str);
|
|
untokenize(str);
|
|
|
|
**endp = sav;
|
|
return str;
|
|
}
|
|
|
|
/* parameter substitution */
|
|
|
|
#define isstring(c) ((c) == '$' || (char)(c) == String || (char)(c) == Qstring)
|
|
#define isbrack(c) ((c) == '[' || (char)(c) == Inbrack)
|
|
|
|
/*
|
|
* Given a linked list l with node n, perform parameter substitution
|
|
* starting from *str. Return the node with the substitutuion performed
|
|
* or NULL if it failed.
|
|
*
|
|
* If qt is true, the `$' was quoted. TODO: why can't we just look
|
|
* to see if the first character was String or Qstring?
|
|
*
|
|
* If ssub is true, we are being called via singsubst(), which means
|
|
* the result will be a single word. TODO: can we generate the
|
|
* single word at the end? TODO: if not, or maybe in any case,
|
|
* can we pass down the ssub flag from prefork with the other flags
|
|
* instead of pushing it into different arguments? (How exactly
|
|
* to qt and ssub differ? Are both necessary, if so is there some
|
|
* better way of separating the two?)
|
|
*/
|
|
|
|
/**/
|
|
static LinkNode
|
|
paramsubst(LinkList l, LinkNode n, char **str, int qt, int pf_flags,
|
|
int *ret_flags)
|
|
{
|
|
char *aptr = *str, c, cc;
|
|
char *s = aptr, *fstr, *idbeg, *idend, *ostr = (char *) getdata(n);
|
|
int colf; /* != 0 means we found a colon after the name */
|
|
/*
|
|
* There are far too many flags. They need to be grouped
|
|
* together into some structure which ties them to where they
|
|
* came from.
|
|
*
|
|
* Some flags have a an obscure relationship to their effect which
|
|
* depends on incrementing them to particular values in particular
|
|
* ways.
|
|
*/
|
|
/*
|
|
* Whether the value is an array (in aval) or not (in val). There's
|
|
* a movement from storing the value in the stuff read from the
|
|
* parameter (the value v) to storing them in val and aval.
|
|
* However, sometimes you find v reappearing temporarily.
|
|
*
|
|
* The values -1 and 2 are special to isarr. The value -1 is used
|
|
* to force us to keep an empty array. It's tested in the YUK chunk
|
|
* (I mean the one explicitly marked as such). The value 2
|
|
* indicates an array has come from splitting a scalar. We use
|
|
* that to override the usual rule that in double quotes we don't
|
|
* remove empty elements (so "${(s.:):-foo::bar}" produces two
|
|
* words). This seems to me to be quite the wrong thing to do,
|
|
* but it looks like code may be relying on it. So we require (@)
|
|
* as well before we keep the empty fields (look for assignments
|
|
* like "isarr = nojoin ? 1 : 2").
|
|
*/
|
|
int isarr = 0;
|
|
/*
|
|
* This is just the setting of the option except we need to
|
|
* take account of ^ and ^^.
|
|
*/
|
|
int plan9 = isset(RCEXPANDPARAM);
|
|
/*
|
|
* Likwise, but with ~ and ~~. Also, we turn it off later
|
|
* on if qt is passed down. The value can go to 2 if we
|
|
* use ~ to force this on.
|
|
*/
|
|
int globsubst = isset(GLOBSUBST);
|
|
/*
|
|
* Indicates ${(#)...}.
|
|
*/
|
|
int evalchar = 0;
|
|
/*
|
|
* Indicates ${#pm}, massaged by whichlen which is set by
|
|
* the (c), (w), and (W) flags to indicate how we take the length.
|
|
*/
|
|
int getlen = 0;
|
|
int whichlen = 0;
|
|
/*
|
|
* Indicates ${+pm}: a simple boolean for once.
|
|
*/
|
|
int chkset = 0;
|
|
/*
|
|
* Indicates we have tried to get a value in v but that was
|
|
* unset. I don't quite understand why (v == NULL) isn't
|
|
* good enough, but there are places where we seem to need
|
|
* to second guess whether a value is a real value or not.
|
|
* See in particular the (colf && !vunset) test below.
|
|
*/
|
|
int vunset = 0;
|
|
/*
|
|
* Indicates (t) flag, i.e. print out types. The code for
|
|
* this actually isn't too horrifically inbred compared with
|
|
* that for (P).
|
|
*/
|
|
int wantt = 0;
|
|
/*
|
|
* Indicates spliting a string into an array. There aren't
|
|
* actually that many special cases for this --- which may
|
|
* be why it doesn't work properly; we split in some cases
|
|
* where we shouldn't, in particular on the multsubs for
|
|
* handling embedded values for ${...=...} and the like.
|
|
*/
|
|
int spbreak = (pf_flags & PREFORK_SHWORDSPLIT) &&
|
|
!(pf_flags & PREFORK_SINGLE) && !qt;
|
|
/* Scalar and array value, see isarr above */
|
|
char *val = NULL, **aval = NULL;
|
|
/*
|
|
* vbuf and v are both used to retrieve parameter values; this
|
|
* is a kludge, we pass down vbuf and it may or may not return v.
|
|
*/
|
|
struct value vbuf;
|
|
Value v = NULL;
|
|
/*
|
|
* This expressive name refers to the set of flags which
|
|
* is applied to matching for #, %, / and their doubled variants:
|
|
* (M), (R), (B), (E), (N), (S).
|
|
*/
|
|
int flags = 0;
|
|
/* Value from (I) flag, used for ditto. */
|
|
int flnum = 0;
|
|
/*
|
|
* sortit is to be passed to strmetasort().
|
|
* indord is the (a) flag, which for consistency doesn't get
|
|
* combined into sortit.
|
|
*/
|
|
int sortit = SORTIT_ANYOLDHOW, indord = 0;
|
|
/* (u): straightforward. */
|
|
int unique = 0;
|
|
/* combination of (L), (U) and (C) flags. */
|
|
int casmod = CASMOD_NONE;
|
|
/*
|
|
* quotemod says we are doing either (q/b) (positive), (Q) (negative)
|
|
* or not (0). quotetype counts the q's for the first case.
|
|
* quoterr is simply (X) but gets passed around a lot because the
|
|
* combination (eX) needs it.
|
|
*/
|
|
int quotemod = 0, quotetype = QT_NONE, quoteerr = 0;
|
|
/*
|
|
* Various fairly straightforward modifications, except that as with so
|
|
* many flags it's not easy to decide where to put them in the order.
|
|
* bit 0: (D) flag.
|
|
* bit 1: (V) flag.
|
|
*/
|
|
int mods = 0;
|
|
/*
|
|
* The (z) flag, nothing to do with SH_WORD_SPLIT which is tied
|
|
* spbreak, see above; fairly straighforward in use but c.f.
|
|
* the comment for mods.
|
|
*
|
|
* This gets set to one of the LEXFLAGS_* values.
|
|
*/
|
|
int shsplit = 0;
|
|
/*
|
|
* "ssub" is true when we are called from singsub (via prefork):
|
|
* it means that we must join arrays and should not split words.
|
|
*/
|
|
int ssub = (pf_flags & PREFORK_SINGLE);
|
|
/*
|
|
* The separator from (j) and (s) respectively, or (F) and (f)
|
|
* respectively (hardwired to "\n" in that case). Slightly
|
|
* confusingly also used for ${#pm}, thought that's at least
|
|
* documented in the manual
|
|
*/
|
|
char *sep = NULL, *spsep = NULL;
|
|
/*
|
|
* Padding strings. The left and right padding strings which
|
|
* are repeated, then the ones which only occur once, for
|
|
* the (l) and (r) flags.
|
|
*/
|
|
char *premul = NULL, *postmul = NULL, *preone = NULL, *postone = NULL;
|
|
/* Replacement string for /orig/repl and //orig/repl */
|
|
char *replstr = NULL;
|
|
/* The numbers for (l) and (r) */
|
|
zlong prenum = 0, postnum = 0;
|
|
#ifdef MULTIBYTE_SUPPORT
|
|
/* The (m) flag: use width of multibyte characters */
|
|
int multi_width = 0;
|
|
#endif
|
|
/*
|
|
* Whether the value has been copied. Optimisation: if we
|
|
* are modifying an expression, we only need to copy it the
|
|
* first time, and if we don't modify it we can just use the
|
|
* value from the parameter or input.
|
|
*/
|
|
int copied = 0;
|
|
/*
|
|
* The (A) flag for array assignment, with consequences for
|
|
* splitting and joining; (AA) gives arrasg == 2 for associative
|
|
* arrays.
|
|
*/
|
|
int arrasg = 0;
|
|
/*
|
|
* The (e) flag. As we need to do extra work not quite
|
|
* at the end, the effect of this is kludged in several places.
|
|
*/
|
|
int eval = 0;
|
|
/*
|
|
* The (P) flag. This interacts a bit obscurely with whether
|
|
* or not we are dealing with a sub expression (subexp).
|
|
*/
|
|
int aspar = 0;
|
|
/*
|
|
* The (%) flag, c.f. mods again.
|
|
*/
|
|
int presc = 0;
|
|
/*
|
|
* The (g) flag. Process escape sequences with various GETKEY_ flags.
|
|
*/
|
|
int getkeys = -1;
|
|
/*
|
|
* The (@) flag; interacts obscurely with qt and isarr.
|
|
* This is one of the things that decides whether multsub
|
|
* will produce an array, but in an extremely indirect fashion.
|
|
*/
|
|
int nojoin = (pf_flags & PREFORK_SHWORDSPLIT) ? !(ifs && *ifs) && !qt : 0;
|
|
/*
|
|
* != 0 means ${...}, otherwise $... What works without braces
|
|
* is largely a historical artefact (everything works with braces,
|
|
* I sincerely hope).
|
|
*/
|
|
char inbrace = 0;
|
|
/*
|
|
* Use for the (k) flag. Goes down into the parameter code,
|
|
* sometimes.
|
|
*/
|
|
char hkeys = 0;
|
|
/*
|
|
* Used for the (v) flag, ditto. Not quite sure why they're
|
|
* separate, but the tradition seems to be that things only
|
|
* get combined when that makes the result more obscure rather
|
|
* than less.
|
|
*/
|
|
char hvals = 0;
|
|
/*
|
|
* Whether we had to evaluate a subexpression, i.e. an
|
|
* internal ${...} or $(...) or plain $pm. We almost don't
|
|
* need to remember this (which would be neater), but the (P)
|
|
* flag means the subexp and !subexp code is obscurely combined,
|
|
* and the argument passing to fetchvalue has another kludge.
|
|
*/
|
|
int subexp;
|
|
/*
|
|
* If we're referring to the positional parameters, then
|
|
* e.g ${*:1:1} refers to $1.
|
|
* This is for compatibility.
|
|
*/
|
|
int horrible_offset_hack = 0;
|
|
/*
|
|
* Signal back from multsub: with something like
|
|
* x${:- $foo}
|
|
* with word-splitting active we need to split on that leading
|
|
* whitespace. However, if there's no "x" the whitespace is
|
|
* simply removed.
|
|
*/
|
|
int ms_flags = 0;
|
|
/*
|
|
* We need to do an extra fetch to honour the (P) flag.
|
|
* Complicated by the use of subexpressions that may have
|
|
* nested (P) flags.
|
|
*/
|
|
int fetch_needed;
|
|
|
|
*s++ = '\0';
|
|
/*
|
|
* Nothing to do unless the character following the $ is
|
|
* something we recognise.
|
|
*
|
|
* Shouldn't this be a table or something? We test for all
|
|
* these later on, too.
|
|
*/
|
|
c = *s;
|
|
if (itype_end(s, IIDENT, 1) == s && *s != '#' && c != Pound &&
|
|
c != '-' && c != '!' && c != '$' && c != String && c != Qstring &&
|
|
c != '?' && c != Quest &&
|
|
c != '*' && c != Star && c != '@' && c != '{' &&
|
|
c != Inbrace && c != '=' && c != Equals && c != Hat &&
|
|
c != '^' && c != '~' && c != Tilde && c != '+') {
|
|
s[-1] = '$';
|
|
*str = s;
|
|
return n;
|
|
}
|
|
DPUTS(c == '{', "BUG: inbrace == '{' in paramsubst()");
|
|
/*
|
|
* Extra processing if there is an opening brace: mostly
|
|
* flags in parentheses, but also one ksh hack.
|
|
*/
|
|
if (c == Inbrace) {
|
|
inbrace = 1;
|
|
s++;
|
|
/*
|
|
* In ksh emulation a leading `!' is a special flag working
|
|
* sort of like our (k).
|
|
* TODO: this is one of very few cases tied directly to
|
|
* the emulation mode rather than an option. Since ksh
|
|
* doesn't have parameter flags it might be neater to
|
|
* handle this with the ^, =, ~ stuff, below.
|
|
*/
|
|
if ((c = *s) == '!' && s[1] != Outbrace && EMULATION(EMULATE_KSH)) {
|
|
hkeys = SCANPM_WANTKEYS;
|
|
s++;
|
|
} else if (c == '(' || c == Inpar) {
|
|
char *t, sav;
|
|
int tt = 0;
|
|
zlong num;
|
|
/*
|
|
* The (p) flag is only remembered within
|
|
* this block. It says we do print-style handling
|
|
* on the values for flags, but only on those.
|
|
*/
|
|
int escapes = 0;
|
|
/*
|
|
* '~' in parentheses caused tokenization of string arg:
|
|
* similar to (p).
|
|
*/
|
|
int tok_arg = 0;
|
|
|
|
for (s++; (c = *s) != ')' && c != Outpar; s++, tt = 0) {
|
|
int arglen; /* length of modifier argument */
|
|
int dellen; /* length of matched delimiter, 0 if not */
|
|
char *del0; /* pointer to initial delimiter */
|
|
|
|
switch (c) {
|
|
case ')':
|
|
case Outpar:
|
|
/* how can this happen? */
|
|
break;
|
|
case '~':
|
|
case Tilde:
|
|
tok_arg = !tok_arg;
|
|
break;
|
|
case 'A':
|
|
++arrasg;
|
|
break;
|
|
case '@':
|
|
nojoin = 2; /* nojoin = 2 means force */
|
|
break;
|
|
case 'M':
|
|
flags |= SUB_MATCH;
|
|
break;
|
|
case 'R':
|
|
flags |= SUB_REST;
|
|
break;
|
|
case 'B':
|
|
flags |= SUB_BIND;
|
|
break;
|
|
case 'E':
|
|
flags |= SUB_EIND;
|
|
break;
|
|
case 'N':
|
|
flags |= SUB_LEN;
|
|
break;
|
|
case 'S':
|
|
flags |= SUB_SUBSTR;
|
|
break;
|
|
case 'I':
|
|
s++;
|
|
flnum = get_intarg(&s, &dellen);
|
|
if (flnum < 0)
|
|
goto flagerr;
|
|
s--;
|
|
break;
|
|
|
|
case 'L':
|
|
casmod = CASMOD_LOWER;
|
|
break;
|
|
case 'U':
|
|
casmod = CASMOD_UPPER;
|
|
break;
|
|
case 'C':
|
|
casmod = CASMOD_CAPS;
|
|
break;
|
|
|
|
case 'o':
|
|
if (!sortit)
|
|
sortit |= SORTIT_SOMEHOW; /* sort, no modifiers */
|
|
break;
|
|
case 'O':
|
|
sortit |= SORTIT_BACKWARDS;
|
|
break;
|
|
case 'i':
|
|
sortit |= SORTIT_IGNORING_CASE;
|
|
break;
|
|
case 'n':
|
|
sortit |= SORTIT_NUMERICALLY;
|
|
break;
|
|
case 'a':
|
|
sortit |= SORTIT_SOMEHOW;
|
|
indord = 1;
|
|
break;
|
|
|
|
case 'D':
|
|
mods |= 1;
|
|
break;
|
|
case 'V':
|
|
mods |= 2;
|
|
break;
|
|
|
|
case 'q':
|
|
if (quotetype == QT_DOLLARS ||
|
|
quotetype == QT_BACKSLASH_PATTERN)
|
|
goto flagerr;
|
|
if (s[1] == '-' || s[1] == '+') {
|
|
if (quotemod)
|
|
goto flagerr;
|
|
s++;
|
|
quotemod = 1;
|
|
quotetype = (*s == '-') ? QT_SINGLE_OPTIONAL :
|
|
QT_QUOTEDZPUTS;
|
|
} else {
|
|
if (quotetype == QT_SINGLE_OPTIONAL) {
|
|
/* extra q's after '-' not allowed */
|
|
goto flagerr;
|
|
}
|
|
quotemod++, quotetype++;
|
|
}
|
|
break;
|
|
case 'b':
|
|
if (quotemod || quotetype != QT_NONE)
|
|
goto flagerr;
|
|
quotemod = 1;
|
|
quotetype = QT_BACKSLASH_PATTERN;
|
|
break;
|
|
case 'Q':
|
|
quotemod--;
|
|
break;
|
|
case 'X':
|
|
quoteerr = 1;
|
|
break;
|
|
|
|
case 'e':
|
|
eval = 1;
|
|
break;
|
|
case 'P':
|
|
aspar = 1;
|
|
break;
|
|
|
|
case 'c':
|
|
whichlen = 1;
|
|
break;
|
|
case 'w':
|
|
whichlen = 2;
|
|
break;
|
|
case 'W':
|
|
whichlen = 3;
|
|
break;
|
|
|
|
case 'f':
|
|
spsep = "\n";
|
|
break;
|
|
case 'F':
|
|
sep = "\n";
|
|
break;
|
|
|
|
case '0':
|
|
spsep = zhalloc(3);
|
|
spsep[0] = Meta;
|
|
spsep[1] = '\0' ^ 32;
|
|
spsep[2] = '\0';
|
|
break;
|
|
|
|
case 's':
|
|
tt = 1;
|
|
/* fall through */
|
|
case 'j':
|
|
t = get_strarg(++s, &arglen);
|
|
if (*t) {
|
|
sav = *t;
|
|
*t = '\0';
|
|
if (tt)
|
|
spsep = untok_and_escape(s + arglen,
|
|
escapes, tok_arg);
|
|
else
|
|
sep = untok_and_escape(s + arglen,
|
|
escapes, tok_arg);
|
|
*t = sav;
|
|
s = t + arglen - 1;
|
|
} else
|
|
goto flagerr;
|
|
break;
|
|
|
|
case 'l':
|
|
tt = 1;
|
|
/* fall through */
|
|
case 'r':
|
|
s++;
|
|
/* delimiter position */
|
|
del0 = s;
|
|
num = get_intarg(&s, &dellen);
|
|
if (num < 0)
|
|
goto flagerr;
|
|
if (tt)
|
|
prenum = num;
|
|
else
|
|
postnum = num;
|
|
/* must have same delimiter if more arguments */
|
|
if (!dellen || memcmp(del0, s, dellen)) {
|
|
/* decrement since loop will increment */
|
|
s--;
|
|
break;
|
|
}
|
|
t = get_strarg(s, &arglen);
|
|
if (!*t)
|
|
goto flagerr;
|
|
sav = *t;
|
|
*t = '\0';
|
|
if (tt)
|
|
premul = untok_and_escape(s + arglen, escapes,
|
|
tok_arg);
|
|
else
|
|
postmul = untok_and_escape(s + arglen, escapes,
|
|
tok_arg);
|
|
*t = sav;
|
|
sav = *s;
|
|
s = t + arglen;
|
|
/* again, continue only if another start delimiter */
|
|
if (memcmp(del0, s, dellen)) {
|
|
/* decrement since loop will increment */
|
|
s--;
|
|
break;
|
|
}
|
|
t = get_strarg(s, &arglen);
|
|
if (!*t)
|
|
goto flagerr;
|
|
sav = *t;
|
|
*t = '\0';
|
|
if (tt)
|
|
preone = untok_and_escape(s + arglen,
|
|
escapes, tok_arg);
|
|
else
|
|
postone = untok_and_escape(s + arglen,
|
|
escapes, tok_arg);
|
|
*t = sav;
|
|
/* -1 since loop will increment */
|
|
s = t + arglen - 1;
|
|
break;
|
|
|
|
case 'm':
|
|
#ifdef MULTIBYTE_SUPPORT
|
|
multi_width++;
|
|
#endif
|
|
break;
|
|
|
|
case 'p':
|
|
escapes = 1;
|
|
break;
|
|
|
|
case 'k':
|
|
hkeys = SCANPM_WANTKEYS;
|
|
break;
|
|
case 'v':
|
|
hvals = SCANPM_WANTVALS;
|
|
break;
|
|
|
|
case 't':
|
|
wantt = 1;
|
|
break;
|
|
|
|
case '%':
|
|
presc++;
|
|
break;
|
|
|
|
case 'g':
|
|
t = get_strarg(++s, &arglen);
|
|
if (getkeys < 0)
|
|
getkeys = 0;
|
|
if (*t) {
|
|
sav = *t;
|
|
*t = 0;
|
|
while (*++s) {
|
|
switch (*s) {
|
|
case 'e':
|
|
getkeys |= GETKEY_EMACS;
|
|
break;
|
|
case 'o':
|
|
getkeys |= GETKEY_OCTAL_ESC;
|
|
break;
|
|
case 'c':
|
|
getkeys |= GETKEY_CTRL;
|
|
break;
|
|
|
|
default:
|
|
*t = sav;
|
|
goto flagerr;
|
|
}
|
|
}
|
|
*t = sav;
|
|
s = t + arglen - 1;
|
|
} else
|
|
goto flagerr;
|
|
break;
|
|
|
|
case 'z':
|
|
shsplit = LEXFLAGS_ACTIVE;
|
|
break;
|
|
|
|
case 'Z':
|
|
t = get_strarg(++s, &arglen);
|
|
if (*t) {
|
|
sav = *t;
|
|
*t = 0;
|
|
while (*++s) {
|
|
switch (*s) {
|
|
case 'c':
|
|
/* Parse and keep comments */
|
|
shsplit |= LEXFLAGS_COMMENTS_KEEP;
|
|
break;
|
|
|
|
case 'C':
|
|
/* Parse and remove comments */
|
|
shsplit |= LEXFLAGS_COMMENTS_STRIP;
|
|
break;
|
|
|
|
case 'n':
|
|
/* Treat newlines as whitespace */
|
|
shsplit |= LEXFLAGS_NEWLINE;
|
|
break;
|
|
|
|
default:
|
|
*t = sav;
|
|
goto flagerr;
|
|
}
|
|
}
|
|
*t = sav;
|
|
s = t + arglen - 1;
|
|
} else
|
|
goto flagerr;
|
|
break;
|
|
|
|
case 'u':
|
|
unique = 1;
|
|
break;
|
|
|
|
case '#':
|
|
case Pound:
|
|
evalchar = 1;
|
|
break;
|
|
|
|
case '_':
|
|
t = get_strarg(++s, &arglen);
|
|
if (*t) {
|
|
sav = *t;
|
|
*t = 0;
|
|
while (*++s) {
|
|
/* Reserved for future use */
|
|
switch (*s) {
|
|
default:
|
|
*t = sav;
|
|
goto flagerr;
|
|
}
|
|
}
|
|
*t = sav;
|
|
s = t + arglen - 1;
|
|
} else
|
|
goto flagerr;
|
|
break;
|
|
|
|
default:
|
|
flagerr:
|
|
zerr("error in flags");
|
|
return NULL;
|
|
}
|
|
}
|
|
s++;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* premul, postmul specify the padding character to be used
|
|
* multiple times with the (l) and (r) flags respectively.
|
|
*/
|
|
if (!premul)
|
|
premul = " ";
|
|
if (!postmul)
|
|
postmul = " ";
|
|
|
|
/*
|
|
* Look for special unparenthesised flags.
|
|
* TODO: could make these able to appear inside parentheses, too,
|
|
* i.e. ${(^)...} etc.
|
|
*/
|
|
for (;;) {
|
|
if ((c = *s) == '^' || c == Hat) {
|
|
/* RC_EXPAND_PARAM on or off (doubled )*/
|
|
if ((c = *++s) == '^' || c == Hat) {
|
|
plan9 = 0;
|
|
s++;
|
|
} else
|
|
plan9 = 1;
|
|
} else if ((c = *s) == '=' || c == Equals) {
|
|
/* SH_WORD_SPLIT on or off (doubled). spbreak = 2 means force */
|
|
if ((c = *++s) == '=' || c == Equals) {
|
|
spbreak = 0;
|
|
if (nojoin < 2)
|
|
nojoin = 0;
|
|
s++;
|
|
} else {
|
|
spbreak = 2;
|
|
if (nojoin < 2)
|
|
nojoin = !(ifs && *ifs);
|
|
}
|
|
} else if ((c == '#' || c == Pound) &&
|
|
(inbrace || !isset(POSIXIDENTIFIERS)) &&
|
|
(itype_end(s+1, IIDENT, 0) != s + 1
|
|
|| (cc = s[1]) == '*' || cc == Star || cc == '@'
|
|
|| cc == '?' || cc == Quest
|
|
|| cc == '$' || cc == String || cc == Qstring
|
|
/*
|
|
* Me And My Squiggle:
|
|
* ${##} is the length of $#, but ${##foo}
|
|
* is $# with a "foo" removed from the start.
|
|
* If someone had defined the *@!@! language
|
|
* properly in the first place we wouldn't
|
|
* have this nonsense.
|
|
*/
|
|
|| ((cc == '#' || cc == Pound) &&
|
|
s[2] == Outbrace)
|
|
|| cc == '-' || (cc == ':' && s[2] == '-')
|
|
|| (isstring(cc) && (s[2] == Inbrace || s[2] == Inpar)))) {
|
|
getlen = 1 + whichlen, s++;
|
|
/*
|
|
* Return the length of the parameter.
|
|
* getlen can be more than 1 to indicate characters (2),
|
|
* words ignoring multiple delimiters (3), words taking
|
|
* account of multiple delimiters. delimiter is in
|
|
* spsep, NULL means $IFS.
|
|
*/
|
|
} else if (c == '~' || c == Tilde) {
|
|
/* GLOB_SUBST (forced) on or off (doubled) */
|
|
if ((c = *++s) == '~' || c == Tilde) {
|
|
globsubst = 0;
|
|
s++;
|
|
} else
|
|
globsubst = 2;
|
|
} else if (c == '+') {
|
|
/*
|
|
* Return whether indicated parameter is set.
|
|
* Try to handle this when parameter is named
|
|
* by (P) (second part of test).
|
|
*/
|
|
if (itype_end(s+1, IIDENT, 0) != s+1 || (aspar && isstring(s[1]) &&
|
|
(s[2] == Inbrace || s[2] == Inpar)))
|
|
chkset = 1, s++;
|
|
else if (!inbrace) {
|
|
/* Special case for `$+' on its own --- leave unmodified */
|
|
*aptr = '$';
|
|
*str = aptr + 1;
|
|
return n;
|
|
} else {
|
|
zerr("bad substitution");
|
|
return NULL;
|
|
}
|
|
} else if (inbrace && inull(*s)) {
|
|
/*
|
|
* Handles things like ${(f)"$(<file)"} by skipping
|
|
* the double quotes. We don't need to know what was
|
|
* actually there; the presence of a String or Qstring
|
|
* is good enough.
|
|
*/
|
|
s++;
|
|
} else
|
|
break;
|
|
}
|
|
/* Don't activate special pattern characters if inside quotes */
|
|
if (qt)
|
|
globsubst = 0;
|
|
|
|
/*
|
|
* At this point, we usually expect a parameter name.
|
|
* However, there may be a nested ${...} or $(...).
|
|
* These say that the parameter itself is somewhere inside,
|
|
* or that there isn't a parameter and we will get the values
|
|
* from a command substitution itself. In either case,
|
|
* the current instance of paramsubst() doesn't fetch a value,
|
|
* it just operates on what gets passed up.
|
|
* (The first ought to have been {...}, reserving ${...}
|
|
* for substituting a value at that point, but it's too late now.)
|
|
*/
|
|
idbeg = s;
|
|
if ((subexp = (inbrace && s[-1] && isstring(*s) &&
|
|
(s[1] == Inbrace || s[1] == Inpar || s[1] == Inparmath)))) {
|
|
int sav;
|
|
int quoted = *s == Qstring;
|
|
int outtok;
|
|
|
|
val = s++;
|
|
switch (*s) {
|
|
case Inbrace:
|
|
outtok = Outbrace;
|
|
break;
|
|
case Inpar:
|
|
outtok = Outpar;
|
|
break;
|
|
case Inparmath:
|
|
outtok = Outparmath;
|
|
break;
|
|
default:
|
|
/* "Can't Happen" (TM) */
|
|
DPUTS(1, "Nested substitution: This Can't Happen (TM)");
|
|
return NULL;
|
|
}
|
|
skipparens(*s, outtok, &s);
|
|
sav = *s;
|
|
*s = 0;
|
|
/*
|
|
* This handles arrays. TODO: this is not the most obscure call to
|
|
* multsub() (see below) but even so it would be nicer to pass down
|
|
* and back the arrayness more rationally. In that case, we should
|
|
* remove the aspar test and extract a value from an array, if
|
|
* necessary, when we handle (P) lower down.
|
|
*/
|
|
if (multsub(&val, PREFORK_SUBEXP, (aspar ? NULL : &aval), &isarr, NULL,
|
|
&ms_flags) && quoted) {
|
|
/* Empty quoted string --- treat as null string, not elided */
|
|
isarr = -1;
|
|
aval = (char **) hcalloc(sizeof(char *));
|
|
aspar = 0;
|
|
} else if (aspar)
|
|
idbeg = val;
|
|
*s = sav;
|
|
/*
|
|
* This tests for the second double quote in an expression
|
|
* like ${(f)"$(<file)"}, compare above.
|
|
*/
|
|
while (inull(*s))
|
|
s++;
|
|
if (ms_flags & MULTSUB_PARAM_NAME) {
|
|
/*
|
|
* Downbelow has told us this is a parameter name, e.g.
|
|
* ${${(P)name}...}. We're going to behave as if
|
|
* we have exactly that name followed by the rest of
|
|
* the parameter for subscripting etc.
|
|
*
|
|
* See below for where we set the flag in the nested
|
|
* substitution.
|
|
*/
|
|
if (isarr) {
|
|
if (aval[0] && aval[1]) {
|
|
zerr("parameter name reference used with array");
|
|
return NULL;
|
|
}
|
|
val = aval[0];
|
|
isarr = 0;
|
|
}
|
|
s = dyncat(val, s);
|
|
/* Now behave po-faced as if it was always like that... */
|
|
subexp = 0;
|
|
/*
|
|
* If this is a (P) (first test) and at the top level
|
|
* (second test) we can't rely on the caller fetching
|
|
* the result from the pending aspar. So do it below.
|
|
*/
|
|
fetch_needed = aspar && !(pf_flags & PREFORK_SUBEXP);
|
|
} else
|
|
fetch_needed = 0; /* any initial aspar fetch already done */
|
|
v = (Value) NULL;
|
|
} else
|
|
fetch_needed = aspar; /* aspar fetch still needed */
|
|
if (fetch_needed) {
|
|
/*
|
|
* No subexpression, but in any case the value is going
|
|
* to give us the name of a parameter on which we do
|
|
* our remaining processing. In other words, this
|
|
* makes ${(P)param} work like ${(P)${param}}. (Probably
|
|
* better looked at, this is the basic code for ${(P)param}
|
|
* and it's been kludged into the subexp code because no
|
|
* opportunity for a kludge has been neglected.)
|
|
*/
|
|
if ((v = fetchvalue(&vbuf, &s, 1, (qt ? SCANPM_DQUOTED : 0)))) {
|
|
val = idbeg = getstrvalue(v);
|
|
subexp = 1;
|
|
} else
|
|
vunset = 1;
|
|
}
|
|
if (aspar && (pf_flags & PREFORK_SUBEXP)) {
|
|
/*
|
|
* This is the inner handling for the case referred to above
|
|
* where we have something like ${${(P)name}...}.
|
|
*
|
|
* Treat this as a normal value here; all transformations on
|
|
* result are in outer instance.
|
|
*/
|
|
aspar = 0;
|
|
*ret_flags |= MULTSUB_PARAM_NAME;
|
|
}
|
|
/*
|
|
* We need to retrieve a value either if we haven't already
|
|
* got it from a subexpression, or if the processing so
|
|
* far has just yielded us a parameter name to be processed
|
|
* with (P).
|
|
*/
|
|
if (!subexp || aspar) {
|
|
char *ov = val;
|
|
|
|
/*
|
|
* Second argument: decide whether to use the subexpression or
|
|
* the string next on the line as the parameter name.
|
|
* Third argument: decide how processing for brackets
|
|
* 1 means full processing
|
|
* -1 appears to mean something along the lines of
|
|
* only handle single digits and don't handle brackets.
|
|
* I *think* (but it's really only a guess) that this
|
|
* is used by the test below the wantt handling, so
|
|
* that in certain cases we handle brackets there.
|
|
* 0 would apparently mean something like we know we
|
|
* should have the name of a scalar and we get cross
|
|
* if there's anything present which disagrees with that
|
|
* but you will search fetchvalue() in vain for comments on this.
|
|
* Fourth argument gives flags to do with keys, values, quoting,
|
|
* assigning depending on context and parameter flags.
|
|
*
|
|
* This is the last mention of subexp, so presumably this
|
|
* is what the code which makes sure subexp is set if aspar (the
|
|
* (P) flag) is set. I *think* what's going on here is the
|
|
* second argument is for both input and output: with
|
|
* subexp, we only want the input effect, whereas normally
|
|
* we let fetchvalue set the main string pointer s to
|
|
* the end of the bit it's fetched.
|
|
*/
|
|
if (!(v = fetchvalue(&vbuf, (subexp ? &ov : &s),
|
|
(wantt ? -1 :
|
|
((unset(KSHARRAYS) || inbrace) ? 1 : -1)),
|
|
hkeys|hvals|
|
|
(arrasg ? SCANPM_ASSIGNING : 0)|
|
|
(qt ? SCANPM_DQUOTED : 0))) ||
|
|
(v->pm && (v->pm->node.flags & PM_UNSET)) ||
|
|
(v->flags & VALFLAG_EMPTY))
|
|
vunset = 1;
|
|
|
|
if (wantt) {
|
|
/*
|
|
* Handle the (t) flag: value now becomes the type
|
|
* information for the parameter.
|
|
*/
|
|
if (v && v->pm && !(v->pm->node.flags & PM_UNSET)) {
|
|
int f = v->pm->node.flags;
|
|
|
|
switch (PM_TYPE(f)) {
|
|
case PM_SCALAR: val = "scalar"; break;
|
|
case PM_ARRAY: val = "array"; break;
|
|
case PM_INTEGER: val = "integer"; break;
|
|
case PM_EFLOAT:
|
|
case PM_FFLOAT: val = "float"; break;
|
|
case PM_HASHED: val = "association"; break;
|
|
}
|
|
val = dupstring(val);
|
|
if (v->pm->level)
|
|
val = dyncat(val, "-local");
|
|
if (f & PM_LEFT)
|
|
val = dyncat(val, "-left");
|
|
if (f & PM_RIGHT_B)
|
|
val = dyncat(val, "-right_blanks");
|
|
if (f & PM_RIGHT_Z)
|
|
val = dyncat(val, "-right_zeros");
|
|
if (f & PM_LOWER)
|
|
val = dyncat(val, "-lower");
|
|
if (f & PM_UPPER)
|
|
val = dyncat(val, "-upper");
|
|
if (f & PM_READONLY)
|
|
val = dyncat(val, "-readonly");
|
|
if (f & PM_TAGGED)
|
|
val = dyncat(val, "-tag");
|
|
if (f & PM_TAGGED_LOCAL)
|
|
val = dyncat(val, "-tag_local");
|
|
if (f & PM_EXPORTED)
|
|
val = dyncat(val, "-export");
|
|
if (f & PM_UNIQUE)
|
|
val = dyncat(val, "-unique");
|
|
if (f & PM_HIDE)
|
|
val = dyncat(val, "-hide");
|
|
if (f & PM_HIDEVAL)
|
|
val = dyncat(val, "-hideval");
|
|
if (f & PM_SPECIAL)
|
|
val = dyncat(val, "-special");
|
|
vunset = 0;
|
|
} else
|
|
val = dupstring("");
|
|
|
|
v = NULL;
|
|
isarr = 0;
|
|
}
|
|
}
|
|
/*
|
|
* We get in here two ways; either we need to convert v into
|
|
* the local value system, or we need to get rid of brackets
|
|
* even if there isn't a v.
|
|
*/
|
|
while (v || ((inbrace || (unset(KSHARRAYS) && vunset)) && isbrack(*s))) {
|
|
if (!v) {
|
|
/*
|
|
* Index applied to non-existent parameter; we may or may
|
|
* not have a value to index, however. Create a temporary
|
|
* empty parameter as a trick, and index on that. This
|
|
* usually happens the second time around the loop when
|
|
* we've used up the original parameter value and want to
|
|
* apply a subscript to what's left. However, it's also
|
|
* possible it's got something to do with some of that murky
|
|
* passing of -1's as the third argument to fetchvalue() to
|
|
* inhibit bracket parsing at that stage.
|
|
*/
|
|
Param pm;
|
|
char *os = s;
|
|
|
|
if (!isbrack(*s))
|
|
break;
|
|
if (vunset) {
|
|
val = dupstring("");
|
|
isarr = 0;
|
|
}
|
|
pm = createparam(nulstring, isarr ? PM_ARRAY : PM_SCALAR);
|
|
DPUTS(!pm, "BUG: parameter not created");
|
|
if (isarr)
|
|
pm->u.arr = aval;
|
|
else
|
|
pm->u.str = val;
|
|
v = (Value) hcalloc(sizeof *v);
|
|
v->isarr = isarr;
|
|
v->pm = pm;
|
|
v->end = -1;
|
|
if (getindex(&s, v, qt ? SCANPM_DQUOTED : 0) || s == os)
|
|
break;
|
|
}
|
|
/*
|
|
* This is where we extract a value (we know now we have
|
|
* one) into the local parameters for a scalar (val) or
|
|
* array (aval) value. TODO: move val and aval into
|
|
* a structure with a discriminator. Hope we can make
|
|
* more things array values at this point and dearrayify later.
|
|
* v->isarr tells us whether the stuff from down below looks
|
|
* like an array.
|
|
*
|
|
* I think we get to discard the existing value of isarr
|
|
* here because it's already been taken account of, either
|
|
* in the subexp stuff or immediately above.
|
|
*/
|
|
if ((isarr = v->isarr)) {
|
|
/*
|
|
* No way to get here with v->flags & VALFLAG_INV, so
|
|
* getvaluearr() is called by getarrvalue(); needn't test
|
|
* PM_HASHED.
|
|
*/
|
|
if (v->isarr == SCANPM_WANTINDEX) {
|
|
isarr = v->isarr = 0;
|
|
val = dupstring(v->pm->node.nam);
|
|
} else
|
|
aval = getarrvalue(v);
|
|
} else {
|
|
/* Value retrieved from parameter/subexpression is scalar */
|
|
if (v->pm->node.flags & PM_ARRAY) {
|
|
/*
|
|
* Although the value is a scalar, the parameter
|
|
* itself is an array. Presumably this is due to
|
|
* being quoted, or doing single substitution or something,
|
|
* TODO: we're about to do some definitely stringy
|
|
* stuff, so something like this bit is probably
|
|
* necessary. However, I'd like to leave any
|
|
* necessary joining of arrays until this point
|
|
* to avoid the multsub() horror.
|
|
*/
|
|
|
|
/* arrlen() is expensive, so only compute it if needed. */
|
|
int tmplen = -1;
|
|
|
|
if (v->start < 0) {
|
|
tmplen = arrlen(v->pm->gsu.a->getfn(v->pm));
|
|
v->start += tmplen + ((v->flags & VALFLAG_INV) ? 1 : 0);
|
|
}
|
|
if (!(v->flags & VALFLAG_INV))
|
|
if (v->start < 0 ||
|
|
(tmplen != -1
|
|
? v->start >= tmplen
|
|
: arrlen_le(v->pm->gsu.a->getfn(v->pm), v->start)))
|
|
vunset = 1;
|
|
}
|
|
if (!vunset) {
|
|
/*
|
|
* There really is a value. Padding and case
|
|
* transformations used to be handled here, but
|
|
* are now handled in getstrvalue() for greater
|
|
* consistency. However, we get unexpected effects
|
|
* if we allow them to applied on every call, so
|
|
* set the flag that allows them to be substituted.
|
|
*/
|
|
v->flags |= VALFLAG_SUBST;
|
|
val = getstrvalue(v);
|
|
}
|
|
}
|
|
/* See if this is a reference to the positional parameters. */
|
|
if (v && v->pm && v->pm->gsu.a == &vararray_gsu &&
|
|
(char ***)v->pm->u.data == &pparams)
|
|
horrible_offset_hack = 1;
|
|
else
|
|
horrible_offset_hack = 0;
|
|
/*
|
|
* Finished with the original parameter and its indices;
|
|
* carry on looping to see if we need to do more indexing.
|
|
* This means we final get rid of v in favour of val and
|
|
* aval. We could do with somehow encapsulating the bit
|
|
* where we need v.
|
|
*/
|
|
v = NULL;
|
|
if (!inbrace)
|
|
break;
|
|
}
|
|
/*
|
|
* We're now past the name or subexpression; the only things
|
|
* which can happen now are a closing brace, one of the standard
|
|
* parameter postmodifiers, or a history-style colon-modifier.
|
|
*
|
|
* Again, this duplicates tests for characters we're about to
|
|
* examine properly later on.
|
|
*/
|
|
if (inbrace &&
|
|
(c = *s) != '-' && c != '+' && c != ':' && c != '%' && c != '/' &&
|
|
c != '=' && c != Equals &&
|
|
c != '#' && c != Pound &&
|
|
c != '?' && c != Quest &&
|
|
c != '}' && c != Outbrace) {
|
|
zerr("bad substitution");
|
|
return NULL;
|
|
}
|
|
/*
|
|
* Join arrays up if we're in quotes and there isn't some
|
|
* override such as (@).
|
|
* TODO: hmm, if we're called as part of some recursive
|
|
* substitution do we want to delay this until we get back to
|
|
* the top level? Or is if there's a qt (i.e. this parameter
|
|
* substitution is in quotes) always good enough? Potentially
|
|
* we may be OK by now --- all potential `@'s and subexpressions
|
|
* have been handled, including any [@] index which comes up
|
|
* by virture of v->isarr being set to SCANPM_ISVAR_AT which
|
|
* is now in isarr.
|
|
*
|
|
* However, if we are replacing multsub() with something that
|
|
* doesn't mangle arrays, we may need to delay this step until after
|
|
* the foo:- or foo:= or whatever that causes that. Note the value
|
|
* (string or array) at this point is irrelevant if we are going to
|
|
* be doing that. This would mean // and stuff get applied
|
|
* arraywise even if quoted. That's probably wrong, so maybe
|
|
* this just stays.
|
|
*
|
|
* We do a separate stage of dearrayification in the YUK chunk,
|
|
* I think mostly because of the way we make array or scalar
|
|
* values appear to the caller.
|
|
*/
|
|
if (isarr) {
|
|
if (nojoin)
|
|
isarr = -1;
|
|
if (qt && !getlen && isarr > 0) {
|
|
val = sepjoin(aval, sep, 1);
|
|
isarr = 0;
|
|
}
|
|
}
|
|
|
|
idend = s;
|
|
if (inbrace) {
|
|
/*
|
|
* This is to match a closing double quote in case
|
|
* we didn't have a subexpression, e.g. ${"foo"}.
|
|
* This form is pointless, but logically it ought to work.
|
|
*/
|
|
while (inull(*s))
|
|
s++;
|
|
}
|
|
/*
|
|
* We don't yet know whether a `:' introduces a history-style
|
|
* colon modifier or qualifies something like ${...:=...}.
|
|
* But if we remember the colon here it's easy to check later.
|
|
*/
|
|
if ((colf = *s == ':'))
|
|
s++;
|
|
|
|
|
|
/* fstr is to be the text following the substitution. If we have *
|
|
* braces, we look for it here, else we infer it later on. */
|
|
fstr = s;
|
|
if (inbrace) {
|
|
int bct;
|
|
for (bct = 1; (c = *fstr); fstr++) {
|
|
if (c == Inbrace)
|
|
bct++;
|
|
else if (c == Outbrace && !--bct)
|
|
break;
|
|
}
|
|
|
|
if (bct) {
|
|
noclosebrace:
|
|
zerr("closing brace expected");
|
|
return NULL;
|
|
}
|
|
if (c)
|
|
*fstr++ = '\0';
|
|
}
|
|
|
|
/* Check for ${..?..} or ${..=..} or one of those. *
|
|
* Only works if the name is in braces. */
|
|
|
|
if (inbrace && ((c = *s) == '-' ||
|
|
c == '+' ||
|
|
c == ':' || /* i.e. a doubled colon */
|
|
c == '=' || c == Equals ||
|
|
c == '%' ||
|
|
c == '#' || c == Pound ||
|
|
c == '?' || c == Quest ||
|
|
c == '/')) {
|
|
|
|
/*
|
|
* Default index is 1 if no (I) or (I) gave zero. But
|
|
* why don't we set the default explicitly at the start
|
|
* and massage any passed index where we set flnum anyway?
|
|
*/
|
|
if (!flnum)
|
|
flnum++;
|
|
if (c == '%')
|
|
flags |= SUB_END;
|
|
|
|
/* Check for ${..%%..} or ${..##..} */
|
|
if ((c == '%' || c == '#' || c == Pound) && c == s[1]) {
|
|
s++;
|
|
/* we have %%, not %, or ##, not # */
|
|
flags |= SUB_LONG;
|
|
}
|
|
s++;
|
|
if (s[-1] == '/') {
|
|
char *ptr;
|
|
/*
|
|
* previous flags are irrelevant, except for (S) which
|
|
* indicates shortest substring; else look for longest.
|
|
*/
|
|
flags = (flags & SUB_SUBSTR) ? 0 : SUB_LONG;
|
|
if ((c = *s) == '/') {
|
|
/* doubled, so replace all occurrences */
|
|
flags |= SUB_GLOBAL;
|
|
c = *++s;
|
|
}
|
|
/* Check for anchored substitution */
|
|
if (c == '#' || c == Pound) {
|
|
/*
|
|
* anchor at head: this is the `normal' case in
|
|
* getmatch and we only require the flag if SUB_END
|
|
* is also present.
|
|
*/
|
|
flags |= SUB_START;
|
|
s++;
|
|
}
|
|
if (*s == '%') {
|
|
/* anchor at tail */
|
|
flags |= SUB_END;
|
|
s++;
|
|
}
|
|
if (!(flags & (SUB_START|SUB_END))) {
|
|
/* No anchor, so substring */
|
|
flags |= SUB_SUBSTR;
|
|
}
|
|
/*
|
|
* Find the / marking the end of the search pattern.
|
|
* If there isn't one, we're just going to delete that,
|
|
* i.e. replace it with an empty string.
|
|
*
|
|
* We used to use double backslashes to quote slashes,
|
|
* but actually that was buggy and using a single backslash
|
|
* is easier and more obvious.
|
|
*/
|
|
for (ptr = s; (c = *ptr) && c != '/'; ptr++)
|
|
{
|
|
if ((c == Bnull || c == Bnullkeep || c == '\\') && ptr[1])
|
|
{
|
|
if (ptr[1] == '/')
|
|
chuck(ptr);
|
|
else
|
|
ptr++;
|
|
}
|
|
}
|
|
replstr = (*ptr && ptr[1]) ? ptr+1 : "";
|
|
*ptr = '\0';
|
|
}
|
|
|
|
/* See if this was ${...:-...}, ${...:=...}, etc. */
|
|
if (colf)
|
|
flags |= SUB_ALL;
|
|
/*
|
|
* With no special flags, i.e. just a # or % or whatever,
|
|
* the matched portion is removed and we keep the rest.
|
|
* We also want the rest when we're doing a substitution.
|
|
*/
|
|
if (!(flags & (SUB_MATCH|SUB_REST|SUB_BIND|SUB_EIND|SUB_LEN)))
|
|
flags |= SUB_REST;
|
|
|
|
/*
|
|
* With ":" treat a value as unset if the variable is set but
|
|
* - (array) contains no elements
|
|
* - (scalar) contains an empty string
|
|
*/
|
|
if (colf && !vunset) {
|
|
vunset = (isarr) ? !*aval : !*val || (*val == Nularg && !val[1]);
|
|
vunset *= -1; /* Record that vunset was originally false */
|
|
}
|
|
|
|
switch (s[-1]) {
|
|
case '+':
|
|
if (vunset) {
|
|
val = dupstring("");
|
|
copied = 1;
|
|
isarr = 0;
|
|
break;
|
|
}
|
|
vunset = 1;
|
|
/* Fall Through! */
|
|
case '-':
|
|
if (vunset) {
|
|
int split_flags;
|
|
val = dupstring(s);
|
|
/* If word-splitting is enabled, we ask multsub() to split
|
|
* the substituted string at unquoted whitespace. Then, we
|
|
* turn off spbreak so that no further splitting occurs.
|
|
* This allows a construct such as ${1+"$@"} to correctly
|
|
* keep its array splits, and weird constructs such as
|
|
* ${str+"one two" "3 2 1" foo "$str"} to only be split
|
|
* at the unquoted spaces. */
|
|
if (spbreak) {
|
|
split_flags = PREFORK_SHWORDSPLIT;
|
|
if (!aspar)
|
|
split_flags |= PREFORK_SPLIT;
|
|
} else {
|
|
/*
|
|
* It's not good enough not passing the flag to use
|
|
* SHWORDSPLIT, because when we get to a nested
|
|
* paramsubst we need to ignore isset(SHWORDSPLIT).
|
|
*/
|
|
split_flags = PREFORK_NOSHWORDSPLIT;
|
|
}
|
|
multsub(&val, split_flags, (aspar ? NULL : &aval),
|
|
&isarr, NULL, &ms_flags);
|
|
copied = 1;
|
|
spbreak = 0;
|
|
/* Leave globsubst on if forced */
|
|
if (globsubst != 2)
|
|
globsubst = 0;
|
|
}
|
|
break;
|
|
case ':':
|
|
/* this must be `::=', unconditional assignment */
|
|
if (*s != '=' && *s != Equals)
|
|
goto noclosebrace;
|
|
vunset = 1;
|
|
s++;
|
|
/* Fall through */
|
|
case '=':
|
|
case Equals:
|
|
if (vunset) {
|
|
char sav = *idend;
|
|
int l, split_flags;
|
|
|
|
*idend = '\0';
|
|
val = dupstring(s);
|
|
if (spsep || !arrasg) {
|
|
/* POSIX requires PREFORK_SINGLE semantics here, but
|
|
* traditional zsh used PREFORK_NOSHWORDSPLIT. Base
|
|
* behavior on caller choice of PREFORK_SHWORDSPLIT. */
|
|
multsub(&val,
|
|
spbreak ? PREFORK_SINGLE : PREFORK_NOSHWORDSPLIT,
|
|
NULL, &isarr, NULL, &ms_flags);
|
|
} else {
|
|
if (spbreak)
|
|
split_flags = PREFORK_SPLIT|PREFORK_SHWORDSPLIT;
|
|
else
|
|
split_flags = PREFORK_NOSHWORDSPLIT;
|
|
multsub(&val, split_flags, &aval, &isarr, NULL,
|
|
&ms_flags);
|
|
spbreak = 0;
|
|
}
|
|
if (arrasg) {
|
|
/* This is an array assignment. */
|
|
char *arr[2], **t, **a, **p;
|
|
if (spsep || spbreak) {
|
|
aval = sepsplit(val, spsep, 0, 1);
|
|
isarr = nojoin ? 1 : 2;
|
|
l = arrlen(aval);
|
|
if (l && !*(aval[l-1]))
|
|
l--;
|
|
if (l && !**aval)
|
|
l--, t = aval + 1;
|
|
else
|
|
t = aval;
|
|
} else if (!isarr) {
|
|
if (!*val && arrasg > 1) {
|
|
arr[0] = NULL;
|
|
l = 0;
|
|
} else {
|
|
arr[0] = val;
|
|
arr[1] = NULL;
|
|
l = 1;
|
|
}
|
|
t = aval = arr;
|
|
} else
|
|
l = arrlen(aval), t = aval;
|
|
p = a = zalloc(sizeof(char *) * (l + 1));
|
|
while (l--) {
|
|
untokenize(*t);
|
|
*p++ = ztrdup(*t++);
|
|
}
|
|
*p++ = NULL;
|
|
if (arrasg > 1) {
|
|
Param pm = sethparam(idbeg, a);
|
|
if (pm)
|
|
aval = paramvalarr(pm->gsu.h->getfn(pm), hkeys|hvals);
|
|
} else
|
|
setaparam(idbeg, a);
|
|
isarr = 1;
|
|
} else {
|
|
untokenize(val);
|
|
setsparam(idbeg, ztrdup(val));
|
|
}
|
|
*idend = sav;
|
|
copied = 1;
|
|
if (isarr) {
|
|
if (nojoin)
|
|
isarr = -1;
|
|
if (qt && !getlen && isarr > 0 && !spsep && spbreak < 2) {
|
|
val = sepjoin(aval, sep, 1);
|
|
isarr = 0;
|
|
}
|
|
sep = spsep = NULL;
|
|
spbreak = 0;
|
|
}
|
|
}
|
|
break;
|
|
case '?':
|
|
case Quest:
|
|
if (vunset) {
|
|
if (isset(EXECOPT)) {
|
|
*idend = '\0';
|
|
zerr("%s: %s", idbeg, *s ? s : "parameter not set");
|
|
/*
|
|
* In interactive shell we need to return to
|
|
* top-level prompt --- don't clear this error
|
|
* after handling a command as we do with
|
|
* most errors.
|
|
*/
|
|
errflag |= ERRFLAG_HARD;
|
|
if (!interact) {
|
|
if (mypid == getpid()) {
|
|
/*
|
|
* paranoia: don't check for jobs, but there
|
|
* shouldn't be any if not interactive.
|
|
*/
|
|
stopmsg = 1;
|
|
zexit(1, 0);
|
|
} else
|
|
_exit(1);
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
break;
|
|
case '%':
|
|
case '#':
|
|
case Pound:
|
|
case '/':
|
|
/* This once was executed only `if (qt) ...'. But with that
|
|
* patterns in a expansion resulting from a ${(e)...} aren't
|
|
* tokenized even though this function thinks they are (it thinks
|
|
* they are because parse_subst_str() turns Qstring tokens
|
|
* into String tokens and for unquoted parameter expansions the
|
|
* lexer normally does tokenize patterns inside parameter
|
|
* expansions). */
|
|
{
|
|
int one = noerrs, oef = errflag, haserr;
|
|
|
|
if (!quoteerr)
|
|
noerrs = 1;
|
|
haserr = parse_subst_string(s);
|
|
noerrs = one;
|
|
if (!quoteerr) {
|
|
/* Retain user interrupt error status */
|
|
errflag = oef | (errflag & ERRFLAG_INT);
|
|
if (haserr)
|
|
shtokenize(s);
|
|
} else if (haserr || errflag) {
|
|
zerr("parse error in ${...%c...} substitution", s[-1]);
|
|
return NULL;
|
|
}
|
|
}
|
|
{
|
|
#if 0
|
|
/*
|
|
* This allows # and % to be at the start of
|
|
* a parameter in the substitution, which is
|
|
* a bit nasty, and can be done (although
|
|
* less efficiently) with anchors.
|
|
*/
|
|
|
|
char t = s[-1];
|
|
|
|
singsub(&s);
|
|
|
|
if (t == '/' && (flags & SUB_SUBSTR)) {
|
|
if ((c = *s) == '#' || c == '%') {
|
|
flags &= ~SUB_SUBSTR;
|
|
if (c == '%')
|
|
flags |= SUB_END;
|
|
s++;
|
|
} else if (c == '\\') {
|
|
s++;
|
|
}
|
|
}
|
|
#else
|
|
singsub(&s);
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
* Either loop over an array doing replacements or
|
|
* do the replacment on a string.
|
|
*
|
|
* We need an untokenized value for matching.
|
|
*/
|
|
if (!vunset && isarr) {
|
|
char **ap;
|
|
if (!copied) {
|
|
aval = arrdup(aval);
|
|
copied = 1;
|
|
}
|
|
for (ap = aval; *ap; ap++) {
|
|
untokenize(*ap);
|
|
}
|
|
getmatcharr(&aval, s, flags, flnum, replstr);
|
|
} else {
|
|
if (vunset) {
|
|
if (vunset > 0 && unset(UNSET)) {
|
|
*idend = '\0';
|
|
zerr("%s: parameter not set", idbeg);
|
|
return NULL;
|
|
}
|
|
val = dupstring("");
|
|
}
|
|
if (!copied) {
|
|
val = dupstring(val);
|
|
copied = 1;
|
|
untokenize(val);
|
|
}
|
|
getmatch(&val, s, flags, flnum, replstr);
|
|
}
|
|
break;
|
|
}
|
|
} else if (inbrace && (*s == '^' || *s == Hat)) {
|
|
char **zip;
|
|
int shortest = 1;
|
|
++s;
|
|
if (*s == '^' || *s == Hat) {
|
|
shortest = 0;
|
|
++s;
|
|
}
|
|
if (*itype_end(s, IIDENT, 0)) {
|
|
untokenize(s);
|
|
zerr("not an identifier: %s", s);
|
|
return NULL;
|
|
}
|
|
if (vunset) {
|
|
if (vunset > 0 && unset(UNSET)) {
|
|
*idend = '\0';
|
|
zerr("%s: parameter not set", idbeg);
|
|
return NULL;
|
|
}
|
|
val = dupstring("");
|
|
} else {
|
|
char *sval;
|
|
zip = getaparam(s);
|
|
if (!zip) {
|
|
sval = getsparam(s);
|
|
if (sval)
|
|
zip = hmkarray(sval);
|
|
}
|
|
if (!isarr) {
|
|
aval = mkarray(val);
|
|
isarr = 1;
|
|
}
|
|
if (zip) {
|
|
char **out;
|
|
int alen, ziplen, outlen, i = 0;
|
|
alen = arrlen(aval);
|
|
ziplen = arrlen(zip);
|
|
outlen = shortest ^ (alen > ziplen) ? alen : ziplen;
|
|
if (!shortest && (alen == 0 || ziplen == 0)) {
|
|
if (ziplen)
|
|
aval = arrdup(zip);
|
|
} else {
|
|
out = zhalloc(sizeof(char *) * (2 * outlen + 1));
|
|
while (i < outlen) {
|
|
if (copied)
|
|
out[i*2] = aval[i % alen];
|
|
else
|
|
out[i*2] = dupstring(aval[i % alen]);
|
|
out[i*2+1] = dupstring(zip[i % ziplen]);
|
|
i++;
|
|
}
|
|
out[i*2] = NULL;
|
|
aval = out;
|
|
copied = 1;
|
|
}
|
|
} else {
|
|
if (unset(UNSET)) {
|
|
zerr("%s: parameter not set", s);
|
|
return NULL;
|
|
}
|
|
val = dupstring("");
|
|
}
|
|
}
|
|
} else if (inbrace && (*s == '|' || *s == Bar ||
|
|
*s == '*' || *s == Star)) {
|
|
int intersect = (*s == '*' || *s == Star);
|
|
char **compare, **ap, **apsrc;
|
|
++s;
|
|
if (*itype_end(s, IIDENT, 0)) {
|
|
untokenize(s);
|
|
zerr("not an identifier: %s", s);
|
|
return NULL;
|
|
}
|
|
compare = getaparam(s);
|
|
if (compare) {
|
|
HashTable ht = newuniqtable(arrlen(compare)+1);
|
|
int present;
|
|
for (ap = compare; *ap; ap++)
|
|
(void)addhashnode2(ht, *ap, (HashNode)
|
|
zhalloc(sizeof(struct hashnode)));
|
|
if (!vunset && isarr) {
|
|
if (!copied) {
|
|
aval = arrdup(aval);
|
|
copied = 1;
|
|
}
|
|
for (ap = apsrc = aval; *apsrc; apsrc++) {
|
|
untokenize(*apsrc);
|
|
present = (gethashnode2(ht, *apsrc) != NULL);
|
|
if (intersect ? present : !present) {
|
|
if (ap != apsrc) {
|
|
*ap = *apsrc;
|
|
}
|
|
ap++;
|
|
}
|
|
}
|
|
*ap = NULL;
|
|
} else {
|
|
if (vunset) {
|
|
if (vunset > 0 && unset(UNSET)) {
|
|
*idend = '\0';
|
|
zerr("%s: parameter not set", idbeg);
|
|
deletehashtable(ht);
|
|
return NULL;
|
|
}
|
|
val = dupstring("");
|
|
} else {
|
|
present = (gethashnode2(ht, val) != NULL);
|
|
if (intersect ? !present : present)
|
|
val = dupstring("");
|
|
}
|
|
}
|
|
deletehashtable(ht);
|
|
} else if (intersect) {
|
|
/*
|
|
* The intersection with nothing is nothing...
|
|
* Seems a bit pointless complaining that the first
|
|
* expression is unset here if the second is, too.
|
|
*/
|
|
if (!vunset) {
|
|
if (isarr) {
|
|
aval = hmkarray(NULL);
|
|
} else {
|
|
val = dupstring("");
|
|
}
|
|
}
|
|
}
|
|
if (vunset) {
|
|
if (vunset > 0 && unset(UNSET)) {
|
|
*idend = '\0';
|
|
zerr("%s: parameter not set", idbeg);
|
|
return NULL;
|
|
}
|
|
val = dupstring("");
|
|
}
|
|
} else { /* no ${...=...} or anything, but possible modifiers. */
|
|
/*
|
|
* Handler ${+...}. TODO: strange, why do we handle this only
|
|
* if there isn't a trailing modifier? Why don't we do this
|
|
* e.g. when we handle the ${(t)...} flag?
|
|
*/
|
|
if (chkset) {
|
|
val = dupstring(vunset ? "0" : "1");
|
|
isarr = 0;
|
|
} else if (vunset) {
|
|
if (vunset > 0 && unset(UNSET)) {
|
|
*idend = '\0';
|
|
zerr("%s: parameter not set", idbeg);
|
|
return NULL;
|
|
}
|
|
val = dupstring("");
|
|
}
|
|
if (colf && inbrace) {
|
|
/*
|
|
* Look for ${PARAM:OFFSET} or ${PARAM:OFFSET:LENGTH}.
|
|
* This must appear before modifiers. For compatibility
|
|
* with bash we perform both standard string substitutions
|
|
* and math eval.
|
|
*/
|
|
char *check_offset2;
|
|
char *check_offset = check_colon_subscript(s, &check_offset2);
|
|
if (check_offset) {
|
|
zlong offset = mathevali(check_offset);
|
|
zlong length = 0;
|
|
int length_set = 0;
|
|
int offset_hack_argzero = 0;
|
|
if (errflag)
|
|
return NULL;
|
|
if ((*check_offset2 && *check_offset2 != ':')) {
|
|
zerr("invalid subscript: %s", check_offset);
|
|
return NULL;
|
|
}
|
|
if (*check_offset2) {
|
|
check_offset = check_colon_subscript(check_offset2 + 1,
|
|
&check_offset2);
|
|
if (*check_offset2 && *check_offset2 != ':') {
|
|
zerr("invalid length: %s", check_offset);
|
|
return NULL;
|
|
}
|
|
if (check_offset) {
|
|
length = mathevali(check_offset);
|
|
length_set = 1;
|
|
if (errflag)
|
|
return NULL;
|
|
}
|
|
}
|
|
if (isarr) {
|
|
int alen, count;
|
|
char **srcptr, **dstptr, **newarr;
|
|
|
|
if (horrible_offset_hack) {
|
|
/*
|
|
* As part of the 'orrible hoffset 'ack,
|
|
* (what hare you? Han 'orrible hoffset 'ack,
|
|
* sergeant major), if we are given a ksh/bash/POSIX
|
|
* style positional parameter array which includes
|
|
* offset 0, we use $0.
|
|
*/
|
|
if (offset == 0) {
|
|
offset_hack_argzero = 1;
|
|
} else if (offset > 0) {
|
|
offset--;
|
|
}
|
|
}
|
|
|
|
alen = arrlen(aval);
|
|
if (offset < 0) {
|
|
offset += alen;
|
|
if (offset < 0)
|
|
offset = 0;
|
|
}
|
|
if (offset_hack_argzero)
|
|
alen++;
|
|
if (length_set) {
|
|
if (length < 0)
|
|
length += alen - offset;
|
|
if (length < 0) {
|
|
zerr("substring expression: %d < %d",
|
|
(int)(length + offset), (int)offset);
|
|
return NULL;
|
|
}
|
|
} else
|
|
length = alen;
|
|
if (offset > alen)
|
|
offset = alen;
|
|
if (offset + length > alen)
|
|
length = alen - offset;
|
|
count = length;
|
|
srcptr = aval + offset;
|
|
newarr = dstptr = (char **)
|
|
zhalloc((length+1)*sizeof(char *));
|
|
if (count && offset_hack_argzero) {
|
|
*dstptr++ = dupstring(argzero);
|
|
count--;
|
|
}
|
|
while (count--)
|
|
*dstptr++ = dupstring(*srcptr++);
|
|
*dstptr = (char *)NULL;
|
|
aval = newarr;
|
|
} else {
|
|
char *sptr, *eptr;
|
|
int given_offset;
|
|
if (offset < 0) {
|
|
MB_METACHARINIT();
|
|
for (sptr = val; *sptr; ) {
|
|
sptr += MB_METACHARLEN(sptr);
|
|
offset++;
|
|
}
|
|
if (offset < 0)
|
|
offset = 0;
|
|
}
|
|
given_offset = offset;
|
|
MB_METACHARINIT();
|
|
if (length_set && length < 0)
|
|
length -= offset;
|
|
for (sptr = val; *sptr && offset; ) {
|
|
sptr += MB_METACHARLEN(sptr);
|
|
offset--;
|
|
}
|
|
if (length_set) {
|
|
if (length < 0) {
|
|
MB_METACHARINIT();
|
|
for (eptr = val; *eptr; ) {
|
|
eptr += MB_METACHARLEN(eptr);
|
|
length++;
|
|
}
|
|
if (length < 0) {
|
|
zerr("substring expression: %d < %d",
|
|
(int)(length + given_offset),
|
|
(int)given_offset);
|
|
return NULL;
|
|
}
|
|
}
|
|
for (eptr = sptr; *eptr && length; ) {
|
|
eptr += MB_METACHARLEN(eptr);
|
|
length--;
|
|
}
|
|
val = dupstrpfx(sptr, eptr - sptr);
|
|
} else {
|
|
val = dupstring(sptr);
|
|
}
|
|
}
|
|
if (!*check_offset2) {
|
|
colf = 0;
|
|
} else {
|
|
s = check_offset2 + 1;
|
|
}
|
|
}
|
|
}
|
|
if (colf) {
|
|
/*
|
|
* History style colon modifiers. May need to apply
|
|
* on multiple elements of an array.
|
|
*/
|
|
s--;
|
|
if (unset(KSHARRAYS) || inbrace) {
|
|
if (!isarr)
|
|
modify(&val, &s);
|
|
else {
|
|
char *ss;
|
|
char **ap = aval;
|
|
char **pp = aval = (char **) hcalloc(sizeof(char *) *
|
|
(arrlen(aval) + 1));
|
|
|
|
while ((*pp = *ap++)) {
|
|
ss = s;
|
|
modify(pp++, &ss);
|
|
}
|
|
if (pp == aval) {
|
|
char *t = "";
|
|
ss = s;
|
|
modify(&t, &ss);
|
|
}
|
|
s = ss;
|
|
}
|
|
copied = 1;
|
|
if (inbrace && *s) {
|
|
if (*s == ':' && !imeta(s[1]))
|
|
zerr("unrecognized modifier `%c'", s[1]);
|
|
else
|
|
zerr("unrecognized modifier");
|
|
return NULL;
|
|
}
|
|
}
|
|
}
|
|
if (!inbrace)
|
|
fstr = s;
|
|
}
|
|
if (errflag)
|
|
return NULL;
|
|
if (evalchar) {
|
|
int one = noerrs, oef = errflag, haserr = 0;
|
|
|
|
if (!quoteerr)
|
|
noerrs = 1;
|
|
/*
|
|
* Evaluate the value numerically and output the result as
|
|
* a character.
|
|
*/
|
|
if (isarr) {
|
|
char **aval2, **avptr, **av2ptr;
|
|
|
|
aval2 = (char **)zhalloc((arrlen(aval)+1)*sizeof(char *));
|
|
|
|
for (avptr = aval, av2ptr = aval2; *avptr; avptr++, av2ptr++)
|
|
{
|
|
/* When noerrs = 1, the only error is out-of-memory */
|
|
if (!(*av2ptr = substevalchar(*avptr))) {
|
|
haserr = 1;
|
|
break;
|
|
}
|
|
}
|
|
*av2ptr = NULL;
|
|
aval = aval2;
|
|
} else {
|
|
/* When noerrs = 1, the only error is out-of-memory */
|
|
if (!(val = substevalchar(val)))
|
|
haserr = 1;
|
|
}
|
|
noerrs = one;
|
|
if (!quoteerr) {
|
|
/* Retain user interrupt error status */
|
|
errflag = oef | (errflag & ERRFLAG_INT);
|
|
}
|
|
if (haserr || errflag)
|
|
return NULL;
|
|
ms_flags = 0;
|
|
}
|
|
/*
|
|
* This handles taking a length with ${#foo} and variations.
|
|
* TODO: again. one might naively have thought this had the
|
|
* same sort of effect as the ${(t)...} flag and the ${+...}
|
|
* test, although in this case we do need the value rather
|
|
* the parameter, so maybe it's a bit different.
|
|
*/
|
|
if (getlen) {
|
|
long len = 0;
|
|
char buf[14];
|
|
|
|
if (isarr) {
|
|
char **ctr;
|
|
int sl = sep ? MB_METASTRLEN(sep) : 1;
|
|
|
|
if (getlen == 1)
|
|
for (ctr = aval; *ctr; ctr++, len++);
|
|
else if (getlen == 2) {
|
|
if (*aval)
|
|
for (len = -sl, ctr = aval;
|
|
len += sl + MB_METASTRLEN2(*ctr, multi_width),
|
|
*++ctr;);
|
|
}
|
|
else
|
|
for (ctr = aval;
|
|
*ctr;
|
|
len += wordcount(*ctr, spsep, getlen > 3), ctr++);
|
|
} else {
|
|
if (getlen < 3)
|
|
len = MB_METASTRLEN2(val, multi_width);
|
|
else
|
|
len = wordcount(val, spsep, getlen > 3);
|
|
}
|
|
|
|
sprintf(buf, "%ld", len);
|
|
val = dupstring(buf);
|
|
isarr = 0;
|
|
ms_flags = 0;
|
|
}
|
|
/* At this point we make sure that our arrayness has affected the
|
|
* arrayness of the linked list. Then, we can turn our value into
|
|
* a scalar for convenience sake without affecting the arrayness
|
|
* of the resulting value. ## This is the YUK chunk. ## */
|
|
if (isarr)
|
|
l->list.flags |= LF_ARRAY;
|
|
else
|
|
l->list.flags &= ~LF_ARRAY;
|
|
if (isarr > 0 && !plan9 && (!aval || !aval[0])) {
|
|
val = dupstring("");
|
|
isarr = 0;
|
|
} else if (isarr && aval && aval[0] && !aval[1]) {
|
|
/* treat a one-element array as a scalar for purposes of *
|
|
* concatenation with surrounding text (some${param}thing) *
|
|
* and rc_expand_param handling. Note: LF_ARRAY (above) *
|
|
* propagates the true array type from nested expansions. */
|
|
val = aval[0];
|
|
isarr = 0;
|
|
}
|
|
/* This is where we may join arrays together, e.g. (j:,:) sets "sep", and
|
|
* (afterward) may split the joined value (e.g. (s:-:) sets "spsep"). One
|
|
* exception is that ${name:-word} and ${name:+word} will have already
|
|
* done any requested splitting of the word value with quoting preserved.
|
|
*/
|
|
if (ssub || spbreak || spsep || sep) {
|
|
int force_split = !ssub && (spbreak || spsep);
|
|
if (isarr) {
|
|
/* sep non-null here means F or j flag, force join */
|
|
if (nojoin == 0 || sep) {
|
|
val = sepjoin(aval, sep, 1);
|
|
isarr = 0;
|
|
ms_flags = 0;
|
|
} else if (force_split && (spsep || nojoin == 2)) {
|
|
/* Hack to simulate splitting individual elements:
|
|
* forced joining as previously determined, or
|
|
* join on what we later use to forcibly split
|
|
*/
|
|
val = sepjoin(aval, (nojoin == 1 ? NULL : spsep), 1);
|
|
isarr = 0;
|
|
}
|
|
}
|
|
if (force_split && !isarr) {
|
|
aval = sepsplit(val, spsep, 0, 1);
|
|
if (!aval || !aval[0])
|
|
val = dupstring("");
|
|
else if (!aval[1])
|
|
val = aval[0];
|
|
else
|
|
isarr = nojoin ? 1 : 2;
|
|
}
|
|
if (isarr)
|
|
l->list.flags |= LF_ARRAY;
|
|
else
|
|
l->list.flags &= ~LF_ARRAY;
|
|
}
|
|
/*
|
|
* Perform case modififications.
|
|
*/
|
|
if (casmod != CASMOD_NONE) {
|
|
copied = 1; /* string is always modified by copy */
|
|
if (isarr) {
|
|
char **ap, **ap2;
|
|
|
|
ap = aval;
|
|
ap2 = aval = (char **) zhalloc(sizeof(char *) * (arrlen(aval)+1));
|
|
|
|
while (*ap)
|
|
*ap2++ = casemodify(*ap++, casmod);
|
|
*ap2++ = NULL;
|
|
} else {
|
|
val = casemodify(val, casmod);
|
|
}
|
|
}
|
|
/*
|
|
* Process echo- and print-style escape sequences.
|
|
*/
|
|
if (getkeys >= 0) {
|
|
int len;
|
|
|
|
copied = 1; /* string is always copied */
|
|
if (isarr) {
|
|
char **ap, **ap2;
|
|
|
|
ap = aval;
|
|
aval = (char **) zhalloc(sizeof(char *) * (arrlen(aval)+1));
|
|
for (ap2 = aval; *ap; ap++, ap2++) {
|
|
*ap2 = getkeystring(*ap, &len, getkeys, NULL);
|
|
*ap2 = metafy(*ap2, len, META_USEHEAP);
|
|
}
|
|
*ap2++ = NULL;
|
|
} else {
|
|
val = getkeystring(val, &len, getkeys, NULL);
|
|
val = metafy(val, len, META_USEHEAP);
|
|
}
|
|
}
|
|
/*
|
|
* Perform prompt-style modifications.
|
|
*/
|
|
if (presc) {
|
|
int ops = opts[PROMPTSUBST], opb = opts[PROMPTBANG];
|
|
int opp = opts[PROMPTPERCENT];
|
|
|
|
if (presc < 2) {
|
|
opts[PROMPTPERCENT] = 1;
|
|
opts[PROMPTSUBST] = opts[PROMPTBANG] = 0;
|
|
}
|
|
/*
|
|
* TODO: It would be really quite nice to abstract the
|
|
* isarr and !isarr code into a function which gets
|
|
* passed a pointer to a function with the effect of
|
|
* the promptexpand bit. Then we could use this for
|
|
* a lot of stuff and bury val/aval/isarr inside a structure
|
|
* which gets passed to it.
|
|
*/
|
|
if (isarr) {
|
|
char **ap;
|
|
|
|
if (!copied)
|
|
aval = arrdup(aval), copied = 1;
|
|
ap = aval;
|
|
for (; *ap; ap++) {
|
|
char *tmps;
|
|
untokenize(*ap);
|
|
tmps = promptexpand(*ap, 0, NULL, NULL, NULL);
|
|
*ap = dupstring(tmps);
|
|
free(tmps);
|
|
}
|
|
} else {
|
|
char *tmps;
|
|
if (!copied)
|
|
val = dupstring(val), copied = 1;
|
|
untokenize(val);
|
|
tmps = promptexpand(val, 0, NULL, NULL, NULL);
|
|
val = dupstring(tmps);
|
|
free(tmps);
|
|
}
|
|
opts[PROMPTSUBST] = ops;
|
|
opts[PROMPTBANG] = opb;
|
|
opts[PROMPTPERCENT] = opp;
|
|
}
|
|
/*
|
|
* One of the possible set of quotes to apply, depending on
|
|
* the repetitions of the (q) flag.
|
|
*/
|
|
if (quotemod) {
|
|
int pre = 0, post = 0;
|
|
|
|
if (quotemod > 0) {
|
|
switch (quotetype)
|
|
{
|
|
case QT_DOLLARS:
|
|
/* space for "$" */
|
|
pre = 2;
|
|
post = 1;
|
|
break;
|
|
|
|
case QT_SINGLE_OPTIONAL:
|
|
/* quotes will be added for us */
|
|
case QT_BACKSLASH:
|
|
case QT_BACKSLASH_PATTERN:
|
|
/* no quotes */
|
|
break;
|
|
|
|
default:
|
|
pre = post = 1;
|
|
break;
|
|
}
|
|
}
|
|
if (isarr) {
|
|
char **ap;
|
|
|
|
if (!copied)
|
|
aval = arrdup(aval), copied = 1;
|
|
ap = aval;
|
|
|
|
if (quotemod > 0) {
|
|
if (quotetype == QT_QUOTEDZPUTS) {
|
|
for (; *ap; ap++)
|
|
*ap = quotedzputs(*ap, NULL);
|
|
} else if (quotetype > QT_BACKSLASH) {
|
|
int sl;
|
|
char *tmp;
|
|
|
|
for (; *ap; ap++) {
|
|
tmp = quotestring(*ap, quotetype);
|
|
sl = strlen(tmp);
|
|
*ap = (char *) zhalloc(pre + sl + post + 1);
|
|
strcpy((*ap) + pre, tmp);
|
|
if (pre)
|
|
ap[0][pre - 1] = ap[0][pre + sl] =
|
|
(quotetype != QT_DOUBLE ? '\'' : '"');
|
|
ap[0][pre + sl + post] = '\0';
|
|
if (quotetype == QT_DOLLARS)
|
|
ap[0][0] = '$';
|
|
}
|
|
} else
|
|
for (; *ap; ap++)
|
|
*ap = quotestring(*ap, QT_BACKSLASH_SHOWNULL);
|
|
} else {
|
|
int one = noerrs, oef = errflag, haserr = 0;
|
|
|
|
if (!quoteerr)
|
|
noerrs = 1;
|
|
for (; *ap; ap++) {
|
|
haserr |= parse_subst_string(*ap);
|
|
remnulargs(*ap);
|
|
untokenize(*ap);
|
|
}
|
|
noerrs = one;
|
|
if (!quoteerr) {
|
|
/* Retain any user interrupt error status */
|
|
errflag = oef | (errflag & ERRFLAG_INT);
|
|
}
|
|
else if (haserr || errflag) {
|
|
zerr("parse error in parameter value");
|
|
return NULL;
|
|
}
|
|
}
|
|
} else {
|
|
if (!copied)
|
|
val = dupstring(val), copied = 1;
|
|
if (quotemod > 0) {
|
|
if (quotetype == QT_QUOTEDZPUTS) {
|
|
val = quotedzputs(val, NULL);
|
|
} else if (quotetype > QT_BACKSLASH) {
|
|
int sl;
|
|
char *tmp;
|
|
tmp = quotestring(val, quotetype);
|
|
sl = strlen(tmp);
|
|
val = (char *) zhalloc(pre + sl + post + 1);
|
|
strcpy(val + pre, tmp);
|
|
if (pre)
|
|
val[pre - 1] = val[pre + sl] =
|
|
(quotetype != QT_DOUBLE ? '\'' : '"');
|
|
val[pre + sl + post] = '\0';
|
|
if (quotetype == QT_DOLLARS)
|
|
val[0] = '$';
|
|
} else
|
|
val = quotestring(val, QT_BACKSLASH_SHOWNULL);
|
|
} else {
|
|
int one = noerrs, oef = errflag, haserr;
|
|
|
|
if (!quoteerr)
|
|
noerrs = 1;
|
|
haserr = parse_subst_string(val);
|
|
noerrs = one;
|
|
if (!quoteerr) {
|
|
/* Retain any user interrupt error status */
|
|
errflag = oef | (errflag & ERRFLAG_INT);
|
|
}
|
|
else if (haserr || errflag) {
|
|
zerr("parse error in parameter value");
|
|
return NULL;
|
|
}
|
|
remnulargs(val);
|
|
untokenize(val);
|
|
}
|
|
}
|
|
}
|
|
/*
|
|
* Transform special characters in the string to make them
|
|
* printable, or to show directories, or possibly even both.
|
|
*/
|
|
if (mods) {
|
|
if (isarr) {
|
|
char **ap;
|
|
if (!copied)
|
|
aval = arrdup(aval), copied = 1;
|
|
for (ap = aval; *ap; ap++) {
|
|
if (mods & 1)
|
|
*ap = substnamedir(*ap);
|
|
if (mods & 2)
|
|
*ap = nicedupstring(*ap);
|
|
}
|
|
} else {
|
|
if (!copied)
|
|
val = dupstring(val), copied = 1;
|
|
if (mods & 1)
|
|
val = substnamedir(val);
|
|
if (mods & 2)
|
|
val = nicedupstring(val);
|
|
}
|
|
}
|
|
/*
|
|
* Nothing particularly to do with SH_WORD_SPLIT --- this
|
|
* performs lexical splitting on a string as specified by
|
|
* the (z) flag.
|
|
*/
|
|
if (shsplit) {
|
|
LinkList list = NULL;
|
|
|
|
if (isarr) {
|
|
char **ap;
|
|
for (ap = aval; *ap; ap++)
|
|
list = bufferwords(list, *ap, NULL, shsplit);
|
|
isarr = 0;
|
|
} else
|
|
list = bufferwords(NULL, val, NULL, shsplit);
|
|
|
|
if (!list || !firstnode(list))
|
|
val = dupstring("");
|
|
else if (!nextnode(firstnode(list)))
|
|
val = getdata(firstnode(list));
|
|
else {
|
|
aval = hlinklist2array(list, 0);
|
|
isarr = nojoin ? 1 : 2;
|
|
l->list.flags |= LF_ARRAY;
|
|
}
|
|
copied = 1;
|
|
}
|
|
/*
|
|
* TODO: hmm. At this point we have to be on our toes about
|
|
* whether we're putting stuff into a line or not, i.e.
|
|
* we don't want to do this from a recursive call.
|
|
* Rather than passing back flags in a non-trivial way, maybe
|
|
* we could decide on the basis of flags passed down to us.
|
|
*
|
|
* This is the ideal place to do any last-minute conversion from
|
|
* array to strings. However, given all the transformations we've
|
|
* already done, probably if it's going to be done it will already
|
|
* have been. (I'd really like to keep everying in aval or
|
|
* equivalent and only locally decide if we need to treat it
|
|
* as a scalar.)
|
|
*/
|
|
|
|
/*
|
|
* If a multsub result had whitespace at the start and we're
|
|
* splitting and there's a previous string, now's the time to do so.
|
|
*/
|
|
if ((ms_flags & MULTSUB_WS_AT_START) && aptr > ostr) {
|
|
insertlinknode(l, n, dupstrpfx(ostr, aptr - ostr)), incnode(n);
|
|
ostr = aptr;
|
|
}
|
|
/* Likewise at the end */
|
|
if ((ms_flags & MULTSUB_WS_AT_END) && *fstr) {
|
|
insertlinknode(l, n, dupstring(fstr)); /* appended, no incnode */
|
|
*fstr = '\0';
|
|
}
|
|
if (isarr) {
|
|
char *x;
|
|
char *y;
|
|
int xlen;
|
|
int i;
|
|
LinkNode on = n;
|
|
|
|
/* Handle the (u) flag; we need this before the next test */
|
|
if (unique) {
|
|
if(!copied)
|
|
aval = arrdup(aval);
|
|
|
|
i = arrlen(aval);
|
|
if (i > 1)
|
|
zhuniqarray(aval);
|
|
}
|
|
if ((!aval[0] || !aval[1]) && !plan9) {
|
|
/*
|
|
* Empty array or single element. Currently you only
|
|
* get a single element array at this point from the
|
|
* unique expansion above. but we can potentially
|
|
* have other reasons.
|
|
*
|
|
* The following test removes the markers
|
|
* from surrounding double quotes, but I don't know why
|
|
* that's necessary.
|
|
*/
|
|
int vallen;
|
|
if (aptr > (char *) getdata(n) &&
|
|
aptr[-1] == Dnull && *fstr == Dnull)
|
|
*--aptr = '\0', fstr++;
|
|
vallen = aval[0] ? strlen(aval[0]) : 0;
|
|
y = (char *) hcalloc((aptr - ostr) + vallen + strlen(fstr) + 1);
|
|
strcpy(y, ostr);
|
|
*str = y + (aptr - ostr);
|
|
if (vallen)
|
|
{
|
|
strcpy(*str, aval[0]);
|
|
*str += vallen;
|
|
}
|
|
strcpy(*str, fstr);
|
|
setdata(n, y);
|
|
return n;
|
|
}
|
|
/* Handle (o) and (O) and their variants */
|
|
if (sortit != SORTIT_ANYOLDHOW) {
|
|
if (!copied)
|
|
aval = arrdup(aval);
|
|
if (indord) {
|
|
if (sortit & SORTIT_BACKWARDS) {
|
|
char *copy;
|
|
char **end = aval + arrlen(aval) - 1, **start = aval;
|
|
|
|
/* reverse the array */
|
|
while (start < end) {
|
|
copy = *end;
|
|
*end-- = *start;
|
|
*start++ = copy;
|
|
}
|
|
}
|
|
} else {
|
|
/*
|
|
* HERE: we tested if the last element of the array
|
|
* was not a NULL string. Why the last element?
|
|
* Why didn't we expect NULL strings to work?
|
|
* Was it just a clumsy way of testing whether there
|
|
* was enough in the array to sort?
|
|
*/
|
|
strmetasort(aval, sortit, NULL);
|
|
}
|
|
}
|
|
if (plan9) {
|
|
/* Handle RC_EXPAND_PARAM */
|
|
LinkNode tn;
|
|
local_list1(tl);
|
|
|
|
*--fstr = Marker;
|
|
init_list1(tl, fstr);
|
|
if (!eval && !stringsubst(&tl, firstnode(&tl), ssub, ret_flags, 0))
|
|
return NULL;
|
|
*str = aptr;
|
|
tn = firstnode(&tl);
|
|
while ((x = *aval++)) {
|
|
if (prenum || postnum)
|
|
x = dopadding(x, prenum, postnum, preone, postone,
|
|
premul, postmul
|
|
#ifdef MULTIBYTE_SUPPORT
|
|
, multi_width
|
|
#endif
|
|
);
|
|
if (eval && subst_parse_str(&x, (qt && !nojoin), quoteerr))
|
|
return NULL;
|
|
xlen = strlen(x);
|
|
for (tn = firstnode(&tl);
|
|
tn && *(y = (char *) getdata(tn)) == Marker;
|
|
incnode(tn)) {
|
|
strcatsub(&y, ostr, aptr, x, xlen, y + 1, globsubst,
|
|
copied);
|
|
if (qt && !*y && isarr != 2)
|
|
y = dupstring(nulstring);
|
|
if (plan9)
|
|
setdata(n, (void *) y), plan9 = 0;
|
|
else
|
|
insertlinknode(l, n, (void *) y), incnode(n);
|
|
}
|
|
}
|
|
for (; tn; incnode(tn)) {
|
|
y = (char *) getdata(tn);
|
|
if (*y == Marker)
|
|
continue;
|
|
if (qt && !*y && isarr != 2)
|
|
y = dupstring(nulstring);
|
|
if (plan9)
|
|
setdata(n, (void *) y), plan9 = 0;
|
|
else
|
|
insertlinknode(l, n, (void *) y), incnode(n);
|
|
}
|
|
if (plan9) {
|
|
uremnode(l, n);
|
|
return n;
|
|
}
|
|
} else {
|
|
/*
|
|
* Not RC_EXPAND_PARAM: simply join the first and
|
|
* last values.
|
|
* TODO: how about removing the restriction that
|
|
* aval[1] is non-NULL to promote consistency?, or
|
|
* simply changing the test so that we drop into
|
|
* the scalar branch, instead of tricking isarr?
|
|
*/
|
|
x = aval[0];
|
|
if (prenum || postnum)
|
|
x = dopadding(x, prenum, postnum, preone, postone,
|
|
premul, postmul
|
|
#ifdef MULTIBYTE_SUPPORT
|
|
, multi_width
|
|
#endif
|
|
);
|
|
if (eval && subst_parse_str(&x, (qt && !nojoin), quoteerr))
|
|
return NULL;
|
|
xlen = strlen(x);
|
|
strcatsub(&y, ostr, aptr, x, xlen, NULL, globsubst, copied);
|
|
if (qt && !*y && isarr != 2)
|
|
y = dupstring(nulstring);
|
|
setdata(n, (void *) y);
|
|
|
|
i = 1;
|
|
/* aval[1] is non-null here */
|
|
while (aval[i + 1]) {
|
|
x = aval[i++];
|
|
if (prenum || postnum)
|
|
x = dopadding(x, prenum, postnum, preone, postone,
|
|
premul, postmul
|
|
#ifdef MULTIBYTE_SUPPORT
|
|
, multi_width
|
|
#endif
|
|
);
|
|
if (eval && subst_parse_str(&x, (qt && !nojoin), quoteerr))
|
|
return NULL;
|
|
if (qt && !*x && isarr != 2)
|
|
y = dupstring(nulstring);
|
|
else {
|
|
y = dupstring(x);
|
|
if (globsubst)
|
|
shtokenize(y);
|
|
}
|
|
insertlinknode(l, n, (void *) y), incnode(n);
|
|
}
|
|
|
|
x = aval[i];
|
|
if (prenum || postnum)
|
|
x = dopadding(x, prenum, postnum, preone, postone,
|
|
premul, postmul
|
|
#ifdef MULTIBYTE_SUPPORT
|
|
, multi_width
|
|
#endif
|
|
);
|
|
if (eval && subst_parse_str(&x, (qt && !nojoin), quoteerr))
|
|
return NULL;
|
|
xlen = strlen(x);
|
|
*str = strcatsub(&y, aptr, aptr, x, xlen, fstr, globsubst, copied);
|
|
if (qt && !*y && isarr != 2)
|
|
y = dupstring(nulstring);
|
|
insertlinknode(l, n, (void *) y), incnode(n);
|
|
}
|
|
/* This used to omit restoring of *str and instead test
|
|
* if (eval)
|
|
* n = on;
|
|
* but that causes strange behavior of history modifiers when
|
|
* applied across all values of an array. What is magic about
|
|
* eval here that *str seemed not to need restoring?
|
|
*/
|
|
*str = getdata(n = on);
|
|
} else {
|
|
/*
|
|
* Scalar value. Handle last minute transformations
|
|
* such as left- or right-padding and the (e) flag to
|
|
* revaluate the result.
|
|
*/
|
|
int xlen;
|
|
char *x;
|
|
char *y;
|
|
|
|
x = val;
|
|
if (!x) {
|
|
/* Shouldn't have got here with a NULL string. */
|
|
DPUTS(1, "value is NULL in paramsubst");
|
|
return NULL;
|
|
}
|
|
if (prenum || postnum)
|
|
x = dopadding(x, prenum, postnum, preone, postone,
|
|
premul, postmul
|
|
#ifdef MULTIBYTE_SUPPORT
|
|
, multi_width
|
|
#endif
|
|
);
|
|
if (eval && subst_parse_str(&x, (qt && !nojoin), quoteerr))
|
|
return NULL;
|
|
xlen = strlen(x);
|
|
*str = strcatsub(&y, ostr, aptr, x, xlen, fstr, globsubst, copied);
|
|
if (qt && !*y)
|
|
y = dupstring(nulstring);
|
|
setdata(n, (void *) y);
|
|
}
|
|
if (eval)
|
|
*str = (char *) getdata(n);
|
|
|
|
return n;
|
|
}
|
|
|
|
/*
|
|
* Arithmetic substitution: `a' is the string to be evaluated, `bptr'
|
|
* points to the beginning of the string containing it. The tail of
|
|
* the string is given by `rest'. *bptr is modified with the substituted
|
|
* string. The function returns a pointer to the tail in the substituted
|
|
* string.
|
|
*/
|
|
|
|
/**/
|
|
static char *
|
|
arithsubst(char *a, char **bptr, char *rest)
|
|
{
|
|
char *s = *bptr, *t;
|
|
char buf[BDIGBUFSIZE], *b;
|
|
mnumber v;
|
|
|
|
singsub(&a);
|
|
v = matheval(a);
|
|
if ((v.type & MN_FLOAT) && !outputradix)
|
|
b = convfloat_underscore(v.u.d, outputunderscore);
|
|
else {
|
|
if (v.type & MN_FLOAT)
|
|
v.u.l = (zlong) v.u.d;
|
|
b = convbase_underscore(buf, v.u.l, outputradix, outputunderscore);
|
|
}
|
|
t = *bptr = (char *) hcalloc(strlen(*bptr) + strlen(b) +
|
|
strlen(rest) + 1);
|
|
t--;
|
|
while ((*++t = *s++));
|
|
t--;
|
|
while ((*++t = *b++));
|
|
strcat(t, rest);
|
|
return t;
|
|
}
|
|
|
|
/* This function implements colon modifiers.
|
|
*
|
|
* STR is an in/out parameter. On entry it is the string (e.g., path)
|
|
* to modified. On return it is the modified path.
|
|
*
|
|
* PTR is an in/out parameter. On entry it contains the string of colon
|
|
* modifiers. On return it points past the last recognised modifier.
|
|
*
|
|
* Example:
|
|
* ENTRY: *str is "." *ptr is ":AN"
|
|
* RETURN: *str is "/home/foobar" (equal to $PWD) *ptr points to the "N"
|
|
*/
|
|
|
|
/**/
|
|
void
|
|
modify(char **str, char **ptr)
|
|
{
|
|
char *ptr1, *ptr2, *ptr3, *lptr, c, *test, *sep, *t, *tt, tc, *e;
|
|
char *copy, *all, *tmp, sav, sav1, *ptr1end;
|
|
int gbal, wall, rec, al, nl, charlen, dellen;
|
|
convchar_t del;
|
|
|
|
test = NULL;
|
|
|
|
if (**ptr == ':')
|
|
*str = dupstring(*str);
|
|
|
|
while (**ptr == ':') {
|
|
lptr = *ptr;
|
|
(*ptr)++;
|
|
wall = gbal = 0;
|
|
rec = 1;
|
|
c = '\0';
|
|
sep = NULL;
|
|
|
|
for (; !c && **ptr;) {
|
|
switch (**ptr) {
|
|
case 'a':
|
|
case 'A':
|
|
case 'c':
|
|
case 'h':
|
|
case 'r':
|
|
case 'e':
|
|
case 't':
|
|
case 'l':
|
|
case 'u':
|
|
case 'q':
|
|
case 'Q':
|
|
case 'P':
|
|
c = **ptr;
|
|
break;
|
|
|
|
case 's':
|
|
c = **ptr;
|
|
(*ptr)++;
|
|
ptr1 = *ptr;
|
|
MB_METACHARINIT();
|
|
charlen = MB_METACHARLENCONV(ptr1, &del);
|
|
#ifdef MULTIBYTE_SUPPORT
|
|
if (del == WEOF)
|
|
del = (wint_t)((*ptr1 == Meta) ? ptr1[1] ^ 32 : *ptr1);
|
|
#endif
|
|
ptr1 += charlen;
|
|
for (ptr2 = ptr1, charlen = 0; *ptr2; ptr2 += charlen) {
|
|
convchar_t del2;
|
|
if ((*ptr2 == Bnull || *ptr2 == '\\') && ptr2[1]) {
|
|
/* in double quotes, the backslash isn't tokenized */
|
|
if (*ptr2 == '\\')
|
|
*ptr2 = Bnull;
|
|
charlen = 2;
|
|
continue;
|
|
}
|
|
charlen = MB_METACHARLENCONV(ptr2, &del2);
|
|
#ifdef MULTIBYTE_SUPPORT
|
|
if (del2 == WEOF)
|
|
del2 = (wint_t)((*ptr2 == Meta) ?
|
|
ptr2[1] ^ 32 : *ptr2);
|
|
#endif
|
|
if (del2 == del)
|
|
break;
|
|
}
|
|
if (!*ptr2) {
|
|
zerr("bad substitution");
|
|
return;
|
|
}
|
|
ptr1end = ptr2;
|
|
ptr2 += charlen;
|
|
sav1 = *ptr1end;
|
|
*ptr1end = '\0';
|
|
for (ptr3 = ptr2, charlen = 0; *ptr3; ptr3 += charlen) {
|
|
convchar_t del3;
|
|
if ((*ptr3 == Bnull || *ptr3 == '\\') && ptr3[1]) {
|
|
/* in double quotes, the backslash isn't tokenized */
|
|
if (*ptr3 == '\\')
|
|
*ptr3 = Bnull;
|
|
charlen = 2;
|
|
continue;
|
|
}
|
|
charlen = MB_METACHARLENCONV(ptr3, &del3);
|
|
#ifdef MULTIBYTE_SUPPORT
|
|
if (del3 == WEOF)
|
|
del3 = (wint_t)((*ptr3 == Meta) ?
|
|
ptr3[1] ^ 32 : *ptr3);
|
|
#endif
|
|
if (del3 == del)
|
|
break;
|
|
}
|
|
sav = *ptr3;
|
|
*ptr3 = '\0';
|
|
if (*ptr1) {
|
|
zsfree(hsubl);
|
|
hsubl = ztrdup(ptr1);
|
|
}
|
|
if (!hsubl) {
|
|
zerr("no previous substitution");
|
|
return;
|
|
}
|
|
zsfree(hsubr);
|
|
for (tt = hsubl; *tt; tt++)
|
|
if (inull(*tt) && *tt != Bnullkeep)
|
|
chuck(tt--);
|
|
if (!isset(HISTSUBSTPATTERN))
|
|
untokenize(hsubl);
|
|
for (tt = hsubr = ztrdup(ptr2); *tt; tt++) {
|
|
if (inull(*tt) && *tt != Bnullkeep) {
|
|
if (*tt == Bnull && (tt[1] == '&' || tt[1] == '\\')) {
|
|
/*
|
|
* The substitution will treat \& and \\
|
|
* specially. We need to leave real \'s
|
|
* as the first character for this to work.
|
|
*/
|
|
*tt = '\\';
|
|
} else {
|
|
chuck(tt--);
|
|
}
|
|
}
|
|
}
|
|
*ptr1end = sav1;
|
|
*ptr3 = sav;
|
|
*ptr = ptr3 - 1;
|
|
if (*ptr3) {
|
|
/* Final terminator is optional. */
|
|
*ptr += charlen;
|
|
}
|
|
break;
|
|
|
|
case '&':
|
|
c = 's';
|
|
break;
|
|
|
|
case 'g':
|
|
(*ptr)++;
|
|
gbal = 1;
|
|
break;
|
|
|
|
case 'w':
|
|
wall = 1;
|
|
(*ptr)++;
|
|
break;
|
|
case 'W':
|
|
wall = 1;
|
|
(*ptr)++;
|
|
ptr1 = get_strarg(ptr2 = *ptr, &charlen);
|
|
if ((sav = *ptr1))
|
|
*ptr1 = '\0';
|
|
sep = dupstring(ptr2 + charlen);
|
|
if (sav)
|
|
*ptr1 = sav;
|
|
*ptr = ptr1 + charlen;
|
|
c = '\0';
|
|
break;
|
|
|
|
case 'f':
|
|
rec = -1;
|
|
(*ptr)++;
|
|
break;
|
|
case 'F':
|
|
(*ptr)++;
|
|
rec = get_intarg(ptr, &dellen);
|
|
break;
|
|
default:
|
|
*ptr = lptr;
|
|
return;
|
|
}
|
|
}
|
|
(*ptr)++;
|
|
if (!c) {
|
|
*ptr = lptr;
|
|
return;
|
|
}
|
|
if (rec < 0)
|
|
test = dupstring(*str);
|
|
|
|
while (rec--) {
|
|
if (wall) {
|
|
al = 0;
|
|
all = NULL;
|
|
for (t = e = *str; (tt = findword(&e, sep));) {
|
|
tc = *e;
|
|
*e = '\0';
|
|
if (c != 'l' && c != 'u')
|
|
copy = dupstring(tt);
|
|
*e = tc;
|
|
switch (c) {
|
|
case 'a':
|
|
chabspath(©);
|
|
break;
|
|
case 'A':
|
|
chrealpath(©);
|
|
break;
|
|
case 'c':
|
|
{
|
|
char *copy2 = equalsubstr(copy, 0, 0);
|
|
if (copy2)
|
|
copy = copy2;
|
|
break;
|
|
}
|
|
case 'h':
|
|
remtpath(©);
|
|
break;
|
|
case 'r':
|
|
remtext(©);
|
|
break;
|
|
case 'e':
|
|
rembutext(©);
|
|
break;
|
|
case 't':
|
|
remlpaths(©);
|
|
break;
|
|
case 'l':
|
|
copy = casemodify(tt, CASMOD_LOWER);
|
|
break;
|
|
case 'u':
|
|
copy = casemodify(tt, CASMOD_UPPER);
|
|
break;
|
|
case 's':
|
|
if (hsubl && hsubr)
|
|
subst(©, hsubl, hsubr, gbal);
|
|
break;
|
|
case 'q':
|
|
copy = quotestring(copy, QT_BACKSLASH_SHOWNULL);
|
|
break;
|
|
case 'Q':
|
|
{
|
|
int one = noerrs, oef = errflag;
|
|
|
|
noerrs = 1;
|
|
parse_subst_string(copy);
|
|
noerrs = one;
|
|
/* Retain any user interrupt error status */
|
|
errflag = oef | (errflag & ERRFLAG_INT);
|
|
remnulargs(copy);
|
|
untokenize(copy);
|
|
}
|
|
break;
|
|
case 'P':
|
|
if (*copy != '/') {
|
|
copy = zhtricat(metafy(zgetcwd(), -1, META_HEAPDUP), "/", copy);
|
|
}
|
|
copy = xsymlink(copy, 1);
|
|
break;
|
|
}
|
|
tc = *tt;
|
|
*tt = '\0';
|
|
nl = al + strlen(t) + strlen(copy);
|
|
ptr1 = tmp = (char *)zhalloc(nl + 1);
|
|
if (all)
|
|
for (ptr2 = all; *ptr2;)
|
|
*ptr1++ = *ptr2++;
|
|
for (ptr2 = t; *ptr2;)
|
|
*ptr1++ = *ptr2++;
|
|
*tt = tc;
|
|
for (ptr2 = copy; *ptr2;)
|
|
*ptr1++ = *ptr2++;
|
|
*ptr1 = '\0';
|
|
al = nl;
|
|
all = tmp;
|
|
t = e;
|
|
}
|
|
if (!all)
|
|
*str = dupstring("");
|
|
else
|
|
*str = all;
|
|
|
|
} else {
|
|
switch (c) {
|
|
case 'a':
|
|
chabspath(str);
|
|
break;
|
|
case 'A':
|
|
chrealpath(str);
|
|
break;
|
|
case 'c':
|
|
{
|
|
char *copy2 = equalsubstr(*str, 0, 0);
|
|
if (copy2)
|
|
*str = copy2;
|
|
break;
|
|
}
|
|
case 'h':
|
|
remtpath(str);
|
|
break;
|
|
case 'r':
|
|
remtext(str);
|
|
break;
|
|
case 'e':
|
|
rembutext(str);
|
|
break;
|
|
case 't':
|
|
remlpaths(str);
|
|
break;
|
|
case 'l':
|
|
*str = casemodify(*str, CASMOD_LOWER);
|
|
break;
|
|
case 'u':
|
|
*str = casemodify(*str, CASMOD_UPPER);
|
|
break;
|
|
case 's':
|
|
if (hsubl && hsubr)
|
|
subst(str, hsubl, hsubr, gbal);
|
|
break;
|
|
case 'q':
|
|
*str = quotestring(*str, QT_BACKSLASH);
|
|
break;
|
|
case 'Q':
|
|
{
|
|
int one = noerrs, oef = errflag;
|
|
|
|
noerrs = 1;
|
|
parse_subst_string(*str);
|
|
noerrs = one;
|
|
/* Retain any user interrupt error status */
|
|
errflag = oef | (errflag & ERRFLAG_INT);
|
|
remnulargs(*str);
|
|
untokenize(*str);
|
|
}
|
|
break;
|
|
case 'P':
|
|
if (**str != '/') {
|
|
*str = zhtricat(metafy(zgetcwd(), -1, META_HEAPDUP), "/", *str);
|
|
}
|
|
*str = xsymlink(*str, 1);
|
|
break;
|
|
}
|
|
}
|
|
if (rec < 0) {
|
|
if (!strcmp(test, *str))
|
|
rec = 0;
|
|
else
|
|
test = dupstring(*str);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* get a directory stack entry */
|
|
|
|
/**/
|
|
static char *
|
|
dstackent(char ch, int val)
|
|
{
|
|
int backwards;
|
|
LinkNode end=(LinkNode)dirstack, n;
|
|
|
|
backwards = ch == (isset(PUSHDMINUS) ? '+' : '-');
|
|
if(!backwards && !val--)
|
|
return pwd;
|
|
if (backwards)
|
|
for (n=lastnode(dirstack); n != end && val; val--, n=prevnode(n));
|
|
else
|
|
for (end=NULL, n=firstnode(dirstack); n && val; val--, n=nextnode(n));
|
|
if (n == end) {
|
|
if (backwards && !val)
|
|
return pwd;
|
|
if (isset(NOMATCH))
|
|
zerr("not enough directory stack entries.");
|
|
return NULL;
|
|
}
|
|
return (char *)getdata(n);
|
|
}
|