mirror of https://github.com/rusefi/lua.git
'gmatch' keeps its state in a userdata (keeping the same 'MatchState'
across calls)
This commit is contained in:
parent
8264dbc2bb
commit
10fffcd80a
46
lstrlib.c
46
lstrlib.c
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
** $Id: lstrlib.c,v 1.232 2015/07/20 16:30:22 roberto Exp roberto $
|
** $Id: lstrlib.c,v 1.233 2015/09/26 18:45:03 roberto Exp roberto $
|
||||||
** Standard library for string operations and pattern-matching
|
** Standard library for string operations and pattern-matching
|
||||||
** See Copyright Notice in lua.h
|
** See Copyright Notice in lua.h
|
||||||
*/
|
*/
|
||||||
|
@ -677,24 +677,26 @@ static int str_match (lua_State *L) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* state for 'gmatch' */
|
||||||
|
typedef struct GMatchState {
|
||||||
|
const char *src; /* current position */
|
||||||
|
const char *p; /* pattern */
|
||||||
|
MatchState ms; /* match state */
|
||||||
|
} GMatchState;
|
||||||
|
|
||||||
|
|
||||||
static int gmatch_aux (lua_State *L) {
|
static int gmatch_aux (lua_State *L) {
|
||||||
MatchState ms;
|
GMatchState *gm = (GMatchState *)lua_touserdata(L, lua_upvalueindex(3));
|
||||||
size_t ls, lp;
|
|
||||||
const char *s = lua_tolstring(L, lua_upvalueindex(1), &ls);
|
|
||||||
const char *p = lua_tolstring(L, lua_upvalueindex(2), &lp);
|
|
||||||
const char *src;
|
const char *src;
|
||||||
prepstate(&ms, L, s, ls, p, lp);
|
for (src = gm->src; src <= gm->ms.src_end; src++) {
|
||||||
for (src = s + (size_t)lua_tointeger(L, lua_upvalueindex(3));
|
|
||||||
src <= ms.src_end;
|
|
||||||
src++) {
|
|
||||||
const char *e;
|
const char *e;
|
||||||
reprepstate(&ms);
|
reprepstate(&gm->ms);
|
||||||
if ((e = match(&ms, src, p)) != NULL) {
|
if ((e = match(&gm->ms, src, gm->p)) != NULL) {
|
||||||
lua_Integer newstart = e-s;
|
if (e == src) /* empty match? */
|
||||||
if (e == src) newstart++; /* empty match? go at least one position */
|
gm->src =src + 1; /* go at least one position */
|
||||||
lua_pushinteger(L, newstart);
|
else
|
||||||
lua_replace(L, lua_upvalueindex(3));
|
gm->src = e;
|
||||||
return push_captures(&ms, src, e);
|
return push_captures(&gm->ms, src, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return 0; /* not found */
|
return 0; /* not found */
|
||||||
|
@ -702,10 +704,14 @@ static int gmatch_aux (lua_State *L) {
|
||||||
|
|
||||||
|
|
||||||
static int gmatch (lua_State *L) {
|
static int gmatch (lua_State *L) {
|
||||||
luaL_checkstring(L, 1);
|
size_t ls, lp;
|
||||||
luaL_checkstring(L, 2);
|
const char *s = luaL_checklstring(L, 1, &ls);
|
||||||
lua_settop(L, 2);
|
const char *p = luaL_checklstring(L, 2, &lp);
|
||||||
lua_pushinteger(L, 0);
|
GMatchState *gm;
|
||||||
|
lua_settop(L, 2); /* keep them on closure to avoid being collected */
|
||||||
|
gm = (GMatchState *)lua_newuserdata(L, sizeof(GMatchState));
|
||||||
|
prepstate(&gm->ms, L, s, ls, p, lp);
|
||||||
|
gm->src = s; gm->p = p;
|
||||||
lua_pushcclosure(L, gmatch_aux, 3);
|
lua_pushcclosure(L, gmatch_aux, 3);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue