--resolve FS-5566
When you use $12345 in regex substitutions, it isn't obvious whenever you mean $1-concatenated-2345 or $12-concatenated-345 or any other variation. In all other languages, in order to solve this ambiguity, a braces {} are allowed to be used to separate variable name (or a reference) from surrounding text, like ${1}2345 or ${12}345. Use the same for freeswitch too. While at it, fix a buffer overflow as well: the index[] variable which is used to copy the "variable" name is 10 chars long, but it is used in the code without bounds checking, so a reference which is >9 chars long ($1234567890) will overflow the buffer, crashing freeswitch. And another overflow is in the way how size of the "substituted" variable is handled. First, in the outer loop, we compare the wrong variable with the size of `substituted' buffer (amount of bytes we took from the source instead of amount of bytes we used in `substituted'). And second, when actual regex match is being substitured, amount of room in `substituted' variable is not checked at all. Patch contributed by Michael Tokarev <mjt@tls.msk.ru>
This commit is contained in:
parent
8bcff4ca4d
commit
c2c8fba14a
|
@ -132,20 +132,37 @@ SWITCH_DECLARE(void) switch_perform_substitution(switch_regex_t *re, int match_c
|
||||||
const char *replace = NULL;
|
const char *replace = NULL;
|
||||||
switch_size_t x, y = 0, z = 0;
|
switch_size_t x, y = 0, z = 0;
|
||||||
int num = 0;
|
int num = 0;
|
||||||
|
int brace;
|
||||||
|
|
||||||
for (x = 0; x < (len - 1) && x < strlen(data);) {
|
for (x = 0; y < (len - 1) && x < strlen(data);) {
|
||||||
if (data[x] == '$') {
|
if (data[x] == '$') {
|
||||||
x++;
|
x++;
|
||||||
|
|
||||||
|
brace = data[x] == '{';
|
||||||
|
if (brace) {
|
||||||
|
x++;
|
||||||
|
}
|
||||||
|
|
||||||
if (!(data[x] > 47 && data[x] < 58)) {
|
if (!(data[x] > 47 && data[x] < 58)) {
|
||||||
|
x -= brace;
|
||||||
substituted[y++] = data[x - 1];
|
substituted[y++] = data[x - 1];
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
while (data[x] > 47 && data[x] < 58) {
|
while (data[x] > 47 && data[x] < 58 && z < sizeof(index) - 1) {
|
||||||
index[z++] = data[x];
|
index[z++] = data[x];
|
||||||
x++;
|
x++;
|
||||||
}
|
}
|
||||||
|
if (brace) {
|
||||||
|
if (data[x] != '}') {
|
||||||
|
x -= z - 1;
|
||||||
|
substituted[y++] = data[x - 1];
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
x++;
|
||||||
|
}
|
||||||
|
}
|
||||||
index[z++] = '\0';
|
index[z++] = '\0';
|
||||||
z = 0;
|
z = 0;
|
||||||
num = atoi(index);
|
num = atoi(index);
|
||||||
|
@ -156,7 +173,7 @@ SWITCH_DECLARE(void) switch_perform_substitution(switch_regex_t *re, int match_c
|
||||||
|
|
||||||
if (pcre_get_substring(field_data, ovector, match_count, num, &replace) > 0) {
|
if (pcre_get_substring(field_data, ovector, match_count, num, &replace) > 0) {
|
||||||
switch_size_t r;
|
switch_size_t r;
|
||||||
for (r = 0; r < strlen(replace); r++) {
|
for (r = 0; r < strlen(replace) && y < (len - 1); r++) {
|
||||||
substituted[y++] = replace[r];
|
substituted[y++] = replace[r];
|
||||||
}
|
}
|
||||||
pcre_free_substring(replace);
|
pcre_free_substring(replace);
|
||||||
|
|
Loading…
Reference in New Issue