Skip to content

Commit 4196464

Browse files
committedSep 8, 2015
long strings are created directly in final position when possible
(instead of using an auxiliar buffer to first create the string and then allocate the final string and copy result there)
1 parent 502214f commit 4196464

11 files changed

+62
-58
lines changed
 

‎ldo.c

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
** $Id: ldo.c,v 2.138 2015/05/22 17:48:19 roberto Exp roberto $
2+
** $Id: ldo.c,v 2.139 2015/06/18 14:19:52 roberto Exp roberto $
33
** Stack and Call structure of Lua
44
** See Copyright Notice in lua.h
55
*/
@@ -684,7 +684,7 @@ static void f_parser (lua_State *L, void *ud) {
684684
int c = zgetc(p->z); /* read first character */
685685
if (c == LUA_SIGNATURE[0]) {
686686
checkmode(L, p->mode, "binary");
687-
cl = luaU_undump(L, p->z, &p->buff, p->name);
687+
cl = luaU_undump(L, p->z, p->name);
688688
}
689689
else {
690690
checkmode(L, p->mode, "text");

‎lgc.c

+2-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
** $Id: lgc.c,v 2.205 2015/03/25 13:42:19 roberto Exp roberto $
2+
** $Id: lgc.c,v 2.206 2015/07/13 13:30:03 roberto Exp roberto $
33
** Garbage Collector
44
** See Copyright Notice in lua.h
55
*/
@@ -769,12 +769,11 @@ static GCObject **sweeptolive (lua_State *L, GCObject **p, int *n) {
769769
*/
770770

771771
/*
772-
** If possible, free concatenation buffer and shrink string table
772+
** If possible, shrink string table
773773
*/
774774
static void checkSizes (lua_State *L, global_State *g) {
775775
if (g->gckind != KGC_EMERGENCY) {
776776
l_mem olddebt = g->GCdebt;
777-
luaZ_freebuffer(L, &g->buff); /* free concatenation buffer */
778777
if (g->strt.nuse < g->strt.size / 4) /* string table too big? */
779778
luaS_resize(L, g->strt.size / 2); /* shrink it a little */
780779
g->GCestimate += g->GCdebt - olddebt; /* update estimate */

‎lstate.c

+1-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
** $Id: lstate.c,v 2.128 2015/03/04 13:31:21 roberto Exp roberto $
2+
** $Id: lstate.c,v 2.129 2015/07/13 13:30:03 roberto Exp roberto $
33
** Global State
44
** See Copyright Notice in lua.h
55
*/
@@ -241,7 +241,6 @@ static void close_state (lua_State *L) {
241241
if (g->version) /* closing a fully built state? */
242242
luai_userstateclose(L);
243243
luaM_freearray(L, G(L)->strt.hash, G(L)->strt.size);
244-
luaZ_freebuffer(L, &g->buff);
245244
freestack(L);
246245
lua_assert(gettotalbytes(g) == sizeof(LG));
247246
(*g->frealloc)(g->ud, fromstate(L), sizeof(LG), 0); /* free main block */
@@ -310,7 +309,6 @@ LUA_API lua_State *lua_newstate (lua_Alloc f, void *ud) {
310309
g->strt.size = g->strt.nuse = 0;
311310
g->strt.hash = NULL;
312311
setnilvalue(&g->l_registry);
313-
luaZ_initbuffer(L, &g->buff);
314312
g->panic = NULL;
315313
g->version = NULL;
316314
g->gcstate = GCSpause;

‎lstate.h

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
** $Id: lstate.h,v 2.122 2015/06/01 16:34:37 roberto Exp roberto $
2+
** $Id: lstate.h,v 2.123 2015/07/04 16:33:17 roberto Exp roberto $
33
** Global State
44
** See Copyright Notice in lua.h
55
*/
@@ -131,7 +131,6 @@ typedef struct global_State {
131131
GCObject *tobefnz; /* list of userdata to be GC */
132132
GCObject *fixedgc; /* list of objects not to be collected */
133133
struct lua_State *twups; /* list of threads with open upvalues */
134-
Mbuffer buff; /* temporary buffer for string concatenation */
135134
unsigned int gcfinnum; /* number of finalizers to call in each GC step */
136135
int gcpause; /* size of pause between successive GCs */
137136
int gcstepmul; /* GC 'granularity' */

‎lstring.c

+13-7
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
** $Id: lstring.c,v 2.49 2015/06/01 16:34:37 roberto Exp roberto $
2+
** $Id: lstring.c,v 2.50 2015/06/18 14:20:32 roberto Exp roberto $
33
** String table (keeps all strings handled by Lua)
44
** See Copyright Notice in lua.h
55
*/
@@ -119,8 +119,7 @@ void luaS_init (lua_State *L) {
119119
/*
120120
** creates a new string object
121121
*/
122-
static TString *createstrobj (lua_State *L, const char *str, size_t l,
123-
int tag, unsigned int h) {
122+
static TString *createstrobj (lua_State *L, size_t l, int tag, unsigned int h) {
124123
TString *ts;
125124
GCObject *o;
126125
size_t totalsize; /* total size of TString object */
@@ -129,12 +128,18 @@ static TString *createstrobj (lua_State *L, const char *str, size_t l,
129128
ts = gco2ts(o);
130129
ts->hash = h;
131130
ts->extra = 0;
132-
memcpy(getaddrstr(ts), str, l * sizeof(char));
133131
getaddrstr(ts)[l] = '\0'; /* ending 0 */
134132
return ts;
135133
}
136134

137135

136+
TString *luaS_createlngstrobj (lua_State *L, size_t l) {
137+
TString *ts = createstrobj(L, l, LUA_TLNGSTR, G(L)->seed);
138+
ts->u.lnglen = l;
139+
return ts;
140+
}
141+
142+
138143
void luaS_remove (lua_State *L, TString *ts) {
139144
stringtable *tb = &G(L)->strt;
140145
TString **p = &tb->hash[lmod(ts->hash, tb->size)];
@@ -166,7 +171,8 @@ static TString *internshrstr (lua_State *L, const char *str, size_t l) {
166171
luaS_resize(L, g->strt.size * 2);
167172
list = &g->strt.hash[lmod(h, g->strt.size)]; /* recompute with new size */
168173
}
169-
ts = createstrobj(L, str, l, LUA_TSHRSTR, h);
174+
ts = createstrobj(L, l, LUA_TSHRSTR, h);
175+
memcpy(getaddrstr(ts), str, l * sizeof(char));
170176
ts->shrlen = cast_byte(l);
171177
ts->u.hnext = *list;
172178
*list = ts;
@@ -185,8 +191,8 @@ TString *luaS_newlstr (lua_State *L, const char *str, size_t l) {
185191
TString *ts;
186192
if (l >= (MAX_SIZE - sizeof(TString))/sizeof(char))
187193
luaM_toobig(L);
188-
ts = createstrobj(L, str, l, LUA_TLNGSTR, G(L)->seed);
189-
ts->u.lnglen = l;
194+
ts = luaS_createlngstrobj(L, l);
195+
memcpy(getaddrstr(ts), str, l * sizeof(char));
190196
return ts;
191197
}
192198
}

‎lstring.h

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
** $Id: lstring.h,v 1.58 2015/03/04 13:31:21 roberto Exp roberto $
2+
** $Id: lstring.h,v 1.59 2015/03/25 13:42:19 roberto Exp roberto $
33
** String table (keep all strings handled by Lua)
44
** See Copyright Notice in lua.h
55
*/
@@ -42,6 +42,7 @@ LUAI_FUNC void luaS_remove (lua_State *L, TString *ts);
4242
LUAI_FUNC Udata *luaS_newudata (lua_State *L, size_t s);
4343
LUAI_FUNC TString *luaS_newlstr (lua_State *L, const char *str, size_t l);
4444
LUAI_FUNC TString *luaS_new (lua_State *L, const char *str);
45+
LUAI_FUNC TString *luaS_createlngstrobj (lua_State *L, size_t l);
4546

4647

4748
#endif

‎lundump.c

+11-9
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
** $Id: lundump.c,v 2.40 2014/06/19 18:27:20 roberto Exp roberto $
2+
** $Id: lundump.c,v 2.41 2014/11/02 19:19:04 roberto Exp roberto $
33
** load precompiled Lua chunks
44
** See Copyright Notice in lua.h
55
*/
@@ -32,7 +32,6 @@
3232
typedef struct {
3333
lua_State *L;
3434
ZIO *Z;
35-
Mbuffer *b;
3635
const char *name;
3736
} LoadState;
3837

@@ -92,10 +91,15 @@ static TString *LoadString (LoadState *S) {
9291
LoadVar(S, size);
9392
if (size == 0)
9493
return NULL;
95-
else {
96-
char *s = luaZ_openspace(S->L, S->b, --size);
97-
LoadVector(S, s, size);
98-
return luaS_newlstr(S->L, s, size);
94+
else if (--size <= LUAI_MAXSHORTLEN) { /* short string? */
95+
char buff[LUAI_MAXSHORTLEN];
96+
LoadVector(S, buff, size);
97+
return luaS_newlstr(S->L, buff, size);
98+
}
99+
else { /* long string */
100+
TString *ts = luaS_createlngstrobj(S->L, size);
101+
LoadVector(S, getaddrstr(ts), size); /* load directly in final place */
102+
return ts;
99103
}
100104
}
101105

@@ -251,8 +255,7 @@ static void checkHeader (LoadState *S) {
251255
/*
252256
** load precompiled chunk
253257
*/
254-
LClosure *luaU_undump(lua_State *L, ZIO *Z, Mbuffer *buff,
255-
const char *name) {
258+
LClosure *luaU_undump(lua_State *L, ZIO *Z, const char *name) {
256259
LoadState S;
257260
LClosure *cl;
258261
if (*name == '@' || *name == '=')
@@ -263,7 +266,6 @@ LClosure *luaU_undump(lua_State *L, ZIO *Z, Mbuffer *buff,
263266
S.name = name;
264267
S.L = L;
265268
S.Z = Z;
266-
S.b = buff;
267269
checkHeader(&S);
268270
cl = luaF_newLclosure(L, LoadByte(&S));
269271
setclLvalue(L, L->top, cl);

‎lundump.h

+2-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
** $Id: lundump.h,v 1.43 2014/04/15 14:28:20 roberto Exp roberto $
2+
** $Id: lundump.h,v 1.44 2014/06/19 18:27:20 roberto Exp roberto $
33
** load precompiled Lua chunks
44
** See Copyright Notice in lua.h
55
*/
@@ -23,8 +23,7 @@
2323
#define LUAC_FORMAT 0 /* this is the official format */
2424

2525
/* load one chunk; from lundump.c */
26-
LUAI_FUNC LClosure* luaU_undump (lua_State* L, ZIO* Z, Mbuffer* buff,
27-
const char* name);
26+
LUAI_FUNC LClosure* luaU_undump (lua_State* L, ZIO* Z, const char* name);
2827

2928
/* dump one chunk; from ldump.c */
3029
LUAI_FUNC int luaU_dump (lua_State* L, const Proto* f, lua_Writer w,

‎lvm.c

+26-15
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
** $Id: lvm.c,v 2.249 2015/08/03 19:50:49 roberto Exp roberto $
2+
** $Id: lvm.c,v 2.250 2015/08/03 20:40:26 roberto Exp roberto $
33
** Lua virtual machine
44
** See Copyright Notice in lua.h
55
*/
@@ -445,6 +445,17 @@ int luaV_equalobj (lua_State *L, const TValue *t1, const TValue *t2) {
445445

446446
#define isemptystr(o) (ttisshrstring(o) && tsvalue(o)->shrlen == 0)
447447

448+
/* copy strings in stack from top - n up to top - 1 to buffer */
449+
static void copy2buff (StkId top, int n, char *buff) {
450+
size_t tl = 0; /* size already copied */
451+
do {
452+
size_t l = vslen(top - n); /* length of string being copied */
453+
memcpy(buff + tl, svalue(top - n), l * sizeof(char));
454+
tl += l;
455+
} while (--n > 0);
456+
}
457+
458+
448459
/*
449460
** Main operation for concatenation: concat 'total' values in the stack,
450461
** from 'L->top - total' up to 'L->top - 1'.
@@ -464,24 +475,24 @@ void luaV_concat (lua_State *L, int total) {
464475
else {
465476
/* at least two non-empty string values; get as many as possible */
466477
size_t tl = vslen(top - 1);
467-
char *buffer;
468-
int i;
469-
/* collect total length */
470-
for (i = 1; i < total && tostring(L, top-i-1); i++) {
471-
size_t l = vslen(top - i - 1);
478+
TString *ts;
479+
/* collect total length and number of strings */
480+
for (n = 1; n < total && tostring(L, top - n - 1); n++) {
481+
size_t l = vslen(top - n - 1);
472482
if (l >= (MAX_SIZE/sizeof(char)) - tl)
473483
luaG_runerror(L, "string length overflow");
474484
tl += l;
475485
}
476-
buffer = luaZ_openspace(L, &G(L)->buff, tl);
477-
tl = 0;
478-
n = i;
479-
do { /* copy all strings to buffer */
480-
size_t l = vslen(top - i);
481-
memcpy(buffer+tl, svalue(top-i), l * sizeof(char));
482-
tl += l;
483-
} while (--i > 0);
484-
setsvalue2s(L, top-n, luaS_newlstr(L, buffer, tl)); /* create result */
486+
if (tl <= LUAI_MAXSHORTLEN) { /* is result a short string? */
487+
char buff[LUAI_MAXSHORTLEN];
488+
copy2buff(top, n, buff); /* copy strings to buffer */
489+
ts = luaS_newlstr(L, buff, tl);
490+
}
491+
else { /* long string; copy strings directly to final result */
492+
ts = luaS_createlngstrobj(L, tl);
493+
copy2buff(top, n, getaddrstr(ts));
494+
}
495+
setsvalue2s(L, top - n, ts); /* create result */
485496
}
486497
total -= n-1; /* got 'n' strings to create 1 new */
487498
L->top -= n-1; /* popped 'n' strings and pushed one */

‎lzio.c

+1-11
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
** $Id: lzio.c,v 1.35 2012/05/14 13:34:18 roberto Exp roberto $
2+
** $Id: lzio.c,v 1.36 2014/11/02 19:19:04 roberto Exp roberto $
33
** Buffered streams
44
** See Copyright Notice in lua.h
55
*/
@@ -66,13 +66,3 @@ size_t luaZ_read (ZIO *z, void *b, size_t n) {
6666
return 0;
6767
}
6868

69-
/* ------------------------------------------------------------------------ */
70-
char *luaZ_openspace (lua_State *L, Mbuffer *buff, size_t n) {
71-
if (n > buff->buffsize) {
72-
if (n < LUA_MINBUFFER) n = LUA_MINBUFFER;
73-
luaZ_resizebuffer(L, buff, n);
74-
}
75-
return buff->buffer;
76-
}
77-
78-

‎lzio.h

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
** $Id: lzio.h,v 1.29 2014/12/19 13:45:40 roberto Exp roberto $
2+
** $Id: lzio.h,v 1.30 2014/12/19 17:26:14 roberto Exp roberto $
33
** Buffered streams
44
** See Copyright Notice in lua.h
55
*/
@@ -44,7 +44,6 @@ typedef struct Mbuffer {
4444
#define luaZ_freebuffer(L, buff) luaZ_resizebuffer(L, buff, 0)
4545

4646

47-
LUAI_FUNC char *luaZ_openspace (lua_State *L, Mbuffer *buff, size_t n);
4847
LUAI_FUNC void luaZ_init (lua_State *L, ZIO *z, lua_Reader reader,
4948
void *data);
5049
LUAI_FUNC size_t luaZ_read (ZIO* z, void *b, size_t n); /* read next n bytes */

0 commit comments

Comments
 (0)
Please sign in to comment.