Skip to content

Commit 4d94cab

Browse files
committed
Fix allocation handling and remove Lua state reference from buffers.
Fix backported from Redis, thanks to Yoav Steinberg for the fix and Oran Agra for additional discussions about the fix.
1 parent dec1810 commit 4d94cab

File tree

1 file changed

+27
-30
lines changed

1 file changed

+27
-30
lines changed

lua_cmsgpack.c

+27-30
Original file line numberDiff line numberDiff line change
@@ -91,7 +91,6 @@ void memrevifle(void *ptr, size_t len) {
9191
* behavior. */
9292

9393
typedef struct mp_buf {
94-
lua_State *L;
9594
unsigned char *b;
9695
size_t len, free;
9796
} mp_buf;
@@ -111,27 +110,26 @@ mp_buf *mp_buf_new(lua_State *L) {
111110
/* Old size = 0; new size = sizeof(*buf) */
112111
buf = (mp_buf*)mp_realloc(L, NULL, 0, sizeof(*buf));
113112

114-
buf->L = L;
115113
buf->b = NULL;
116114
buf->len = buf->free = 0;
117115
return buf;
118116
}
119117

120-
void mp_buf_append(mp_buf *buf, const unsigned char *s, size_t len) {
118+
void mp_buf_append(lua_State *L, mp_buf *buf, const unsigned char *s, size_t len) {
121119
if (buf->free < len) {
122-
size_t newlen = buf->len+len;
120+
size_t newsize = (buf->len+len)*2;
123121

124-
buf->b = (unsigned char*)mp_realloc(buf->L, buf->b, buf->len, newlen*2);
125-
buf->free = newlen;
122+
buf->b = (unsigned char*)mp_realloc(L, buf->b, buf->len + buf->free, newsize);
123+
buf->free = newsize - buf->len;
126124
}
127125
memcpy(buf->b+buf->len,s,len);
128126
buf->len += len;
129127
buf->free -= len;
130128
}
131129

132-
void mp_buf_free(mp_buf *buf) {
133-
mp_realloc(buf->L, buf->b, buf->len, 0); /* realloc to 0 = free */
134-
mp_realloc(buf->L, buf, sizeof(*buf), 0);
130+
void mp_buf_free(lua_State *L, mp_buf *buf) {
131+
mp_realloc(L, buf->b, buf->len + buf->free, 0); /* realloc to 0 = free */
132+
mp_realloc(L, buf, sizeof(*buf), 0);
135133
}
136134

137135
/* ---------------------------- String cursor ----------------------------------
@@ -173,7 +171,7 @@ void mp_cur_init(mp_cur *cursor, const unsigned char *s, size_t len) {
173171

174172
/* ------------------------- Low level MP encoding -------------------------- */
175173

176-
void mp_encode_bytes(mp_buf *buf, const unsigned char *s, size_t len) {
174+
void mp_encode_bytes(lua_State *L, mp_buf *buf, const unsigned char *s, size_t len) {
177175
unsigned char hdr[5];
178176
int hdrlen;
179177

@@ -197,12 +195,12 @@ void mp_encode_bytes(mp_buf *buf, const unsigned char *s, size_t len) {
197195
hdr[4] = len&0xff;
198196
hdrlen = 5;
199197
}
200-
mp_buf_append(buf,hdr,hdrlen);
201-
mp_buf_append(buf,s,len);
198+
mp_buf_append(L,buf,hdr,hdrlen);
199+
mp_buf_append(L,buf,s,len);
202200
}
203201

204202
/* we assume IEEE 754 internal format for single and double precision floats. */
205-
void mp_encode_double(mp_buf *buf, double d) {
203+
void mp_encode_double(lua_State *L, mp_buf *buf, double d) {
206204
unsigned char b[9];
207205
float f = d;
208206

@@ -211,16 +209,16 @@ void mp_encode_double(mp_buf *buf, double d) {
211209
b[0] = 0xca; /* float IEEE 754 */
212210
memcpy(b+1,&f,4);
213211
memrevifle(b+1,4);
214-
mp_buf_append(buf,b,5);
212+
mp_buf_append(L,buf,b,5);
215213
} else if (sizeof(d) == 8) {
216214
b[0] = 0xcb; /* double IEEE 754 */
217215
memcpy(b+1,&d,8);
218216
memrevifle(b+1,8);
219-
mp_buf_append(buf,b,9);
217+
mp_buf_append(L,buf,b,9);
220218
}
221219
}
222220

223-
void mp_encode_int(mp_buf *buf, int64_t n) {
221+
void mp_encode_int(lua_State *L, mp_buf *buf, int64_t n) {
224222
unsigned char b[9];
225223
int enclen;
226224

@@ -289,10 +287,10 @@ void mp_encode_int(mp_buf *buf, int64_t n) {
289287
enclen = 9;
290288
}
291289
}
292-
mp_buf_append(buf,b,enclen);
290+
mp_buf_append(L,buf,b,enclen);
293291
}
294292

295-
void mp_encode_array(mp_buf *buf, int64_t n) {
293+
void mp_encode_array(lua_State *L, mp_buf *buf, int64_t n) {
296294
unsigned char b[5];
297295
int enclen;
298296

@@ -312,10 +310,10 @@ void mp_encode_array(mp_buf *buf, int64_t n) {
312310
b[4] = n & 0xff;
313311
enclen = 5;
314312
}
315-
mp_buf_append(buf,b,enclen);
313+
mp_buf_append(L,buf,b,enclen);
316314
}
317315

318-
void mp_encode_map(mp_buf *buf, int64_t n) {
316+
void mp_encode_map(lua_State *L, mp_buf *buf, int64_t n) {
319317
unsigned char b[5];
320318
int enclen;
321319

@@ -335,7 +333,7 @@ void mp_encode_map(mp_buf *buf, int64_t n) {
335333
b[4] = n & 0xff;
336334
enclen = 5;
337335
}
338-
mp_buf_append(buf,b,enclen);
336+
mp_buf_append(L,buf,b,enclen);
339337
}
340338

341339
/* --------------------------- Lua types encoding --------------------------- */
@@ -345,12 +343,12 @@ void mp_encode_lua_string(lua_State *L, mp_buf *buf) {
345343
const char *s;
346344

347345
s = lua_tolstring(L,-1,&len);
348-
mp_encode_bytes(buf,(const unsigned char*)s,len);
346+
mp_encode_bytes(L,buf,(const unsigned char*)s,len);
349347
}
350348

351349
void mp_encode_lua_bool(lua_State *L, mp_buf *buf) {
352350
unsigned char b = lua_toboolean(L,-1) ? 0xc3 : 0xc2;
353-
mp_buf_append(buf,&b,1);
351+
mp_buf_append(L,buf,&b,1);
354352
}
355353

356354
/* Lua 5.3 has a built in 64-bit integer type */
@@ -360,7 +358,7 @@ void mp_encode_lua_integer(lua_State *L, mp_buf *buf) {
360358
#else
361359
lua_Integer i = lua_tointeger(L,-1);
362360
#endif
363-
mp_encode_int(buf, (int64_t)i);
361+
mp_encode_int(L, buf, (int64_t)i);
364362
}
365363

366364
/* Lua 5.2 and lower only has 64-bit doubles, so we need to
@@ -372,7 +370,7 @@ void mp_encode_lua_number(lua_State *L, mp_buf *buf) {
372370
if (IS_INT64_EQUIVALENT(n)) {
373371
mp_encode_lua_integer(L, buf);
374372
} else {
375-
mp_encode_double(buf,(double)n);
373+
mp_encode_double(L,buf,(double)n);
376374
}
377375
}
378376

@@ -386,7 +384,7 @@ void mp_encode_lua_table_as_array(lua_State *L, mp_buf *buf, int level) {
386384
size_t len = lua_rawlen(L,-1), j;
387385
#endif
388386

389-
mp_encode_array(buf,len);
387+
mp_encode_array(L,buf,len);
390388
for (j = 1; j <= len; j++) {
391389
lua_pushnumber(L,j);
392390
lua_gettable(L,-2);
@@ -409,7 +407,7 @@ void mp_encode_lua_table_as_map(lua_State *L, mp_buf *buf, int level) {
409407
}
410408

411409
/* Step two: actually encoding of the map. */
412-
mp_encode_map(buf,len);
410+
mp_encode_map(L,buf,len);
413411
lua_pushnil(L);
414412
while(lua_next(L,-2)) {
415413
/* Stack: ... key value */
@@ -474,10 +472,9 @@ void mp_encode_lua_table(lua_State *L, mp_buf *buf, int level) {
474472

475473
void mp_encode_lua_null(lua_State *L, mp_buf *buf) {
476474
unsigned char b[1];
477-
(void)L;
478475

479476
b[0] = 0xc0;
480-
mp_buf_append(buf,b,1);
477+
mp_buf_append(L,buf,b,1);
481478
}
482479

483480
void mp_encode_lua_type(lua_State *L, mp_buf *buf, int level) {
@@ -534,7 +531,7 @@ int mp_pack(lua_State *L) {
534531
buf->free += buf->len;
535532
buf->len = 0;
536533
}
537-
mp_buf_free(buf);
534+
mp_buf_free(L, buf);
538535

539536
/* Concatenate all nargs buffers together */
540537
lua_concat(L, nargs);

0 commit comments

Comments
 (0)