1*572c4311Sfengbojiang /*
2*572c4311Sfengbojiang ** {======================================================
3*572c4311Sfengbojiang ** Library for packing/unpacking structures.
4*572c4311Sfengbojiang ** $Id: struct.c,v 1.7 2018/05/11 22:04:31 roberto Exp $
5*572c4311Sfengbojiang ** See Copyright Notice at the end of this file
6*572c4311Sfengbojiang ** =======================================================
7*572c4311Sfengbojiang */
8*572c4311Sfengbojiang /*
9*572c4311Sfengbojiang ** Valid formats:
10*572c4311Sfengbojiang ** > - big endian
11*572c4311Sfengbojiang ** < - little endian
12*572c4311Sfengbojiang ** ![num] - alignment
13*572c4311Sfengbojiang ** x - pading
14*572c4311Sfengbojiang ** b/B - signed/unsigned byte
15*572c4311Sfengbojiang ** h/H - signed/unsigned short
16*572c4311Sfengbojiang ** l/L - signed/unsigned long
17*572c4311Sfengbojiang ** T   - size_t
18*572c4311Sfengbojiang ** i/In - signed/unsigned integer with size 'n' (default is size of int)
19*572c4311Sfengbojiang ** cn - sequence of 'n' chars (from/to a string); when packing, n==0 means
20*572c4311Sfengbojiang         the whole string; when unpacking, n==0 means use the previous
21*572c4311Sfengbojiang         read number as the string length
22*572c4311Sfengbojiang ** s - zero-terminated string
23*572c4311Sfengbojiang ** f - float
24*572c4311Sfengbojiang ** d - double
25*572c4311Sfengbojiang ** ' ' - ignored
26*572c4311Sfengbojiang */
27*572c4311Sfengbojiang 
28*572c4311Sfengbojiang 
29*572c4311Sfengbojiang #include <assert.h>
30*572c4311Sfengbojiang #include <ctype.h>
31*572c4311Sfengbojiang #include <limits.h>
32*572c4311Sfengbojiang #include <stddef.h>
33*572c4311Sfengbojiang #include <string.h>
34*572c4311Sfengbojiang 
35*572c4311Sfengbojiang 
36*572c4311Sfengbojiang #include "lua.h"
37*572c4311Sfengbojiang #include "lauxlib.h"
38*572c4311Sfengbojiang 
39*572c4311Sfengbojiang 
40*572c4311Sfengbojiang #if (LUA_VERSION_NUM >= 502)
41*572c4311Sfengbojiang 
42*572c4311Sfengbojiang #define luaL_register(L,n,f)	luaL_newlib(L,f)
43*572c4311Sfengbojiang 
44*572c4311Sfengbojiang #endif
45*572c4311Sfengbojiang 
46*572c4311Sfengbojiang 
47*572c4311Sfengbojiang /* basic integer type */
48*572c4311Sfengbojiang #if !defined(STRUCT_INT)
49*572c4311Sfengbojiang #define STRUCT_INT	long
50*572c4311Sfengbojiang #endif
51*572c4311Sfengbojiang 
52*572c4311Sfengbojiang typedef STRUCT_INT Inttype;
53*572c4311Sfengbojiang 
54*572c4311Sfengbojiang /* corresponding unsigned version */
55*572c4311Sfengbojiang typedef unsigned STRUCT_INT Uinttype;
56*572c4311Sfengbojiang 
57*572c4311Sfengbojiang 
58*572c4311Sfengbojiang /* maximum size (in bytes) for integral types */
59*572c4311Sfengbojiang #define MAXINTSIZE	32
60*572c4311Sfengbojiang 
61*572c4311Sfengbojiang /* is 'x' a power of 2? */
62*572c4311Sfengbojiang #define isp2(x)		((x) > 0 && ((x) & ((x) - 1)) == 0)
63*572c4311Sfengbojiang 
64*572c4311Sfengbojiang /* dummy structure to get alignment requirements */
65*572c4311Sfengbojiang struct cD {
66*572c4311Sfengbojiang   char c;
67*572c4311Sfengbojiang   double d;
68*572c4311Sfengbojiang };
69*572c4311Sfengbojiang 
70*572c4311Sfengbojiang 
71*572c4311Sfengbojiang #define PADDING		(sizeof(struct cD) - sizeof(double))
72*572c4311Sfengbojiang #define MAXALIGN  	(PADDING > sizeof(int) ? PADDING : sizeof(int))
73*572c4311Sfengbojiang 
74*572c4311Sfengbojiang 
75*572c4311Sfengbojiang /* endian options */
76*572c4311Sfengbojiang #define BIG	0
77*572c4311Sfengbojiang #define LITTLE	1
78*572c4311Sfengbojiang 
79*572c4311Sfengbojiang 
80*572c4311Sfengbojiang static union {
81*572c4311Sfengbojiang   int dummy;
82*572c4311Sfengbojiang   char endian;
83*572c4311Sfengbojiang } const native = {1};
84*572c4311Sfengbojiang 
85*572c4311Sfengbojiang 
86*572c4311Sfengbojiang typedef struct Header {
87*572c4311Sfengbojiang   int endian;
88*572c4311Sfengbojiang   int align;
89*572c4311Sfengbojiang } Header;
90*572c4311Sfengbojiang 
91*572c4311Sfengbojiang 
getnum(const char ** fmt,int df)92*572c4311Sfengbojiang static int getnum (const char **fmt, int df) {
93*572c4311Sfengbojiang   if (!isdigit(**fmt))  /* no number? */
94*572c4311Sfengbojiang     return df;  /* return default value */
95*572c4311Sfengbojiang   else {
96*572c4311Sfengbojiang     int a = 0;
97*572c4311Sfengbojiang     do {
98*572c4311Sfengbojiang       a = a*10 + *((*fmt)++) - '0';
99*572c4311Sfengbojiang     } while (isdigit(**fmt));
100*572c4311Sfengbojiang     return a;
101*572c4311Sfengbojiang   }
102*572c4311Sfengbojiang }
103*572c4311Sfengbojiang 
104*572c4311Sfengbojiang 
105*572c4311Sfengbojiang #define defaultoptions(h)	((h)->endian = native.endian, (h)->align = 1)
106*572c4311Sfengbojiang 
107*572c4311Sfengbojiang 
108*572c4311Sfengbojiang 
optsize(lua_State * L,char opt,const char ** fmt)109*572c4311Sfengbojiang static size_t optsize (lua_State *L, char opt, const char **fmt) {
110*572c4311Sfengbojiang   switch (opt) {
111*572c4311Sfengbojiang     case 'B': case 'b': return sizeof(char);
112*572c4311Sfengbojiang     case 'H': case 'h': return sizeof(short);
113*572c4311Sfengbojiang     case 'L': case 'l': return sizeof(long);
114*572c4311Sfengbojiang     case 'T': return sizeof(size_t);
115*572c4311Sfengbojiang     case 'f':  return sizeof(float);
116*572c4311Sfengbojiang     case 'd':  return sizeof(double);
117*572c4311Sfengbojiang     case 'x': return 1;
118*572c4311Sfengbojiang     case 'c': return getnum(fmt, 1);
119*572c4311Sfengbojiang     case 'i': case 'I': {
120*572c4311Sfengbojiang       int sz = getnum(fmt, sizeof(int));
121*572c4311Sfengbojiang       if (sz > MAXINTSIZE)
122*572c4311Sfengbojiang         luaL_error(L, "integral size %d is larger than limit of %d",
123*572c4311Sfengbojiang                        sz, MAXINTSIZE);
124*572c4311Sfengbojiang       return sz;
125*572c4311Sfengbojiang     }
126*572c4311Sfengbojiang     default: return 0;  /* other cases do not need alignment */
127*572c4311Sfengbojiang   }
128*572c4311Sfengbojiang }
129*572c4311Sfengbojiang 
130*572c4311Sfengbojiang 
131*572c4311Sfengbojiang /*
132*572c4311Sfengbojiang ** return number of bytes needed to align an element of size 'size'
133*572c4311Sfengbojiang ** at current position 'len'
134*572c4311Sfengbojiang */
gettoalign(size_t len,Header * h,int opt,size_t size)135*572c4311Sfengbojiang static int gettoalign (size_t len, Header *h, int opt, size_t size) {
136*572c4311Sfengbojiang   if (size == 0 || opt == 'c') return 0;
137*572c4311Sfengbojiang   if (size > (size_t)h->align)
138*572c4311Sfengbojiang     size = h->align;  /* respect max. alignment */
139*572c4311Sfengbojiang   return (size - (len & (size - 1))) & (size - 1);
140*572c4311Sfengbojiang }
141*572c4311Sfengbojiang 
142*572c4311Sfengbojiang 
143*572c4311Sfengbojiang /*
144*572c4311Sfengbojiang ** options to control endianess and alignment
145*572c4311Sfengbojiang */
controloptions(lua_State * L,int opt,const char ** fmt,Header * h)146*572c4311Sfengbojiang static void controloptions (lua_State *L, int opt, const char **fmt,
147*572c4311Sfengbojiang                             Header *h) {
148*572c4311Sfengbojiang   switch (opt) {
149*572c4311Sfengbojiang     case  ' ': return;  /* ignore white spaces */
150*572c4311Sfengbojiang     case '>': h->endian = BIG; return;
151*572c4311Sfengbojiang     case '<': h->endian = LITTLE; return;
152*572c4311Sfengbojiang     case '!': {
153*572c4311Sfengbojiang       int a = getnum(fmt, MAXALIGN);
154*572c4311Sfengbojiang       if (!isp2(a))
155*572c4311Sfengbojiang         luaL_error(L, "alignment %d is not a power of 2", a);
156*572c4311Sfengbojiang       h->align = a;
157*572c4311Sfengbojiang       return;
158*572c4311Sfengbojiang     }
159*572c4311Sfengbojiang     default: {
160*572c4311Sfengbojiang       const char *msg = lua_pushfstring(L, "invalid format option '%c'", opt);
161*572c4311Sfengbojiang       luaL_argerror(L, 1, msg);
162*572c4311Sfengbojiang     }
163*572c4311Sfengbojiang   }
164*572c4311Sfengbojiang }
165*572c4311Sfengbojiang 
166*572c4311Sfengbojiang 
putinteger(lua_State * L,luaL_Buffer * b,int arg,int endian,int size)167*572c4311Sfengbojiang static void putinteger (lua_State *L, luaL_Buffer *b, int arg, int endian,
168*572c4311Sfengbojiang                         int size) {
169*572c4311Sfengbojiang   lua_Number n = luaL_checknumber(L, arg);
170*572c4311Sfengbojiang   Uinttype value;
171*572c4311Sfengbojiang   char buff[MAXINTSIZE];
172*572c4311Sfengbojiang   if (n < 0)
173*572c4311Sfengbojiang     value = (Uinttype)(Inttype)n;
174*572c4311Sfengbojiang   else
175*572c4311Sfengbojiang     value = (Uinttype)n;
176*572c4311Sfengbojiang   if (endian == LITTLE) {
177*572c4311Sfengbojiang     int i;
178*572c4311Sfengbojiang     for (i = 0; i < size; i++) {
179*572c4311Sfengbojiang       buff[i] = (value & 0xff);
180*572c4311Sfengbojiang       value >>= 8;
181*572c4311Sfengbojiang     }
182*572c4311Sfengbojiang   }
183*572c4311Sfengbojiang   else {
184*572c4311Sfengbojiang     int i;
185*572c4311Sfengbojiang     for (i = size - 1; i >= 0; i--) {
186*572c4311Sfengbojiang       buff[i] = (value & 0xff);
187*572c4311Sfengbojiang       value >>= 8;
188*572c4311Sfengbojiang     }
189*572c4311Sfengbojiang   }
190*572c4311Sfengbojiang   luaL_addlstring(b, buff, size);
191*572c4311Sfengbojiang }
192*572c4311Sfengbojiang 
193*572c4311Sfengbojiang 
correctbytes(char * b,int size,int endian)194*572c4311Sfengbojiang static void correctbytes (char *b, int size, int endian) {
195*572c4311Sfengbojiang   if (endian != native.endian) {
196*572c4311Sfengbojiang     int i = 0;
197*572c4311Sfengbojiang     while (i < --size) {
198*572c4311Sfengbojiang       char temp = b[i];
199*572c4311Sfengbojiang       b[i++] = b[size];
200*572c4311Sfengbojiang       b[size] = temp;
201*572c4311Sfengbojiang     }
202*572c4311Sfengbojiang   }
203*572c4311Sfengbojiang }
204*572c4311Sfengbojiang 
205*572c4311Sfengbojiang 
b_pack(lua_State * L)206*572c4311Sfengbojiang static int b_pack (lua_State *L) {
207*572c4311Sfengbojiang   luaL_Buffer b;
208*572c4311Sfengbojiang   const char *fmt = luaL_checkstring(L, 1);
209*572c4311Sfengbojiang   Header h;
210*572c4311Sfengbojiang   int arg = 2;
211*572c4311Sfengbojiang   size_t totalsize = 0;
212*572c4311Sfengbojiang   defaultoptions(&h);
213*572c4311Sfengbojiang   lua_pushnil(L);  /* mark to separate arguments from string buffer */
214*572c4311Sfengbojiang   luaL_buffinit(L, &b);
215*572c4311Sfengbojiang   while (*fmt != '\0') {
216*572c4311Sfengbojiang     int opt = *fmt++;
217*572c4311Sfengbojiang     size_t size = optsize(L, opt, &fmt);
218*572c4311Sfengbojiang     int toalign = gettoalign(totalsize, &h, opt, size);
219*572c4311Sfengbojiang     totalsize += toalign;
220*572c4311Sfengbojiang     while (toalign-- > 0) luaL_addchar(&b, '\0');
221*572c4311Sfengbojiang     switch (opt) {
222*572c4311Sfengbojiang       case 'b': case 'B': case 'h': case 'H':
223*572c4311Sfengbojiang       case 'l': case 'L': case 'T': case 'i': case 'I': {  /* integer types */
224*572c4311Sfengbojiang         putinteger(L, &b, arg++, h.endian, size);
225*572c4311Sfengbojiang         break;
226*572c4311Sfengbojiang       }
227*572c4311Sfengbojiang       case 'x': {
228*572c4311Sfengbojiang         luaL_addchar(&b, '\0');
229*572c4311Sfengbojiang         break;
230*572c4311Sfengbojiang       }
231*572c4311Sfengbojiang       case 'f': {
232*572c4311Sfengbojiang         float f = (float)luaL_checknumber(L, arg++);
233*572c4311Sfengbojiang         correctbytes((char *)&f, size, h.endian);
234*572c4311Sfengbojiang         luaL_addlstring(&b, (char *)&f, size);
235*572c4311Sfengbojiang         break;
236*572c4311Sfengbojiang       }
237*572c4311Sfengbojiang       case 'd': {
238*572c4311Sfengbojiang         double d = luaL_checknumber(L, arg++);
239*572c4311Sfengbojiang         correctbytes((char *)&d, size, h.endian);
240*572c4311Sfengbojiang         luaL_addlstring(&b, (char *)&d, size);
241*572c4311Sfengbojiang         break;
242*572c4311Sfengbojiang       }
243*572c4311Sfengbojiang       case 'c': case 's': {
244*572c4311Sfengbojiang         size_t l;
245*572c4311Sfengbojiang         const char *s = luaL_checklstring(L, arg++, &l);
246*572c4311Sfengbojiang         if (size == 0) size = l;
247*572c4311Sfengbojiang         luaL_argcheck(L, l >= (size_t)size, arg, "string too short");
248*572c4311Sfengbojiang         luaL_addlstring(&b, s, size);
249*572c4311Sfengbojiang         if (opt == 's') {
250*572c4311Sfengbojiang           luaL_addchar(&b, '\0');  /* add zero at the end */
251*572c4311Sfengbojiang           size++;
252*572c4311Sfengbojiang         }
253*572c4311Sfengbojiang         break;
254*572c4311Sfengbojiang       }
255*572c4311Sfengbojiang       default: controloptions(L, opt, &fmt, &h);
256*572c4311Sfengbojiang     }
257*572c4311Sfengbojiang     totalsize += size;
258*572c4311Sfengbojiang   }
259*572c4311Sfengbojiang   luaL_pushresult(&b);
260*572c4311Sfengbojiang   return 1;
261*572c4311Sfengbojiang }
262*572c4311Sfengbojiang 
263*572c4311Sfengbojiang 
getinteger(const char * buff,int endian,int issigned,int size)264*572c4311Sfengbojiang static lua_Number getinteger (const char *buff, int endian,
265*572c4311Sfengbojiang                         int issigned, int size) {
266*572c4311Sfengbojiang   Uinttype l = 0;
267*572c4311Sfengbojiang   int i;
268*572c4311Sfengbojiang   if (endian == BIG) {
269*572c4311Sfengbojiang     for (i = 0; i < size; i++) {
270*572c4311Sfengbojiang       l <<= 8;
271*572c4311Sfengbojiang       l |= (Uinttype)(unsigned char)buff[i];
272*572c4311Sfengbojiang     }
273*572c4311Sfengbojiang   }
274*572c4311Sfengbojiang   else {
275*572c4311Sfengbojiang     for (i = size - 1; i >= 0; i--) {
276*572c4311Sfengbojiang       l <<= 8;
277*572c4311Sfengbojiang       l |= (Uinttype)(unsigned char)buff[i];
278*572c4311Sfengbojiang     }
279*572c4311Sfengbojiang   }
280*572c4311Sfengbojiang   if (!issigned)
281*572c4311Sfengbojiang     return (lua_Number)l;
282*572c4311Sfengbojiang   else {  /* signed format */
283*572c4311Sfengbojiang     Uinttype mask = (Uinttype)(~((Uinttype)0)) << (size*8 - 1);
284*572c4311Sfengbojiang     if (l & mask)  /* negative value? */
285*572c4311Sfengbojiang       l |= mask;  /* signal extension */
286*572c4311Sfengbojiang     return (lua_Number)(Inttype)l;
287*572c4311Sfengbojiang   }
288*572c4311Sfengbojiang }
289*572c4311Sfengbojiang 
290*572c4311Sfengbojiang 
b_unpack(lua_State * L)291*572c4311Sfengbojiang static int b_unpack (lua_State *L) {
292*572c4311Sfengbojiang   Header h;
293*572c4311Sfengbojiang   const char *fmt = luaL_checkstring(L, 1);
294*572c4311Sfengbojiang   size_t ld;
295*572c4311Sfengbojiang   const char *data = luaL_checklstring(L, 2, &ld);
296*572c4311Sfengbojiang   size_t pos = luaL_optinteger(L, 3, 1);
297*572c4311Sfengbojiang   luaL_argcheck(L, pos > 0, 3, "offset must be 1 or greater");
298*572c4311Sfengbojiang   pos--; /* Lua indexes are 1-based, but here we want 0-based for C
299*572c4311Sfengbojiang           * pointer math. */
300*572c4311Sfengbojiang   int n = 0;  /* number of results */
301*572c4311Sfengbojiang   defaultoptions(&h);
302*572c4311Sfengbojiang   while (*fmt) {
303*572c4311Sfengbojiang     int opt = *fmt++;
304*572c4311Sfengbojiang     size_t size = optsize(L, opt, &fmt);
305*572c4311Sfengbojiang     pos += gettoalign(pos, &h, opt, size);
306*572c4311Sfengbojiang     luaL_argcheck(L, size <= ld && pos <= ld - size,
307*572c4311Sfengbojiang                    2, "data string too short");
308*572c4311Sfengbojiang     /* stack space for item + next position */
309*572c4311Sfengbojiang     luaL_checkstack(L, 2, "too many results");
310*572c4311Sfengbojiang     switch (opt) {
311*572c4311Sfengbojiang       case 'b': case 'B': case 'h': case 'H':
312*572c4311Sfengbojiang       case 'l': case 'L': case 'T': case 'i':  case 'I': {  /* integer types */
313*572c4311Sfengbojiang         int issigned = islower(opt);
314*572c4311Sfengbojiang         lua_Number res = getinteger(data+pos, h.endian, issigned, size);
315*572c4311Sfengbojiang         lua_pushnumber(L, res); n++;
316*572c4311Sfengbojiang         break;
317*572c4311Sfengbojiang       }
318*572c4311Sfengbojiang       case 'x': {
319*572c4311Sfengbojiang         break;
320*572c4311Sfengbojiang       }
321*572c4311Sfengbojiang       case 'f': {
322*572c4311Sfengbojiang         float f;
323*572c4311Sfengbojiang         memcpy(&f, data+pos, size);
324*572c4311Sfengbojiang         correctbytes((char *)&f, sizeof(f), h.endian);
325*572c4311Sfengbojiang         lua_pushnumber(L, f); n++;
326*572c4311Sfengbojiang         break;
327*572c4311Sfengbojiang       }
328*572c4311Sfengbojiang       case 'd': {
329*572c4311Sfengbojiang         double d;
330*572c4311Sfengbojiang         memcpy(&d, data+pos, size);
331*572c4311Sfengbojiang         correctbytes((char *)&d, sizeof(d), h.endian);
332*572c4311Sfengbojiang         lua_pushnumber(L, d); n++;
333*572c4311Sfengbojiang         break;
334*572c4311Sfengbojiang       }
335*572c4311Sfengbojiang       case 'c': {
336*572c4311Sfengbojiang         if (size == 0) {
337*572c4311Sfengbojiang           if (n == 0 || !lua_isnumber(L, -1))
338*572c4311Sfengbojiang             luaL_error(L, "format 'c0' needs a previous size");
339*572c4311Sfengbojiang           size = lua_tonumber(L, -1);
340*572c4311Sfengbojiang           lua_pop(L, 1); n--;
341*572c4311Sfengbojiang           luaL_argcheck(L, size <= ld && pos <= ld - size,
342*572c4311Sfengbojiang                            2, "data string too short");
343*572c4311Sfengbojiang         }
344*572c4311Sfengbojiang         lua_pushlstring(L, data+pos, size); n++;
345*572c4311Sfengbojiang         break;
346*572c4311Sfengbojiang       }
347*572c4311Sfengbojiang       case 's': {
348*572c4311Sfengbojiang         const char *e = (const char *)memchr(data+pos, '\0', ld - pos);
349*572c4311Sfengbojiang         if (e == NULL)
350*572c4311Sfengbojiang           luaL_error(L, "unfinished string in data");
351*572c4311Sfengbojiang         size = (e - (data+pos)) + 1;
352*572c4311Sfengbojiang         lua_pushlstring(L, data+pos, size - 1); n++;
353*572c4311Sfengbojiang         break;
354*572c4311Sfengbojiang       }
355*572c4311Sfengbojiang       default: controloptions(L, opt, &fmt, &h);
356*572c4311Sfengbojiang     }
357*572c4311Sfengbojiang     pos += size;
358*572c4311Sfengbojiang   }
359*572c4311Sfengbojiang   lua_pushinteger(L, pos + 1);  /* next position */
360*572c4311Sfengbojiang   return n + 1;
361*572c4311Sfengbojiang }
362*572c4311Sfengbojiang 
363*572c4311Sfengbojiang 
b_size(lua_State * L)364*572c4311Sfengbojiang static int b_size (lua_State *L) {
365*572c4311Sfengbojiang   Header h;
366*572c4311Sfengbojiang   const char *fmt = luaL_checkstring(L, 1);
367*572c4311Sfengbojiang   size_t pos = 0;
368*572c4311Sfengbojiang   defaultoptions(&h);
369*572c4311Sfengbojiang   while (*fmt) {
370*572c4311Sfengbojiang     int opt = *fmt++;
371*572c4311Sfengbojiang     size_t size = optsize(L, opt, &fmt);
372*572c4311Sfengbojiang     pos += gettoalign(pos, &h, opt, size);
373*572c4311Sfengbojiang     if (opt == 's')
374*572c4311Sfengbojiang       luaL_argerror(L, 1, "option 's' has no fixed size");
375*572c4311Sfengbojiang     else if (opt == 'c' && size == 0)
376*572c4311Sfengbojiang       luaL_argerror(L, 1, "option 'c0' has no fixed size");
377*572c4311Sfengbojiang     if (!isalnum(opt))
378*572c4311Sfengbojiang       controloptions(L, opt, &fmt, &h);
379*572c4311Sfengbojiang     pos += size;
380*572c4311Sfengbojiang   }
381*572c4311Sfengbojiang   lua_pushinteger(L, pos);
382*572c4311Sfengbojiang   return 1;
383*572c4311Sfengbojiang }
384*572c4311Sfengbojiang 
385*572c4311Sfengbojiang /* }====================================================== */
386*572c4311Sfengbojiang 
387*572c4311Sfengbojiang 
388*572c4311Sfengbojiang 
389*572c4311Sfengbojiang static const struct luaL_Reg thislib[] = {
390*572c4311Sfengbojiang   {"pack", b_pack},
391*572c4311Sfengbojiang   {"unpack", b_unpack},
392*572c4311Sfengbojiang   {"size", b_size},
393*572c4311Sfengbojiang   {NULL, NULL}
394*572c4311Sfengbojiang };
395*572c4311Sfengbojiang 
396*572c4311Sfengbojiang 
397*572c4311Sfengbojiang LUALIB_API int luaopen_struct (lua_State *L);
398*572c4311Sfengbojiang 
luaopen_struct(lua_State * L)399*572c4311Sfengbojiang LUALIB_API int luaopen_struct (lua_State *L) {
400*572c4311Sfengbojiang   luaL_register(L, "struct", thislib);
401*572c4311Sfengbojiang   return 1;
402*572c4311Sfengbojiang }
403*572c4311Sfengbojiang 
404*572c4311Sfengbojiang 
405*572c4311Sfengbojiang /******************************************************************************
406*572c4311Sfengbojiang * Copyright (C) 2010-2018 Lua.org, PUC-Rio.  All rights reserved.
407*572c4311Sfengbojiang *
408*572c4311Sfengbojiang * Permission is hereby granted, free of charge, to any person obtaining
409*572c4311Sfengbojiang * a copy of this software and associated documentation files (the
410*572c4311Sfengbojiang * "Software"), to deal in the Software without restriction, including
411*572c4311Sfengbojiang * without limitation the rights to use, copy, modify, merge, publish,
412*572c4311Sfengbojiang * distribute, sublicense, and/or sell copies of the Software, and to
413*572c4311Sfengbojiang * permit persons to whom the Software is furnished to do so, subject to
414*572c4311Sfengbojiang * the following conditions:
415*572c4311Sfengbojiang *
416*572c4311Sfengbojiang * The above copyright notice and this permission notice shall be
417*572c4311Sfengbojiang * included in all copies or substantial portions of the Software.
418*572c4311Sfengbojiang *
419*572c4311Sfengbojiang * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
420*572c4311Sfengbojiang * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
421*572c4311Sfengbojiang * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
422*572c4311Sfengbojiang * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
423*572c4311Sfengbojiang * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
424*572c4311Sfengbojiang * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
425*572c4311Sfengbojiang * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
426*572c4311Sfengbojiang ******************************************************************************/
427*572c4311Sfengbojiang 
428