diff options
| author | Roberto Ierusalimschy <roberto@inf.puc-rio.br> | 2007-10-29 13:51:10 -0200 |
|---|---|---|
| committer | Roberto Ierusalimschy <roberto@inf.puc-rio.br> | 2007-10-29 13:51:10 -0200 |
| commit | c06cc609466d4527848bbf87268ec714942d554a (patch) | |
| tree | 34fa2d66dfaf077ab194a9b1c997a5deeb1910e9 /lstrlib.c | |
| parent | 2fa476655fb1a733bcf8c408f14103bd7abed1b8 (diff) | |
| download | lua-c06cc609466d4527848bbf87268ec714942d554a.tar.gz lua-c06cc609466d4527848bbf87268ec714942d554a.tar.bz2 lua-c06cc609466d4527848bbf87268ec714942d554a.zip | |
BUG: 'gsub' may go wild when wrongly called without its third
> argument and with a large subject.
Diffstat (limited to 'lstrlib.c')
| -rw-r--r-- | lstrlib.c | 21 |
1 files changed, 10 insertions, 11 deletions
| @@ -1,5 +1,5 @@ | |||
| 1 | /* | 1 | /* |
| 2 | ** $Id: lstrlib.c,v 1.136 2007/02/07 17:53:08 roberto Exp roberto $ | 2 | ** $Id: lstrlib.c,v 1.137 2007/10/25 19:30:36 roberto Exp roberto $ |
| 3 | ** Standard library for string operations and pattern-matching | 3 | ** Standard library for string operations and pattern-matching |
| 4 | ** See Copyright Notice in lua.h | 4 | ** See Copyright Notice in lua.h |
| 5 | */ | 5 | */ |
| @@ -611,14 +611,9 @@ static void add_s (MatchState *ms, luaL_Buffer *b, const char *s, | |||
| 611 | 611 | ||
| 612 | 612 | ||
| 613 | static void add_value (MatchState *ms, luaL_Buffer *b, const char *s, | 613 | static void add_value (MatchState *ms, luaL_Buffer *b, const char *s, |
| 614 | const char *e) { | 614 | const char *e, int tr) { |
| 615 | lua_State *L = ms->L; | 615 | lua_State *L = ms->L; |
| 616 | switch (lua_type(L, 3)) { | 616 | switch (tr) { |
| 617 | case LUA_TNUMBER: | ||
| 618 | case LUA_TSTRING: { | ||
| 619 | add_s(ms, b, s, e); | ||
| 620 | return; | ||
| 621 | } | ||
| 622 | case LUA_TFUNCTION: { | 617 | case LUA_TFUNCTION: { |
| 623 | int n; | 618 | int n; |
| 624 | lua_pushvalue(L, 3); | 619 | lua_pushvalue(L, 3); |
| @@ -631,8 +626,8 @@ static void add_value (MatchState *ms, luaL_Buffer *b, const char *s, | |||
| 631 | lua_gettable(L, 3); | 626 | lua_gettable(L, 3); |
| 632 | break; | 627 | break; |
| 633 | } | 628 | } |
| 634 | default: { | 629 | default: { /* LUA_TNUMBER or LUA_TSTRING */ |
| 635 | luaL_argerror(L, 3, "string/function/table expected"); | 630 | add_s(ms, b, s, e); |
| 636 | return; | 631 | return; |
| 637 | } | 632 | } |
| 638 | } | 633 | } |
| @@ -650,11 +645,15 @@ static int str_gsub (lua_State *L) { | |||
| 650 | size_t srcl; | 645 | size_t srcl; |
| 651 | const char *src = luaL_checklstring(L, 1, &srcl); | 646 | const char *src = luaL_checklstring(L, 1, &srcl); |
| 652 | const char *p = luaL_checkstring(L, 2); | 647 | const char *p = luaL_checkstring(L, 2); |
| 648 | int tr = lua_type(L, 3); | ||
| 653 | size_t max_s = luaL_optinteger(L, 4, srcl+1); | 649 | size_t max_s = luaL_optinteger(L, 4, srcl+1); |
| 654 | int anchor = (*p == '^') ? (p++, 1) : 0; | 650 | int anchor = (*p == '^') ? (p++, 1) : 0; |
| 655 | size_t n = 0; | 651 | size_t n = 0; |
| 656 | MatchState ms; | 652 | MatchState ms; |
| 657 | luaL_Buffer b; | 653 | luaL_Buffer b; |
| 654 | luaL_argcheck(L, tr == LUA_TNUMBER || tr == LUA_TSTRING || | ||
| 655 | tr == LUA_TFUNCTION || tr == LUA_TTABLE, 3, | ||
| 656 | "string/function/table expected"); | ||
| 658 | luaL_buffinit(L, &b); | 657 | luaL_buffinit(L, &b); |
| 659 | ms.L = L; | 658 | ms.L = L; |
| 660 | ms.src_init = src; | 659 | ms.src_init = src; |
| @@ -665,7 +664,7 @@ static int str_gsub (lua_State *L) { | |||
| 665 | e = match(&ms, src, p); | 664 | e = match(&ms, src, p); |
| 666 | if (e) { | 665 | if (e) { |
| 667 | n++; | 666 | n++; |
| 668 | add_value(&ms, &b, src, e); | 667 | add_value(&ms, &b, src, e, tr); |
| 669 | } | 668 | } |
| 670 | if (e && e>src) /* non empty match? */ | 669 | if (e && e>src) /* non empty match? */ |
| 671 | src = e; /* skip it */ | 670 | src = e; /* skip it */ |
