- Added support for backreferences in substitution expressions up to nine

(\1, \2...\9). This touched a lot of places in this file and I added a new
   function 'print_subst_w_backrefs' in order to keep 'do_subst_command' a
   little more tidy.

	* I tested this good 'n hard, but will always appreciate more testing from
	  other, willing folks.

 - Noticed that the index_of_next_unescaped_slash was subtly wrong so I
   changed both the functionality and behavior (it used to skip over the first
   char in the string you passed it, assuming it was a leading '/'--this
   assumption is no longer made) this necessitated changing the lines that
   call this function just slightly.
This commit is contained in:
Mark Whitley 2000-07-17 20:06:42 +00:00
parent 21ddb38fcf
commit 97562bd9d7
2 changed files with 152 additions and 36 deletions

View File

@ -27,6 +27,7 @@
- address matching: num|/matchstr/[,num|/matchstr/|$]command - address matching: num|/matchstr/[,num|/matchstr/|$]command
- commands: (p)rint, (d)elete, (s)ubstitue (with g & I flags) - commands: (p)rint, (d)elete, (s)ubstitue (with g & I flags)
- edit commands: (a)ppend, (i)nsert, (c)hange - edit commands: (a)ppend, (i)nsert, (c)hange
- backreferences in substitution expressions (\1, \2...\9)
(Note: Specifying an address (range) to match is *optional*; commands (Note: Specifying an address (range) to match is *optional*; commands
default to the whole pattern space if no specific address match was default to the whole pattern space if no specific address match was
@ -73,6 +74,9 @@ struct sed_cmd {
/* substitution command specific fields */ /* substitution command specific fields */
regex_t *sub_match; /* sed -e 's/sub_match/replace/' */ regex_t *sub_match; /* sed -e 's/sub_match/replace/' */
char *replace; /* sed -e 's/sub_match/replace/' XXX: who will hold the \1 \2 \3s? */ char *replace; /* sed -e 's/sub_match/replace/' XXX: who will hold the \1 \2 \3s? */
unsigned int num_backrefs:4; /* how many back references (\1..\9) */
/* Note: GNU/POSIX sed does not save more than nine backrefs, so
* we only use 4 bits to hold the number */
unsigned int sub_g:1; /* sed -e 's/foo/bar/g' (global) */ unsigned int sub_g:1; /* sed -e 's/foo/bar/g' (global) */
/* edit command (a,i,c) speicific field */ /* edit command (a,i,c) speicific field */
@ -166,19 +170,19 @@ static size_t strrspn(const char *s, const char *accept)
#endif #endif
/* /*
* index_of_unescaped_slash - walks left to right through a string beginning * index_of_next_unescaped_slash - walks left to right through a string
* at a specified index and returns the index of the next unescaped slash. * beginning at a specified index and returns the index of the next forward
* slash ('/') not preceeded by a backslash ('\').
*/ */
static int index_of_next_unescaped_slash(const char *str, int idx) static int index_of_next_unescaped_slash(const char *str, int idx)
{ {
do { for ( ; str[idx]; idx++) {
idx++; if (str[idx] == '/' && str[idx-1] != '\\')
/* test if we've hit the end */ return idx;
if (str[idx] == 0) }
return -1;
} while (str[idx] != '/' && str[idx - 1] != '\\');
return idx; /* if we make it to here, we've hit the end of the string */
return -1;
} }
/* /*
@ -201,7 +205,7 @@ static int get_address(const char *str, int *line, regex_t **regex)
idx++; idx++;
} }
else if (my_str[idx] == '/') { else if (my_str[idx] == '/') {
idx = index_of_next_unescaped_slash(my_str, idx); idx = index_of_next_unescaped_slash(my_str, ++idx);
if (idx == -1) if (idx == -1)
fatalError("unterminated match expression\n"); fatalError("unterminated match expression\n");
my_str[idx] = '\0'; my_str[idx] = '\0';
@ -233,6 +237,7 @@ static int parse_subst_cmd(struct sed_cmd *sed_cmd, const char *substr)
int oldidx, cflags = REG_NEWLINE; int oldidx, cflags = REG_NEWLINE;
char *match; char *match;
int idx = 0; int idx = 0;
int j;
/* /*
* the string that gets passed to this function should look like this: * the string that gets passed to this function should look like this:
@ -249,14 +254,26 @@ static int parse_subst_cmd(struct sed_cmd *sed_cmd, const char *substr)
/* save the match string */ /* save the match string */
oldidx = idx+1; oldidx = idx+1;
idx = index_of_next_unescaped_slash(substr, idx); idx = index_of_next_unescaped_slash(substr, ++idx);
if (idx == -1) if (idx == -1)
fatalError("bad format in substitution expression\n"); fatalError("bad format in substitution expression\n");
match = strdup_substr(substr, oldidx, idx); match = strdup_substr(substr, oldidx, idx);
/* determine the number of back references in the match string */
/* Note: we compute this here rather than in the do_subst_command()
* function to save processor time, at the expense of a little more memory
* (4 bits) per sed_cmd */
/* sed_cmd->num_backrefs = 0; */ /* XXX: not needed? --apparently not */
for (j = 0; match[j]; j++) {
/* GNU/POSIX sed does not save more than nine backrefs */
if (match[j] == '\\' && match[j+1] == '(' && sed_cmd->num_backrefs < 9)
sed_cmd->num_backrefs++;
}
/* save the replacement string */ /* save the replacement string */
oldidx = idx+1; oldidx = idx+1;
idx = index_of_next_unescaped_slash(substr, idx); idx = index_of_next_unescaped_slash(substr, ++idx);
if (idx == -1) if (idx == -1)
fatalError("bad format in substitution expression\n"); fatalError("bad format in substitution expression\n");
sed_cmd->replace = strdup_substr(substr, oldidx, idx); sed_cmd->replace = strdup_substr(substr, oldidx, idx);
@ -280,7 +297,7 @@ static int parse_subst_cmd(struct sed_cmd *sed_cmd, const char *substr)
} }
out: out:
/* compile the regex */ /* compile the match string into a regex */
sed_cmd->sub_match = (regex_t *)xmalloc(sizeof(regex_t)); sed_cmd->sub_match = (regex_t *)xmalloc(sizeof(regex_t));
xregcomp(sed_cmd->sub_match, match, cflags); xregcomp(sed_cmd->sub_match, match, cflags);
free(match); free(match);
@ -460,26 +477,64 @@ static void load_cmd_file(char *filename)
} }
} }
static void print_subst_w_backrefs(const char *line, const char *replace, regmatch_t *regmatch)
{
int i;
/* go through the replacement string */
for (i = 0; replace[i]; i++) {
/* if we find a backreference (\1, \2, etc.) print the backref'ed * text */
if (replace[i] == '\\' && isdigit(replace[i+1])) {
int j;
char tmpstr[2];
int backref;
++i; /* i now indexes the backref number, instead of the leading slash */
tmpstr[0] = replace[i];
tmpstr[1] = 0;
backref = atoi(tmpstr);
/* print out the text held in regmatch[backref] */
for (j = regmatch[backref].rm_so; j < regmatch[backref].rm_eo; j++)
fputc(line[j], stdout);
}
/* if we find an unescaped '&' print out the whole matched text.
* fortunately, regmatch[0] contains the indicies to the whole matched
* expression (kinda seems like it was designed for just such a
* purpose...) */
else if (replace[i] == '&' && replace[i-1] != '\\') {
int j;
for (j = regmatch[0].rm_so; j < regmatch[0].rm_eo; j++)
fputc(line[j], stdout);
}
/* nothing special, just print this char of the replacement string to stdout */
else
fputc(replace[i], stdout);
}
}
static int do_subst_command(const struct sed_cmd *sed_cmd, const char *line) static int do_subst_command(const struct sed_cmd *sed_cmd, const char *line)
{ {
int altered = 0; int altered = 0;
/* we only substitute if the substitution 'search' expression matches */ /* we only substitute if the substitution 'search' expression matches */
if (regexec(sed_cmd->sub_match, line, 0, NULL, 0) == 0) { if (regexec(sed_cmd->sub_match, line, 0, NULL, 0) == 0) {
regmatch_t regmatch; regmatch_t *regmatch = xmalloc(sizeof(regmatch_t) * (sed_cmd->num_backrefs+1));
int i; int i;
char *ptr = (char *)line; char *ptr = (char *)line;
while (*ptr) { while (*ptr) {
/* if we can match the search string... */ /* if we can match the search string... */
if (regexec(sed_cmd->sub_match, ptr, 1, &regmatch, 0) == 0) { if (regexec(sed_cmd->sub_match, ptr, sed_cmd->num_backrefs+1, regmatch, 0) == 0) {
/* print everything before the match, */ /* print everything before the match, */
for (i = 0; i < regmatch.rm_so; i++) for (i = 0; i < regmatch[0].rm_so; i++)
fputc(ptr[i], stdout); fputc(ptr[i], stdout);
/* then print the substitution in its place */ /* then print the substitution in its place */
fputs(sed_cmd->replace, stdout); print_subst_w_backrefs(ptr, sed_cmd->replace, regmatch);
/* then advance past the match */ /* then advance past the match */
ptr += regmatch.rm_eo; ptr += regmatch[0].rm_eo;
/* and flag that something has changed */ /* and flag that something has changed */
altered++; altered++;
@ -496,6 +551,9 @@ static int do_subst_command(const struct sed_cmd *sed_cmd, const char *line)
/* is there anything left to print? */ /* is there anything left to print? */
if (*ptr) if (*ptr)
fputs(ptr, stdout); fputs(ptr, stdout);
/* cleanup */
free(regmatch);
} }
return altered; return altered;

94
sed.c
View File

@ -27,6 +27,7 @@
- address matching: num|/matchstr/[,num|/matchstr/|$]command - address matching: num|/matchstr/[,num|/matchstr/|$]command
- commands: (p)rint, (d)elete, (s)ubstitue (with g & I flags) - commands: (p)rint, (d)elete, (s)ubstitue (with g & I flags)
- edit commands: (a)ppend, (i)nsert, (c)hange - edit commands: (a)ppend, (i)nsert, (c)hange
- backreferences in substitution expressions (\1, \2...\9)
(Note: Specifying an address (range) to match is *optional*; commands (Note: Specifying an address (range) to match is *optional*; commands
default to the whole pattern space if no specific address match was default to the whole pattern space if no specific address match was
@ -73,6 +74,9 @@ struct sed_cmd {
/* substitution command specific fields */ /* substitution command specific fields */
regex_t *sub_match; /* sed -e 's/sub_match/replace/' */ regex_t *sub_match; /* sed -e 's/sub_match/replace/' */
char *replace; /* sed -e 's/sub_match/replace/' XXX: who will hold the \1 \2 \3s? */ char *replace; /* sed -e 's/sub_match/replace/' XXX: who will hold the \1 \2 \3s? */
unsigned int num_backrefs:4; /* how many back references (\1..\9) */
/* Note: GNU/POSIX sed does not save more than nine backrefs, so
* we only use 4 bits to hold the number */
unsigned int sub_g:1; /* sed -e 's/foo/bar/g' (global) */ unsigned int sub_g:1; /* sed -e 's/foo/bar/g' (global) */
/* edit command (a,i,c) speicific field */ /* edit command (a,i,c) speicific field */
@ -166,19 +170,19 @@ static size_t strrspn(const char *s, const char *accept)
#endif #endif
/* /*
* index_of_unescaped_slash - walks left to right through a string beginning * index_of_next_unescaped_slash - walks left to right through a string
* at a specified index and returns the index of the next unescaped slash. * beginning at a specified index and returns the index of the next forward
* slash ('/') not preceeded by a backslash ('\').
*/ */
static int index_of_next_unescaped_slash(const char *str, int idx) static int index_of_next_unescaped_slash(const char *str, int idx)
{ {
do { for ( ; str[idx]; idx++) {
idx++; if (str[idx] == '/' && str[idx-1] != '\\')
/* test if we've hit the end */ return idx;
if (str[idx] == 0) }
return -1;
} while (str[idx] != '/' && str[idx - 1] != '\\');
return idx; /* if we make it to here, we've hit the end of the string */
return -1;
} }
/* /*
@ -201,7 +205,7 @@ static int get_address(const char *str, int *line, regex_t **regex)
idx++; idx++;
} }
else if (my_str[idx] == '/') { else if (my_str[idx] == '/') {
idx = index_of_next_unescaped_slash(my_str, idx); idx = index_of_next_unescaped_slash(my_str, ++idx);
if (idx == -1) if (idx == -1)
fatalError("unterminated match expression\n"); fatalError("unterminated match expression\n");
my_str[idx] = '\0'; my_str[idx] = '\0';
@ -233,6 +237,7 @@ static int parse_subst_cmd(struct sed_cmd *sed_cmd, const char *substr)
int oldidx, cflags = REG_NEWLINE; int oldidx, cflags = REG_NEWLINE;
char *match; char *match;
int idx = 0; int idx = 0;
int j;
/* /*
* the string that gets passed to this function should look like this: * the string that gets passed to this function should look like this:
@ -249,14 +254,26 @@ static int parse_subst_cmd(struct sed_cmd *sed_cmd, const char *substr)
/* save the match string */ /* save the match string */
oldidx = idx+1; oldidx = idx+1;
idx = index_of_next_unescaped_slash(substr, idx); idx = index_of_next_unescaped_slash(substr, ++idx);
if (idx == -1) if (idx == -1)
fatalError("bad format in substitution expression\n"); fatalError("bad format in substitution expression\n");
match = strdup_substr(substr, oldidx, idx); match = strdup_substr(substr, oldidx, idx);
/* determine the number of back references in the match string */
/* Note: we compute this here rather than in the do_subst_command()
* function to save processor time, at the expense of a little more memory
* (4 bits) per sed_cmd */
/* sed_cmd->num_backrefs = 0; */ /* XXX: not needed? --apparently not */
for (j = 0; match[j]; j++) {
/* GNU/POSIX sed does not save more than nine backrefs */
if (match[j] == '\\' && match[j+1] == '(' && sed_cmd->num_backrefs < 9)
sed_cmd->num_backrefs++;
}
/* save the replacement string */ /* save the replacement string */
oldidx = idx+1; oldidx = idx+1;
idx = index_of_next_unescaped_slash(substr, idx); idx = index_of_next_unescaped_slash(substr, ++idx);
if (idx == -1) if (idx == -1)
fatalError("bad format in substitution expression\n"); fatalError("bad format in substitution expression\n");
sed_cmd->replace = strdup_substr(substr, oldidx, idx); sed_cmd->replace = strdup_substr(substr, oldidx, idx);
@ -280,7 +297,7 @@ static int parse_subst_cmd(struct sed_cmd *sed_cmd, const char *substr)
} }
out: out:
/* compile the regex */ /* compile the match string into a regex */
sed_cmd->sub_match = (regex_t *)xmalloc(sizeof(regex_t)); sed_cmd->sub_match = (regex_t *)xmalloc(sizeof(regex_t));
xregcomp(sed_cmd->sub_match, match, cflags); xregcomp(sed_cmd->sub_match, match, cflags);
free(match); free(match);
@ -460,26 +477,64 @@ static void load_cmd_file(char *filename)
} }
} }
static void print_subst_w_backrefs(const char *line, const char *replace, regmatch_t *regmatch)
{
int i;
/* go through the replacement string */
for (i = 0; replace[i]; i++) {
/* if we find a backreference (\1, \2, etc.) print the backref'ed * text */
if (replace[i] == '\\' && isdigit(replace[i+1])) {
int j;
char tmpstr[2];
int backref;
++i; /* i now indexes the backref number, instead of the leading slash */
tmpstr[0] = replace[i];
tmpstr[1] = 0;
backref = atoi(tmpstr);
/* print out the text held in regmatch[backref] */
for (j = regmatch[backref].rm_so; j < regmatch[backref].rm_eo; j++)
fputc(line[j], stdout);
}
/* if we find an unescaped '&' print out the whole matched text.
* fortunately, regmatch[0] contains the indicies to the whole matched
* expression (kinda seems like it was designed for just such a
* purpose...) */
else if (replace[i] == '&' && replace[i-1] != '\\') {
int j;
for (j = regmatch[0].rm_so; j < regmatch[0].rm_eo; j++)
fputc(line[j], stdout);
}
/* nothing special, just print this char of the replacement string to stdout */
else
fputc(replace[i], stdout);
}
}
static int do_subst_command(const struct sed_cmd *sed_cmd, const char *line) static int do_subst_command(const struct sed_cmd *sed_cmd, const char *line)
{ {
int altered = 0; int altered = 0;
/* we only substitute if the substitution 'search' expression matches */ /* we only substitute if the substitution 'search' expression matches */
if (regexec(sed_cmd->sub_match, line, 0, NULL, 0) == 0) { if (regexec(sed_cmd->sub_match, line, 0, NULL, 0) == 0) {
regmatch_t regmatch; regmatch_t *regmatch = xmalloc(sizeof(regmatch_t) * (sed_cmd->num_backrefs+1));
int i; int i;
char *ptr = (char *)line; char *ptr = (char *)line;
while (*ptr) { while (*ptr) {
/* if we can match the search string... */ /* if we can match the search string... */
if (regexec(sed_cmd->sub_match, ptr, 1, &regmatch, 0) == 0) { if (regexec(sed_cmd->sub_match, ptr, sed_cmd->num_backrefs+1, regmatch, 0) == 0) {
/* print everything before the match, */ /* print everything before the match, */
for (i = 0; i < regmatch.rm_so; i++) for (i = 0; i < regmatch[0].rm_so; i++)
fputc(ptr[i], stdout); fputc(ptr[i], stdout);
/* then print the substitution in its place */ /* then print the substitution in its place */
fputs(sed_cmd->replace, stdout); print_subst_w_backrefs(ptr, sed_cmd->replace, regmatch);
/* then advance past the match */ /* then advance past the match */
ptr += regmatch.rm_eo; ptr += regmatch[0].rm_eo;
/* and flag that something has changed */ /* and flag that something has changed */
altered++; altered++;
@ -496,6 +551,9 @@ static int do_subst_command(const struct sed_cmd *sed_cmd, const char *line)
/* is there anything left to print? */ /* is there anything left to print? */
if (*ptr) if (*ptr)
fputs(ptr, stdout); fputs(ptr, stdout);
/* cleanup */
free(regmatch);
} }
return altered; return altered;