-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathmsgpack.lua
489 lines (458 loc) · 13.4 KB
/
msgpack.lua
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
--[=[
MessagePack v5 decoder and encoder
Written by Cosmin Apreutesei. Public Domain.
INSTANCING
msgpack([mp]) -> mp create a new msgpack instance
DECODING
mp:decode_next(p, [n], [i]) -> next_i, v decode value at offset i in p
mp:decode_each(p, [n], [i]) -> iter() -> next_i, v decode all values up to n bytes
ENCODING
mp:encoding_buffer([min_size]) -> b create a buffer for encoding
b:encode(v) -> b encode a value (see below)
b:encode_array(t, [n]) -> b encode an array
b:encode_map(t, [pairs]) -> b encode a map
b:encode_int(x) -> b encode a number as integer
b:encode_float(x) -> b encode a float
b:encode_double(x) -> b encode a double
b:encode_bin(v, [n]) -> b encode a byte array
b:encode_ext(type, [n]) -> b encode the header for an ext value
b:encode_ext_int(ctype, x) -> b encode a raw integer (see code)
b:encode_timestamp(ts) -> b encode a timestamp value
b:size() -> n get the buffer content size
b:get() -> p, n get the buffer and its size
b:tostring() -> s get the buffer as a string
b:reset() -> b reset the buffer for reuse
mp.array(...) -> t create an encodable array from args
mp.toarray(t, [n]) -> t add mp.N to table t
CUSTOMIZATION
mp.nil_key value to decode nil keys to (skip)
mp.nan_key value to decode NaN keys to (skip)
mp.nil_element value to decode nil array elements to (nil)
mp.decode_i64 int64_t decoder (tonumber)
mp.decode_u64 uint64_t decoder (tonumber)
mp.decoder[type] = f(mp, p, i, len) -> next_i, v add a decoder for an ext type
mp:decode_unknown(mp, p, i, len, type_code) end decode an unknown ext type
mp:isarray(t) decide if t is an array or map
mp.N key for array element count
mp.error(err) custom error constructor
DECODING BEHAVIOR
* nil and NaN table keys are skipped, unless mp.nil_key / mp.nan_key is set.
* nil array elements create Lua sparse arrays, unless mp.nil_element is set.
* extension types are decoded with mp.decoder[type], falling back to
mp:decode_unknown() (pre-defined as a stub that returns nil).
* decoding errors are raised with mp.error() which defaults to _G.error.
* there's no way to tell an empty array from an empty map.
ENCODING BEHAVIOR
* Lua numbers are packed as either integers (the smallest possible) or doubles.
* 64bit cdata numbers are packed as 64bit integers.
* Lua tables are encoded as arrays or maps based on mp:isarray() which
by default returns true only if there's a mp.N key present in the table.
Use mp.array() to make a Lua table that will be encoded as an array
or call b:encode_array() on any table.
* you can set [mp.N] = true in the array to mean that the element count is #t.
]=]
if not ... then require'msgpack_test'; return end
require'glue'
local
type, bor, band, shr, floor, noop, repl, update, dynarray =
type, bor, band, shr, floor, noop, repl, update, dynarray
local
isctype, copy, cast, ffi_string =
isctype, copy, cast, ffi.string
local
u32, i64, u64, u8a, i8p, u8p, i16p, u16p, i32p, u32p, i64p, u64p, f32p, f64p =
u32, i64, u64, u8a, i8p, u8p, i16p, u16p, i32p, u32p, i64p, u64p, f32p, f64p
local t_buf = u8a(8)
local mp = {decoder = {}}
function mp:decode_unknown() return nil end --stub
mp.decode_i64 = tonumber --stub
mp.decode_u64 = tonumber --stub
mp.error = error --stub
function msgpack(self)
return update(self or {}, mp)
end
local rcopy, rev2, rev4, rev8 = noop, noop, noop
if ffi.abi'le' then
function rcopy(dst, src, len)
for i = 0, len-1 do
dst[i] = src[len-1-i]
end
end
function rev2(p, i)
local a, b = p[i], p[i+1]
p[i], p[i+1] = b, a
end
function rev4(p, i)
local a, b, c, d = p[i], p[i+1], p[i+2], p[i+3]
p[i], p[i+1], p[i+2], p[i+3] = d, c, b, a
end
function rev8(p, i)
local a, b, c, d, e, f, g, h = p[i], p[i+1], p[i+2], p[i+3], p[i+4], p[i+5], p[i+6], p[i+7]
p[i], p[i+1], p[i+2], p[i+3], p[i+4], p[i+5], p[i+6], p[i+7] = h, g, f, e, d, c, b, a
end
end
--decoding -------------------------------------------------------------------
local function num(self, p, n, i, ct, len, tonumber)
if i + len > n then self.error'short read' end
rcopy(t_buf, p + i, len)
local v = cast(ct, t_buf)[0]
return i + len, tonumber and tonumber(v) or v
end
local function str(self, p, n, i, len)
if i + len > n then self.error'short read' end
return i + len, ffi_string(p + i, len)
end
local function ext(self, p, n, i, len)
local i, typ = num(self, p, n, i, i8p, 1)
if i + len > n then self.error'short read' end
local decode = self.decoder[typ] or self.decode_unknown
return i + len, decode(self, p, i, len, typ)
end
local obj --fw. decl.
local function arr(self, d, p, n, i, len)
local t = {}
for j = 1, len do
local v
i, v = obj(self, d+1, p, n, i)
if v == nil then v = self.nil_element end
t[j] = v
end
return i, t
end
local function map(self, d, p, n, i, len)
local t = {}
for j = 1, len do
local k, v
i, k = obj(self, d+1, p, n, i)
i, v = obj(self, d+1, p, n, i)
if k == nil then k = self.nil_key end
if k == 1/0 then k = self.nan_key end
if k ~= nil and k == k then
t[k] = v
end
end
return i, t
end
--[[local]] function obj(self, d, p, n, i)
if i >= n then self.error'short read' end
if d >= 100 then self.error'stack overflow' end
local c = p[i]
i = i + 1
if c < 0x80 then return i, c end
if c < 0x90 then return map(self, d, p, n, i, band(c, 0x0f)) end
if c < 0xa0 then return arr(self, d, p, n, i, band(c, 0x0f)) end
if c < 0xc0 then return str(self, p, n, i, band(c, 0x1f)) end
if c > 0xdf then return i, cast(i8p, p)[i-1] end
if c == 0xc0 then return i, nil end
if c == 0xc2 then return i, false end
if c == 0xc3 then return i, true end
if c == 0xc4 then return str(self, p, n, num(self, p, n, i, u8p , 1)) end
if c == 0xc5 then return str(self, p, n, num(self, p, n, i, u16p, 2)) end
if c == 0xc6 then return str(self, p, n, num(self, p, n, i, u32p, 4)) end
if c == 0xc7 then return ext(self, p, n, num(self, p, n, i, u8p , 1)) end
if c == 0xc8 then return ext(self, p, n, num(self, p, n, i, u16p, 2)) end
if c == 0xc9 then return ext(self, p, n, num(self, p, n, i, u32p, 4)) end
if c == 0xca then return num(self, p, n, i, f32p, 4) end
if c == 0xcb then return num(self, p, n, i, f64p, 8) end
if c == 0xcc then if i >= n then self.error'short read' end; return i+1, p[i] end
if c == 0xcd then return num(self, p, n, i, u16p, 2) end
if c == 0xce then return num(self, p, n, i, u32p, 4) end
if c == 0xcf then return num(self, p, n, i, u64p, 8, self.decode_u64) end
if c == 0xd0 then return num(self, p, n, i, i8p , 1) end
if c == 0xd1 then return num(self, p, n, i, i16p, 2) end
if c == 0xd2 then return num(self, p, n, i, i32p, 4) end
if c == 0xd3 then return num(self, p, n, i, i64p, 8, self.decode_i64) end
if c == 0xd4 then return ext(self, p, n, i, 1) end
if c == 0xd5 then return ext(self, p, n, i, 2) end
if c == 0xd6 then return ext(self, p, n, i, 4) end
if c == 0xd7 then return ext(self, p, n, i, 8) end
if c == 0xd8 then return ext(self, p, n, i, 16) end
if c == 0xd9 then return str(self, p, n, num(self, p, n, i, u8p , 1)) end
if c == 0xda then return str(self, p, n, num(self, p, n, i, u16p, 2)) end
if c == 0xdb then return str(self, p, n, num(self, p, n, i, u32p, 4)) end
if c == 0xdc then return arr(self, d, p, n, num(self, p, n, i, u16p, 2)) end
if c == 0xdd then return arr(self, d, p, n, num(self, p, n, i, u32p, 4)) end
if c == 0xde then return map(self, d, p, n, num(self, p, n, i, u16p, 2)) end
if c == 0xdf then return map(self, d, p, n, num(self, p, n, i, u32p, 4)) end
self.error'invalid message'
end
function mp:decode_next(p, n, i)
p = cast(u8p, p)
return obj(self, 0, p, n, i or 0)
end
function mp:decode_each(p, n, i)
i = i or 0
n = n or #p
p = cast(u8p, p)
return function()
if i >= n then return nil end
local v
i, v = obj(self, 0, p, n, i)
return i, v
end
end
--encoding -------------------------------------------------------------------
mp.N = {}
function mp:isarray(v)
return v[self.N] and true or false
end
function mp.array(...)
return {[mp.N] = select('#', ...), ...}
end
function mp.toarray(t, n)
t[mp.N] = n or t[mp.N] or #t
return t
end
function mp:encoding_buffer(min_size)
local mp = self
local buf = {}
local arr = dynarray(u8a, min_size)
local n = 0
local function b(len)
n = n + len
return arr(n), n-len
end
local function encode_len(n, u8mark, u16mark, u32mark)
if n <= 0xff and u8mark then
local p, i = b(2)
p[i] = u8mark
cast(u8p, p+i+1)[0] = n
elseif n <= 0xffff then
local p, i = b(3)
p[i] = u16mark
cast(u16p, p+i+1)[0] = n
rev2(p, i+1)
else
if n > 0xffffffff then self.error'too many elements' end
local p, i = b(5)
p[i] = u32mark
cast(u32p, p+i+1)[0] = n
rev4(p, i+1)
end
end
function buf:encode_array(t, n)
local n = n or repl(t[mp.N], true, #t) or #t
if n <= 0x0f then
local p, i = b(1)
p[i] = 0x90 + n
else
encode_len(n, nil, 0xdc, 0xdd)
end
for i = 1, n do
self:encode(t[i])
end
return self
end
function buf:encode_map(t, user_pairs)
local pairs = user_pairs or pairs
local n = 0
for k in pairs(t) do
n = n + 1
end
if n <= 0x0f then
local p, i = b(1)
p[i] = 0x80 + n
else
encode_len(n, nil, 0xde, 0xdf)
end
for k,v in pairs(t) do
self:encode(k)
self:encode(v)
end
return self
end
function buf:encode_int(v)
if type(v) == 'number' then
if v < 0 then
if v >= -0x20 then
local p, i = b(1)
p[i] = 0x100 + v
elseif v >= -0x80 then
local p, i = b(2)
p[i] = 0xd0
cast(i8p, p+i+1)[0] = v
elseif v >= -0x8000 then
local p, i = b(3)
p[i] = 0xd1
cast(i16p, p+i+1)[0] = v
rev2(p, i+1)
elseif v >= -0x80000000 then
local p, i = b(5)
p[i] = 0xd2
cast(i32p, p+i+1)[0] = v
rev4(p, i+1)
else
local p, i = b(9)
p[i] = 0xd3
cast(i64p, p+i+1)[0] = v
rev8(p, i+1)
end
elseif v <= 0x7f then
local p, i = b(1)
p[i] = v
elseif v <= 0xff then
local p, i = b(2)
p[i] = 0xcc
p[i+1] = v
elseif v <= 0xffff then
local p, i = b(3)
p[i] = 0xcd
cast(u16p, p+i+1)[0] = v
rev2(p, i+1)
elseif v <= 0xffffffff then
local p, i = b(5)
p[i] = 0xce
cast(u32p, p+i+1)[0] = v
rev4(p, i+1)
else
local p, i = b(9)
p[i] = 0xcf
cast(u64p, p+i+1)[0] = v
rev8(p, i+1)
end
elseif isctype(i64, v) then
local p, i = b(9)
p[i] = 0xd3
cast(i64p, p+i+1)[0] = v
rev8(p, i+1)
elseif isctype(u64, v) then
local p, i = b(9)
p[i] = 0xcf
cast(u64p, p+i+1)[0] = v
rev8(p, i+1)
else
error('number expected, got '..type(v))
end
return self
end
function buf:encode_float(v)
local p, i = b(5)
p[i] = 0xca
cast(f32p, p+i+1)[0] = v
rev4(p, i+1)
return self
end
function buf:encode_double(v)
local p, i = b(9)
p[i] = 0xcb
cast(f64p, p+i+1)[0] = v
rev8(p, i+1)
return self
end
function buf:encode_bin(v, n)
local n = n or #v
encode_len(n, 0xc4, 0xc5, 0xc6)
local p, i = b(n)
copy(p + i, v, n)
return self
end
function buf:encode_ext(typ, n)
local n = n or #v
if n <= 0 then self.error'zero bytes ext' end
if n == 1 then
local p, i = b(2)
p[i] = 0xd4
p[i+1] = typ
elseif n == 2 then
local p, i = b(2)
p[i] = 0xd5
p[i+1] = typ
elseif n == 4 then
local p, i = b(2)
p[i] = 0xd6
p[i+1] = typ
elseif n == 8 then
local p, i = b(2)
p[i] = 0xd7
p[i+1] = typ
elseif n == 16 then
local p, i = b(2)
p[i] = 0xd8
p[i+1] = typ
else
encode_len(0xc7, 0xc8, 0xc9)
local p, i = b(1)
p[i] = typ
end
return self
end
function buf:encode_ext_int(ct, x)
local n = sizeof(ct)
local p, i = b(n)
cast(ct, p+i)[0] = v
local rev = n == 1 and noop or n == 2 and rev2
or n == 4 and rev4 or n == 8 and rev8
rev(p, i)
return self
end
function buf:encode_timestamp(v)
local n
if v >= 0 and v <= 0xffffffff then
if floor(v) == v then
self:encode_ext(-1, 4)
self:encode_ext_int(u32, v)
else
self:encode_ext(-1, 8)
self:encode_ext_int(u32, (v - floor(v)) * 1e9)
self:encode_ext_int(u32, v)
end
else
self:encode_ext(-1, 4+8)
self:encode_ext_int(u32, (v - floor(v)) * 1e9)
self:encode_ext_int(i64, v)
end
return self
end
function buf:encode(v)
if v == nil then
local p, i = b(1)
p[i] = 0xc0
elseif v == false then
local p, i = b(1)
p[i] = 0xc2
elseif v == true then
local p, i = b(1)
p[i] = 0xc3
elseif type(v) == 'number' then
if floor(v) == v and not (v == 0 and 1 / v < 0) then
self:encode_int(v)
else
self:encode_double(v)
end
elseif type(v) == 'string' then
if #v <= 0x1f then
local p, i = b(1)
p[i] = 0xa0 + #v
else
encode_len(#v, 0xd9, 0xda, 0xdb)
end
local p, i = b(#v)
copy(p + i, v, #v)
elseif type(v) == 'table' then
if mp:isarray(v) then
self:encode_array(v)
else
self:encode_map(v)
end
elseif isctype(i64, v) or isctype(u64, v) then
self:encode_int(v)
else
error('invalid type '..type(v))
end
return self
end
function buf:size()
return n
end
function buf:get()
return arr(n)
end
function buf:tostring()
return ffi_string(arr(n))
end
function buf:reset()
n = 0
return self
end
return buf
end