diff options
author | harti <harti@FreeBSD.org> | 2005-03-01 12:26:32 +0000 |
---|---|---|
committer | harti <harti@FreeBSD.org> | 2005-03-01 12:26:32 +0000 |
commit | ba225fbbb5139291473bda22f411b852cef1884c (patch) | |
tree | cfe7a831bdcf24753b2a89a8e929c2c16af446b4 | |
parent | 24ff2a5626bb2dd8b12d7c49e4cc8792502ca1ad (diff) | |
download | FreeBSD-src-ba225fbbb5139291473bda22f411b852cef1884c.zip FreeBSD-src-ba225fbbb5139291473bda22f411b852cef1884c.tar.gz |
Split out ParseModifiers from VarParseLong.
Patch: 7.84
Submitted by: Max Okumoto <okumoto@ucsd.edu>
-rw-r--r-- | usr.bin/make/var.c | 1193 |
1 files changed, 603 insertions, 590 deletions
diff --git a/usr.bin/make/var.c b/usr.bin/make/var.c index fb43bf0..06054b8 100644 --- a/usr.bin/make/var.c +++ b/usr.bin/make/var.c @@ -887,6 +887,606 @@ VarExpand(Var *v, GNode *ctxt, Boolean err) } /* + * Now we need to apply any modifiers the user wants applied. + * These are: + * :M<pattern> + * words which match the given <pattern>. + * <pattern> is of the standard file + * wildcarding form. + * :S<d><pat1><d><pat2><d>[g] + * Substitute <pat2> for <pat1> in the value + * :C<d><pat1><d><pat2><d>[g] + * Substitute <pat2> for regex <pat1> in the value + * :H Substitute the head of each word + * :T Substitute the tail of each word + * :E Substitute the extension (minus '.') of + * each word + * :R Substitute the root of each word + * (pathname minus the suffix). + * :lhs=rhs + * Like :S, but the rhs goes to the end of + * the invocation. + * :U Converts variable to upper-case. + * :L Converts variable to lower-case. + * + * XXXHB update this comment or remove it and point to the man page. + */ +static char * +ParseModifier(const char input[], char tstr[], + char startc, char endc, Boolean dynamic, Var *v, + GNode *ctxt, Boolean err, size_t *lengthPtr, Boolean *freePtr) +{ + char *rw_str; + char *cp; + + rw_str = VarExpand(v, ctxt, err); + *freePtr = TRUE; + + tstr++; + while (*tstr != endc) { + char *newStr; /* New value to return */ + char termc; /* Character which terminated scan */ + + DEBUGF(VAR, ("Applying :%c to \"%s\"\n", *tstr, rw_str)); + switch (*tstr) { + case 'N': + case 'M': + { + char *pattern; + char *cp2; + Boolean copy; + + copy = FALSE; + for (cp = tstr + 1; + *cp != '\0' && *cp != ':' && *cp != endc; + cp++) + { + if (*cp == '\\' && (cp[1] == ':' || cp[1] == endc)) { + copy = TRUE; + cp++; + } + } + termc = *cp; + *cp = '\0'; + if (copy) { + /* + * Need to compress the \:'s out of the pattern, so + * allocate enough room to hold the uncompressed + * pattern (note that cp started at tstr+1, so + * cp - tstr takes the null byte into account) and + * compress the pattern into the space. + */ + pattern = emalloc(cp - tstr); + for (cp2 = pattern, cp = tstr + 1; + *cp != '\0'; + cp++, cp2++) + { + if ((*cp == '\\') && + (cp[1] == ':' || cp[1] == endc)) { + cp++; + } + *cp2 = *cp; + } + *cp2 = '\0'; + } else { + pattern = &tstr[1]; + } + if (*tstr == 'M' || *tstr == 'm') { + newStr = VarModify(rw_str, VarMatch, pattern); + } else { + newStr = VarModify(rw_str, VarNoMatch, pattern); + } + if (copy) { + free(pattern); + } + break; + } + case 'S': + { + VarPattern pattern; + char del; + Buffer *buf; /* Buffer for patterns */ + + pattern.flags = 0; + del = tstr[1]; + tstr += 2; + + /* + * If pattern begins with '^', it is anchored to the + * start of the word -- skip over it and flag pattern. + */ + if (*tstr == '^') { + pattern.flags |= VAR_MATCH_START; + tstr += 1; + } + + buf = Buf_Init(0); + + /* + * Pass through the lhs looking for 1) escaped delimiters, + * '$'s and backslashes (place the escaped character in + * uninterpreted) and 2) unescaped $'s that aren't before + * the delimiter (expand the variable substitution). + * The result is left in the Buffer buf. + */ + for (cp = tstr; *cp != '\0' && *cp != del; cp++) { + if ((*cp == '\\') && + ((cp[1] == del) || + (cp[1] == '$') || + (cp[1] == '\\'))) + { + Buf_AddByte(buf, (Byte)cp[1]); + cp++; + } else if (*cp == '$') { + if (cp[1] != del) { + /* + * If unescaped dollar sign not before the + * delimiter, assume it's a variable + * substitution and recurse. + */ + char *cp2; + size_t len; + Boolean freeIt; + + cp2 = Var_Parse(cp, ctxt, err, &len, &freeIt); + Buf_Append(buf, cp2); + if (freeIt) { + free(cp2); + } + cp += len - 1; + } else { + /* + * Unescaped $ at end of pattern => anchor + * pattern at end. + */ + pattern.flags |= VAR_MATCH_END; + } + } else { + Buf_AddByte(buf, (Byte)*cp); + } + } + + Buf_AddByte(buf, (Byte)'\0'); + + /* + * If lhs didn't end with the delimiter, complain and + * exit. + */ + if (*cp != del) { + Fatal("Unclosed substitution for %s (%c missing)", + v->name, del); + } + + /* + * Fetch pattern and destroy buffer, but preserve the data + * in it, since that's our lhs. Note that Buf_GetAll + * will return the actual number of bytes, which includes + * the null byte, so we have to decrement the length by + * one. + */ + pattern.lhs = (char *)Buf_GetAll(buf, &pattern.leftLen); + pattern.leftLen--; + Buf_Destroy(buf, FALSE); + + /* + * Now comes the replacement string. Three things need to + * be done here: 1) need to compress escaped delimiters and + * ampersands and 2) need to replace unescaped ampersands + * with the l.h.s. (since this isn't regexp, we can do + * it right here) and 3) expand any variable substitutions. + */ + buf = Buf_Init(0); + + tstr = cp + 1; + for (cp = tstr; *cp != '\0' && *cp != del; cp++) { + if ((*cp == '\\') && + ((cp[1] == del) || + (cp[1] == '&') || + (cp[1] == '\\') || + (cp[1] == '$'))) + { + Buf_AddByte(buf, (Byte)cp[1]); + cp++; + } else if ((*cp == '$') && (cp[1] != del)) { + char *cp2; + size_t len; + Boolean freeIt; + + cp2 = Var_Parse(cp, ctxt, err, &len, &freeIt); + Buf_Append(buf, cp2); + cp += len - 1; + if (freeIt) { + free(cp2); + } + } else if (*cp == '&') { + Buf_AddBytes(buf, pattern.leftLen, + (Byte *)pattern.lhs); + } else { + Buf_AddByte(buf, (Byte)*cp); + } + } + + Buf_AddByte(buf, (Byte)'\0'); + + /* + * If didn't end in delimiter character, complain + */ + if (*cp != del) { + Fatal("Unclosed substitution for %s (%c missing)", + v->name, del); + } + + pattern.rhs = (char *)Buf_GetAll(buf, &pattern.rightLen); + pattern.rightLen--; + Buf_Destroy(buf, FALSE); + + /* + * Check for global substitution. If 'g' after the final + * delimiter, substitution is global and is marked that + * way. + */ + cp++; + if (*cp == 'g') { + pattern.flags |= VAR_SUB_GLOBAL; + cp++; + } + + /* + * Global substitution of the empty string causes an + * infinite number of matches, unless anchored by '^' + * (start of string) or '$' (end of string). Catch the + * infinite substitution here. + * Note that flags can only contain the 3 bits we're + * interested in so we don't have to mask unrelated + * bits. We can test for equality. + */ + if (!pattern.leftLen && pattern.flags == VAR_SUB_GLOBAL) + Fatal("Global substitution of the empty string"); + + termc = *cp; + newStr = VarModify(rw_str, VarSubstitute, &pattern); + /* + * Free the two strings. + */ + free(pattern.lhs); + free(pattern.rhs); + break; + } + case 'C': + { + int delim; + VarREPattern pattern; + char *re; + int error; + + pattern.flags = 0; + delim = tstr[1]; + tstr += 2; + + cp = tstr; + + if ((re = VarGetPattern(ctxt, err, &cp, delim, NULL, + NULL, NULL)) == NULL) { + /* was: goto cleanup */ + *lengthPtr = cp - input + 1; + if (*freePtr) + free(rw_str); + if (delim != '\0') + Fatal("Unclosed substitution for %s (%c missing)", + v->name, delim); + return (var_Error); + } + + if ((pattern.replace = VarGetPattern(ctxt, err, &cp, + delim, NULL, NULL, NULL)) == NULL){ + free(re); + + /* was: goto cleanup */ + *lengthPtr = cp - input + 1; + if (*freePtr) + free(rw_str); + if (delim != '\0') + Fatal("Unclosed substitution for %s (%c missing)", + v->name, delim); + return (var_Error); + } + + for (;; cp++) { + switch (*cp) { + case 'g': + pattern.flags |= VAR_SUB_GLOBAL; + continue; + case '1': + pattern.flags |= VAR_SUB_ONE; + continue; + default: + break; + } + break; + } + + termc = *cp; + + error = regcomp(&pattern.re, re, REG_EXTENDED); + free(re); + if (error) { + *lengthPtr = cp - input + 1; + VarREError(error, &pattern.re, "RE substitution error"); + free(pattern.replace); + return (var_Error); + } + + pattern.nsub = pattern.re.re_nsub + 1; + if (pattern.nsub < 1) + pattern.nsub = 1; + if (pattern.nsub > 10) + pattern.nsub = 10; + pattern.matches = emalloc(pattern.nsub * + sizeof(regmatch_t)); + newStr = VarModify(rw_str, VarRESubstitute, &pattern); + regfree(&pattern.re); + free(pattern.replace); + free(pattern.matches); + break; + } + case 'L': + if (tstr[1] == endc || tstr[1] == ':') { + Buffer *buf; + buf = Buf_Init(MAKE_BSIZE); + for (cp = rw_str; *cp ; cp++) + Buf_AddByte(buf, (Byte)tolower(*cp)); + + Buf_AddByte(buf, (Byte)'\0'); + newStr = (char *)Buf_GetAll(buf, (size_t *)NULL); + Buf_Destroy(buf, FALSE); + + cp = tstr + 1; + termc = *cp; + break; + } + /* FALLTHROUGH */ + case 'O': + if (tstr[1] == endc || tstr[1] == ':') { + newStr = VarSortWords(rw_str, SortIncreasing); + cp = tstr + 1; + termc = *cp; + break; + } + /* FALLTHROUGH */ + case 'Q': + if (tstr[1] == endc || tstr[1] == ':') { + newStr = Var_Quote(rw_str); + cp = tstr + 1; + termc = *cp; + break; + } + /*FALLTHRU*/ + case 'T': + if (tstr[1] == endc || tstr[1] == ':') { + newStr = VarModify(rw_str, VarTail, (void *)NULL); + cp = tstr + 1; + termc = *cp; + break; + } + /*FALLTHRU*/ + case 'U': + if (tstr[1] == endc || tstr[1] == ':') { + Buffer *buf; + buf = Buf_Init(MAKE_BSIZE); + for (cp = rw_str; *cp ; cp++) + Buf_AddByte(buf, (Byte)toupper(*cp)); + + Buf_AddByte(buf, (Byte)'\0'); + newStr = (char *)Buf_GetAll(buf, (size_t *)NULL); + Buf_Destroy(buf, FALSE); + + cp = tstr + 1; + termc = *cp; + break; + } + /* FALLTHROUGH */ + case 'H': + if (tstr[1] == endc || tstr[1] == ':') { + newStr = VarModify(rw_str, VarHead, (void *)NULL); + cp = tstr + 1; + termc = *cp; + break; + } + /*FALLTHRU*/ + case 'E': + if (tstr[1] == endc || tstr[1] == ':') { + newStr = VarModify(rw_str, VarSuffix, (void *)NULL); + cp = tstr + 1; + termc = *cp; + break; + } + /*FALLTHRU*/ + case 'R': + if (tstr[1] == endc || tstr[1] == ':') { + newStr = VarModify(rw_str, VarRoot, (void *)NULL); + cp = tstr + 1; + termc = *cp; + break; + } + /*FALLTHRU*/ +#ifdef SUNSHCMD + case 's': + if (tstr[1] == 'h' && (tstr[2] == endc || tstr[2] == ':')) { + const char *error; + Buffer *buf; + + buf = Cmd_Exec(rw_str, &error); + newStr = Buf_GetAll(buf, NULL); + Buf_Destroy(buf, FALSE); + + if (error) + Error(error, rw_str); + cp = tstr + 2; + termc = *cp; + break; + } + /*FALLTHRU*/ +#endif + default: + { +#ifdef SYSVVARSUB + /* + * This can either be a bogus modifier or a System-V + * substitution command. + */ + VarPattern pattern; + Boolean eqFound; + int cnt; + + pattern.flags = 0; + eqFound = FALSE; + /* + * First we make a pass through the string trying + * to verify it is a SYSV-make-style translation: + * it must be: <string1>=<string2>) + */ + cp = tstr; + cnt = 1; + while (*cp != '\0' && cnt) { + if (*cp == '=') { + eqFound = TRUE; + /* continue looking for endc */ + } + else if (*cp == endc) + cnt--; + else if (*cp == startc) + cnt++; + if (cnt) + cp++; + } + if (*cp == endc && eqFound) { + int delim; + + /* + * Now we break this sucker into the lhs and + * rhs. We must null terminate them of course. + */ + cp = tstr; + + delim = '='; + if ((pattern.lhs = VarGetPattern(ctxt, + err, &cp, delim, &pattern.flags, &pattern.leftLen, + NULL)) == NULL) { + /* was: goto cleanup */ + *lengthPtr = cp - input + 1; + if (*freePtr) + free(rw_str); + if (delim != '\0') + Fatal("Unclosed substitution for %s (%c missing)", + v->name, delim); + return (var_Error); + } + + delim = endc; + if ((pattern.rhs = VarGetPattern(ctxt, + err, &cp, delim, NULL, &pattern.rightLen, + &pattern)) == NULL) { + /* was: goto cleanup */ + *lengthPtr = cp - input + 1; + if (*freePtr) + free(rw_str); + if (delim != '\0') + Fatal("Unclosed substitution for %s (%c missing)", + v->name, delim); + return (var_Error); + } + + /* + * SYSV modifications happen through the whole + * string. Note the pattern is anchored at the end. + */ + termc = *--cp; + delim = '\0'; + newStr = VarModify(rw_str, VarSYSVMatch, &pattern); + + free(pattern.lhs); + free(pattern.rhs); + + termc = endc; + } else +#endif + { + Error("Unknown modifier '%c'\n", *tstr); + for (cp = tstr+1; + *cp != ':' && *cp != endc && *cp != '\0'; + cp++) + continue; + termc = *cp; + newStr = var_Error; + } + } + } + DEBUGF(VAR, ("Result is \"%s\"\n", newStr)); + + if (*freePtr) { + free(rw_str); + } + rw_str = newStr; + if (rw_str != var_Error) { + *freePtr = TRUE; + } else { + *freePtr = FALSE; + } + if (termc == '\0') { + Error("Unclosed variable specification for %s", v->name); + } else if (termc == ':') { + *cp++ = termc; + } else { + *cp = termc; + } + tstr = cp; + } + + if (v->flags & VAR_FROM_ENV) { + if (rw_str == (char *)Buf_GetAll(v->val, (size_t *)NULL)) { + /* + * Returning the value unmodified, so tell the caller to free + * the thing. + */ + *freePtr = TRUE; + *lengthPtr = tstr - input + 1; + VarDestroy(v, FALSE); + return (rw_str); + } else { + *lengthPtr = tstr - input + 1; + VarDestroy(v, TRUE); + return (rw_str); + } + } else if (v->flags & VAR_JUNK) { + /* + * Perform any free'ing needed and set *freePtr to FALSE so the caller + * doesn't try to free a static pointer. + */ + if (*freePtr) { + free(rw_str); + } + if (dynamic) { + *freePtr = FALSE; + *lengthPtr = tstr - input + 1; + VarDestroy(v, TRUE); + rw_str = emalloc(*lengthPtr + 1); + strncpy(rw_str, input, *lengthPtr); + rw_str[*lengthPtr] = '\0'; + *freePtr = TRUE; + return (rw_str); + } else { + *freePtr = FALSE; + *lengthPtr = tstr - input + 1; + VarDestroy(v, TRUE); + return (err ? var_Error : varNoError); + } + } else { + *lengthPtr = tstr - input + 1; + return (rw_str); + } +} + +/* * Check if brackets contain a variable name. */ static char * @@ -1085,597 +1685,10 @@ VarParseLong(char foo[], GNode *ctxt, Boolean err, size_t *lengthPtr, Buf_Destroy(buf, TRUE); - /* - * Now we need to apply any modifiers the user wants applied. - * These are: - * :M<pattern> - * words which match the given <pattern>. - * <pattern> is of the standard file - * wildcarding form. - * :S<d><pat1><d><pat2><d>[g] - * Substitute <pat2> for <pat1> in the value - * :C<d><pat1><d><pat2><d>[g] - * Substitute <pat2> for regex <pat1> in the value - * :H Substitute the head of each word - * :T Substitute the tail of each word - * :E Substitute the extension (minus '.') of - * each word - * :R Substitute the root of each word - * (pathname minus the suffix). - * :lhs=rhs - * Like :S, but the rhs goes to the end of - * the invocation. - * :U Converts variable to upper-case. - * :L Converts variable to lower-case. - */ if (haveModifier) { - char *cp; - - rw_str = VarExpand(v, ctxt, err); - *freePtr = TRUE; - - tstr++; - while (*tstr != endc) { - char *newStr; /* New value to return */ - char termc; /* Character which terminated scan */ - - DEBUGF(VAR, ("Applying :%c to \"%s\"\n", *tstr, rw_str)); - switch (*tstr) { - case 'N': - case 'M': - { - char *pattern; - char *cp2; - Boolean copy; - - copy = FALSE; - for (cp = tstr + 1; - *cp != '\0' && *cp != ':' && *cp != endc; - cp++) - { - if (*cp == '\\' && (cp[1] == ':' || cp[1] == endc)) { - copy = TRUE; - cp++; - } - } - termc = *cp; - *cp = '\0'; - if (copy) { - /* - * Need to compress the \:'s out of the pattern, so - * allocate enough room to hold the uncompressed - * pattern (note that cp started at tstr+1, so - * cp - tstr takes the null byte into account) and - * compress the pattern into the space. - */ - pattern = emalloc(cp - tstr); - for (cp2 = pattern, cp = tstr + 1; - *cp != '\0'; - cp++, cp2++) - { - if ((*cp == '\\') && - (cp[1] == ':' || cp[1] == endc)) { - cp++; - } - *cp2 = *cp; - } - *cp2 = '\0'; - } else { - pattern = &tstr[1]; - } - if (*tstr == 'M' || *tstr == 'm') { - newStr = VarModify(rw_str, VarMatch, pattern); - } else { - newStr = VarModify(rw_str, VarNoMatch, pattern); - } - if (copy) { - free(pattern); - } - break; - } - case 'S': - { - VarPattern pattern; - char del; - Buffer *buf; /* Buffer for patterns */ - - pattern.flags = 0; - del = tstr[1]; - tstr += 2; - - /* - * If pattern begins with '^', it is anchored to the - * start of the word -- skip over it and flag pattern. - */ - if (*tstr == '^') { - pattern.flags |= VAR_MATCH_START; - tstr += 1; - } - - buf = Buf_Init(0); - - /* - * Pass through the lhs looking for 1) escaped delimiters, - * '$'s and backslashes (place the escaped character in - * uninterpreted) and 2) unescaped $'s that aren't before - * the delimiter (expand the variable substitution). - * The result is left in the Buffer buf. - */ - for (cp = tstr; *cp != '\0' && *cp != del; cp++) { - if ((*cp == '\\') && - ((cp[1] == del) || - (cp[1] == '$') || - (cp[1] == '\\'))) - { - Buf_AddByte(buf, (Byte)cp[1]); - cp++; - } else if (*cp == '$') { - if (cp[1] != del) { - /* - * If unescaped dollar sign not before the - * delimiter, assume it's a variable - * substitution and recurse. - */ - char *cp2; - size_t len; - Boolean freeIt; - - cp2 = Var_Parse(cp, ctxt, err, &len, &freeIt); - Buf_Append(buf, cp2); - if (freeIt) { - free(cp2); - } - cp += len - 1; - } else { - /* - * Unescaped $ at end of pattern => anchor - * pattern at end. - */ - pattern.flags |= VAR_MATCH_END; - } - } else { - Buf_AddByte(buf, (Byte)*cp); - } - } - - Buf_AddByte(buf, (Byte)'\0'); - - /* - * If lhs didn't end with the delimiter, complain and - * exit. - */ - if (*cp != del) { - Fatal("Unclosed substitution for %s (%c missing)", - v->name, del); - } - - /* - * Fetch pattern and destroy buffer, but preserve the data - * in it, since that's our lhs. Note that Buf_GetAll - * will return the actual number of bytes, which includes - * the null byte, so we have to decrement the length by - * one. - */ - pattern.lhs = (char *)Buf_GetAll(buf, &pattern.leftLen); - pattern.leftLen--; - Buf_Destroy(buf, FALSE); - - /* - * Now comes the replacement string. Three things need to - * be done here: 1) need to compress escaped delimiters and - * ampersands and 2) need to replace unescaped ampersands - * with the l.h.s. (since this isn't regexp, we can do - * it right here) and 3) expand any variable substitutions. - */ - buf = Buf_Init(0); - - tstr = cp + 1; - for (cp = tstr; *cp != '\0' && *cp != del; cp++) { - if ((*cp == '\\') && - ((cp[1] == del) || - (cp[1] == '&') || - (cp[1] == '\\') || - (cp[1] == '$'))) - { - Buf_AddByte(buf, (Byte)cp[1]); - cp++; - } else if ((*cp == '$') && (cp[1] != del)) { - char *cp2; - size_t len; - Boolean freeIt; - - cp2 = Var_Parse(cp, ctxt, err, &len, &freeIt); - Buf_Append(buf, cp2); - cp += len - 1; - if (freeIt) { - free(cp2); - } - } else if (*cp == '&') { - Buf_AddBytes(buf, pattern.leftLen, - (Byte *)pattern.lhs); - } else { - Buf_AddByte(buf, (Byte)*cp); - } - } - - Buf_AddByte(buf, (Byte)'\0'); - - /* - * If didn't end in delimiter character, complain - */ - if (*cp != del) { - Fatal("Unclosed substitution for %s (%c missing)", - v->name, del); - } - - pattern.rhs = (char *)Buf_GetAll(buf, &pattern.rightLen); - pattern.rightLen--; - Buf_Destroy(buf, FALSE); - - /* - * Check for global substitution. If 'g' after the final - * delimiter, substitution is global and is marked that - * way. - */ - cp++; - if (*cp == 'g') { - pattern.flags |= VAR_SUB_GLOBAL; - cp++; - } - - /* - * Global substitution of the empty string causes an - * infinite number of matches, unless anchored by '^' - * (start of string) or '$' (end of string). Catch the - * infinite substitution here. - * Note that flags can only contain the 3 bits we're - * interested in so we don't have to mask unrelated - * bits. We can test for equality. - */ - if (!pattern.leftLen && pattern.flags == VAR_SUB_GLOBAL) - Fatal("Global substitution of the empty string"); - - termc = *cp; - newStr = VarModify(rw_str, VarSubstitute, &pattern); - /* - * Free the two strings. - */ - free(pattern.lhs); - free(pattern.rhs); - break; - } - case 'C': - { - int delim; - VarREPattern pattern; - char *re; - int error; - - pattern.flags = 0; - delim = tstr[1]; - tstr += 2; - - cp = tstr; - - if ((re = VarGetPattern(ctxt, err, &cp, delim, NULL, - NULL, NULL)) == NULL) { - /* was: goto cleanup */ - *lengthPtr = cp - input + 1; - if (*freePtr) - free(rw_str); - if (delim != '\0') - Fatal("Unclosed substitution for %s (%c missing)", - v->name, delim); - return (var_Error); - } - - if ((pattern.replace = VarGetPattern(ctxt, err, &cp, - delim, NULL, NULL, NULL)) == NULL){ - free(re); - - /* was: goto cleanup */ - *lengthPtr = cp - input + 1; - if (*freePtr) - free(rw_str); - if (delim != '\0') - Fatal("Unclosed substitution for %s (%c missing)", - v->name, delim); - return (var_Error); - } - - for (;; cp++) { - switch (*cp) { - case 'g': - pattern.flags |= VAR_SUB_GLOBAL; - continue; - case '1': - pattern.flags |= VAR_SUB_ONE; - continue; - default: - break; - } - break; - } - - termc = *cp; - - error = regcomp(&pattern.re, re, REG_EXTENDED); - free(re); - if (error) { - *lengthPtr = cp - input + 1; - VarREError(error, &pattern.re, "RE substitution error"); - free(pattern.replace); - return (var_Error); - } - - pattern.nsub = pattern.re.re_nsub + 1; - if (pattern.nsub < 1) - pattern.nsub = 1; - if (pattern.nsub > 10) - pattern.nsub = 10; - pattern.matches = emalloc(pattern.nsub * - sizeof(regmatch_t)); - newStr = VarModify(rw_str, VarRESubstitute, &pattern); - regfree(&pattern.re); - free(pattern.replace); - free(pattern.matches); - break; - } - case 'L': - if (tstr[1] == endc || tstr[1] == ':') { - Buffer *buf; - buf = Buf_Init(MAKE_BSIZE); - for (cp = rw_str; *cp ; cp++) - Buf_AddByte(buf, (Byte)tolower(*cp)); - - Buf_AddByte(buf, (Byte)'\0'); - newStr = (char *)Buf_GetAll(buf, (size_t *)NULL); - Buf_Destroy(buf, FALSE); - - cp = tstr + 1; - termc = *cp; - break; - } - /* FALLTHROUGH */ - case 'O': - if (tstr[1] == endc || tstr[1] == ':') { - newStr = VarSortWords(rw_str, SortIncreasing); - cp = tstr + 1; - termc = *cp; - break; - } - /* FALLTHROUGH */ - case 'Q': - if (tstr[1] == endc || tstr[1] == ':') { - newStr = Var_Quote(rw_str); - cp = tstr + 1; - termc = *cp; - break; - } - /*FALLTHRU*/ - case 'T': - if (tstr[1] == endc || tstr[1] == ':') { - newStr = VarModify(rw_str, VarTail, (void *)NULL); - cp = tstr + 1; - termc = *cp; - break; - } - /*FALLTHRU*/ - case 'U': - if (tstr[1] == endc || tstr[1] == ':') { - Buffer *buf; - buf = Buf_Init(MAKE_BSIZE); - for (cp = rw_str; *cp ; cp++) - Buf_AddByte(buf, (Byte)toupper(*cp)); - - Buf_AddByte(buf, (Byte)'\0'); - newStr = (char *)Buf_GetAll(buf, (size_t *)NULL); - Buf_Destroy(buf, FALSE); - - cp = tstr + 1; - termc = *cp; - break; - } - /* FALLTHROUGH */ - case 'H': - if (tstr[1] == endc || tstr[1] == ':') { - newStr = VarModify(rw_str, VarHead, (void *)NULL); - cp = tstr + 1; - termc = *cp; - break; - } - /*FALLTHRU*/ - case 'E': - if (tstr[1] == endc || tstr[1] == ':') { - newStr = VarModify(rw_str, VarSuffix, (void *)NULL); - cp = tstr + 1; - termc = *cp; - break; - } - /*FALLTHRU*/ - case 'R': - if (tstr[1] == endc || tstr[1] == ':') { - newStr = VarModify(rw_str, VarRoot, (void *)NULL); - cp = tstr + 1; - termc = *cp; - break; - } - /*FALLTHRU*/ - #ifdef SUNSHCMD - case 's': - if (tstr[1] == 'h' && (tstr[2] == endc || tstr[2] == ':')) { - const char *error; - Buffer *buf; - - buf = Cmd_Exec(rw_str, &error); - newStr = Buf_GetAll(buf, NULL); - Buf_Destroy(buf, FALSE); - - if (error) - Error(error, rw_str); - cp = tstr + 2; - termc = *cp; - break; - } - /*FALLTHRU*/ - #endif - default: - { - #ifdef SYSVVARSUB - /* - * This can either be a bogus modifier or a System-V - * substitution command. - */ - VarPattern pattern; - Boolean eqFound; - int cnt; - - pattern.flags = 0; - eqFound = FALSE; - /* - * First we make a pass through the string trying - * to verify it is a SYSV-make-style translation: - * it must be: <string1>=<string2>) - */ - cp = tstr; - cnt = 1; - while (*cp != '\0' && cnt) { - if (*cp == '=') { - eqFound = TRUE; - /* continue looking for endc */ - } - else if (*cp == endc) - cnt--; - else if (*cp == startc) - cnt++; - if (cnt) - cp++; - } - if (*cp == endc && eqFound) { - int delim; - - /* - * Now we break this sucker into the lhs and - * rhs. We must null terminate them of course. - */ - cp = tstr; - - delim = '='; - if ((pattern.lhs = VarGetPattern(ctxt, - err, &cp, delim, &pattern.flags, &pattern.leftLen, - NULL)) == NULL) { - /* was: goto cleanup */ - *lengthPtr = cp - input + 1; - if (*freePtr) - free(rw_str); - if (delim != '\0') - Fatal("Unclosed substitution for %s (%c missing)", - v->name, delim); - return (var_Error); - } - - delim = endc; - if ((pattern.rhs = VarGetPattern(ctxt, - err, &cp, delim, NULL, &pattern.rightLen, - &pattern)) == NULL) { - /* was: goto cleanup */ - *lengthPtr = cp - input + 1; - if (*freePtr) - free(rw_str); - if (delim != '\0') - Fatal("Unclosed substitution for %s (%c missing)", - v->name, delim); - return (var_Error); - } - - /* - * SYSV modifications happen through the whole - * string. Note the pattern is anchored at the end. - */ - termc = *--cp; - delim = '\0'; - newStr = VarModify(rw_str, VarSYSVMatch, &pattern); - - free(pattern.lhs); - free(pattern.rhs); - - termc = endc; - } else - #endif - { - Error("Unknown modifier '%c'\n", *tstr); - for (cp = tstr+1; - *cp != ':' && *cp != endc && *cp != '\0'; - cp++) - continue; - termc = *cp; - newStr = var_Error; - } - } - } - DEBUGF(VAR, ("Result is \"%s\"\n", newStr)); - - if (*freePtr) { - free(rw_str); - } - rw_str = newStr; - if (rw_str != var_Error) { - *freePtr = TRUE; - } else { - *freePtr = FALSE; - } - if (termc == '\0') { - Error("Unclosed variable specification for %s", v->name); - } else if (termc == ':') { - *cp++ = termc; - } else { - *cp = termc; - } - tstr = cp; - } - - if (v->flags & VAR_FROM_ENV) { - if (rw_str == (char *)Buf_GetAll(v->val, (size_t *)NULL)) { - /* - * Returning the value unmodified, so tell the caller to free - * the thing. - */ - *freePtr = TRUE; - *lengthPtr = tstr - input + 1; - VarDestroy(v, FALSE); - return (rw_str); - } else { - *lengthPtr = tstr - input + 1; - VarDestroy(v, TRUE); - return (rw_str); - } - } else if (v->flags & VAR_JUNK) { - /* - * Perform any free'ing needed and set *freePtr to FALSE so the caller - * doesn't try to free a static pointer. - */ - if (*freePtr) { - free(rw_str); - } - if (dynamic) { - *freePtr = FALSE; - *lengthPtr = tstr - input + 1; - VarDestroy(v, TRUE); - rw_str = emalloc(*lengthPtr + 1); - strncpy(rw_str, input, *lengthPtr); - rw_str[*lengthPtr] = '\0'; - *freePtr = TRUE; - return (rw_str); - } else { - *freePtr = FALSE; - *lengthPtr = tstr - input + 1; - VarDestroy(v, TRUE); - return (err ? var_Error : varNoError); - } - } else { - *lengthPtr = tstr - input + 1; - return (rw_str); - } + return (ParseModifier(input, tstr, + startc, endc, dynamic, v, + ctxt, err, lengthPtr, freePtr)); } else { char *result; |