]>
cvs.zerfleddert.de Git - proxmark3-svn/blob - client/pm3_binlib.c
e3af7b47e66ee05c55e47a9d55229f0067d0c1e5
3 * a Lua library for packing and unpacking binary data
4 * Luiz Henrique de Figueiredo <lhf@tecgraf.puc-rio.br>
6 * This code is hereby placed in the public domain.
7 * with contributions from Ignacio Castao <castanyo@yahoo.es> and
8 * Roberto Ierusalimschy <roberto@inf.puc-rio.br>.
11 #define OP_ZSTRING 'z' /* zero-terminated string */
12 #define OP_BSTRING 'p' /* string preceded by length byte */
13 #define OP_WSTRING 'P' /* string preceded by length word */
14 #define OP_SSTRING 'a' /* string preceded by length size_t */
15 #define OP_STRING 'A' /* string */
16 #define OP_FLOAT 'f' /* float */
17 #define OP_DOUBLE 'd' /* double */
18 #define OP_NUMBER 'n' /* Lua number */
19 #define OP_CHAR 'c' /* char (1-byte int) */
20 #define OP_BYTE 'C' /* byte = unsigned char (1-byte unsigned int) */
21 #define OP_SHORT 's' /* short (2-byte int) */
22 #define OP_USHORT 'S' /* unsigned short (2-byte unsigned int) */
23 #define OP_INT 'i' /* int (4-byte int) */
24 #define OP_UINT 'I' /* unsigned int (4-byte unsigned int) */
25 #define OP_LONG 'l' /* long (8-byte int) */
26 #define OP_ULONG 'L' /* unsigned long (8-byte unsigned int) */
27 #define OP_LITTLEENDIAN '<' /* little endian */
28 #define OP_BIGENDIAN '>' /* big endian */
29 #define OP_NATIVE '=' /* native endian */
39 #include "pm3_binlib.h"
42 static void badcode(lua_State
*L
, int c
)
44 char s
[]="bad code `?'";
49 static int doendian(int c
)
53 if (c
==OP_LITTLEENDIAN
) return !e
;
54 if (c
==OP_BIGENDIAN
) return e
;
55 if (c
==OP_NATIVE
) return 0;
59 static void doswap(int swap
, void *p
, size_t n
)
65 for (i
=0, j
=n
-1, n
=n
/2; n
--; i
++, j
--)
67 char t
=a
[i
]; a
[i
]=a
[j
]; a
[j
]=t
;
72 #define UNPACKNUMBER(OP,T) \
77 if (i+m>len) { done = 1; break;} \
81 lua_pushnumber(L,(lua_Number)a); \
86 #define UNPACKSTRING(OP,T) \
91 if (i+m>len) { done = 1; break; } \
94 if (i+m+l>len) { done = 1; break;} \
96 lua_pushlstring(L,s+i,l); \
102 #define HEXDIGITS(DIG) \
103 "0123456789ABCDEF"[DIG]
105 static int l_unpack(lua_State
*L
) /** unpack(f,s, [init]) */
108 const char *s
=luaL_checklstring(L
,2,&len
); /* switched s and f */
109 const char *f
=luaL_checkstring(L
,1);
110 int i_read
= luaL_optint(L
,3,1)-1;
121 while (*f
&& done
== 0)
125 if (isdigit((int) (unsigned char) *f
))
128 while (isdigit((int) (unsigned char) *f
)) N
=10*N
+(*f
++)-'0';
129 if (N
==0 && c
==OP_STRING
) { lua_pushliteral(L
,""); ++n
; }
131 while (N
-- && done
== 0) switch (c
)
133 case OP_LITTLEENDIAN
:
144 if (i
+N
>len
) {done
= 1; break; }
145 lua_pushlstring(L
,s
+i
,N
);
154 if (i
>=len
) {done
= 1; break; }
156 lua_pushlstring(L
,s
+i
,l
);
161 UNPACKSTRING(OP_BSTRING
, unsigned char)
162 UNPACKSTRING(OP_WSTRING
, unsigned short)
163 UNPACKSTRING(OP_SSTRING
, size_t)
164 UNPACKNUMBER(OP_NUMBER
, lua_Number
)
165 UNPACKNUMBER(OP_DOUBLE
, double)
166 UNPACKNUMBER(OP_FLOAT
, float)
167 UNPACKNUMBER(OP_CHAR
, char)
168 UNPACKNUMBER(OP_BYTE
, unsigned char)
169 UNPACKNUMBER(OP_SHORT
, short)
170 UNPACKNUMBER(OP_USHORT
, unsigned short)
171 UNPACKNUMBER(OP_INT
, int)
172 UNPACKNUMBER(OP_UINT
, unsigned int)
173 UNPACKNUMBER(OP_LONG
, long)
174 UNPACKNUMBER(OP_ULONG
, unsigned long)
180 luaL_buffinit(L
,&buf
);
182 if (i
+N
> len
) {done
= 1; break;}
183 for (unsigned int ii
= i
; ii
< i
+N
; ii
++) {
186 hdigit
= HEXDIGITS(val
);
187 luaL_addlstring(&buf
, &hdigit
, 1);
190 hdigit
= HEXDIGITS(val
);
191 luaL_addlstring(&buf
, &hdigit
, 1);
193 luaL_pushresult(&buf
);
207 lua_pushnumber(L
,i
+1);
212 #define PACKNUMBER(OP,T) \
215 T a=(T)luaL_checknumber(L,i++); \
216 doswap(swap,&a,sizeof(a)); \
217 luaL_addlstring(&b,(char*)&a,sizeof(a)); \
221 #define PACKSTRING(OP,T) \
225 const char *a=luaL_checklstring(L,i++,&l); \
227 doswap(swap,&ll,sizeof(ll)); \
228 luaL_addlstring(&b,(char*)&ll,sizeof(ll)); \
229 luaL_addlstring(&b,a,l); \
233 static int l_pack(lua_State
*L
) /** pack(f,...) */
236 const char *f
=luaL_checkstring(L
,1);
244 if (isdigit((int) (unsigned char) *f
))
247 while (isdigit((int) (unsigned char) *f
)) N
=10*N
+(*f
++)-'0';
249 while (N
--) switch (c
)
251 case OP_LITTLEENDIAN
:
263 const char *a
=luaL_checklstring(L
,i
++,&l
);
264 luaL_addlstring(&b
,a
,l
+(c
==OP_ZSTRING
));
267 PACKSTRING(OP_BSTRING
, unsigned char)
268 PACKSTRING(OP_WSTRING
, unsigned short)
269 PACKSTRING(OP_SSTRING
, size_t)
270 PACKNUMBER(OP_NUMBER
, lua_Number
)
271 PACKNUMBER(OP_DOUBLE
, double)
272 PACKNUMBER(OP_FLOAT
, float)
273 PACKNUMBER(OP_CHAR
, char)
274 PACKNUMBER(OP_BYTE
, unsigned char)
275 PACKNUMBER(OP_SHORT
, short)
276 PACKNUMBER(OP_USHORT
, unsigned short)
277 PACKNUMBER(OP_INT
, int)
278 PACKNUMBER(OP_UINT
, unsigned int)
279 PACKNUMBER(OP_LONG
, long)
280 PACKNUMBER(OP_ULONG
, unsigned long)
282 { // doing digit parsing the lpack way
283 unsigned char sbyte
= 0;
287 const char *a
= luaL_checklstring(L
, i
++, &l
);
288 for (ii
= 0; ii
< l
; ii
++) {
289 if (isxdigit((int) (unsigned char) a
[ii
])) {
290 if (isdigit((int) (unsigned char) a
[ii
])) {
291 sbyte
+= a
[ii
] - '0';
293 } else if (a
[ii
] >= 'A' && a
[ii
] <= 'F') {
294 sbyte
+= a
[ii
] - 'A' + 10;
296 } else if (a
[ii
] >= 'a' && a
[ii
] <= 'f') {
297 sbyte
+= a
[ii
] - 'a' + 10;
302 } else if (odd
== 2) {
303 luaL_addlstring(&b
, (char *) &sbyte
, 1);
307 } else if (isspace(a
[ii
])) {
310 /* err ... ignore too*/
314 luaL_addlstring(&b
, (char *) &sbyte
, 1);
329 static const luaL_Reg binlib
[] =
332 {"unpack", l_unpack
},
336 LUALIB_API
int luaopen_binlib (lua_State
*L
) {
337 luaL_newlib(L
, binlib
);
343 int set_bin_library (lua_State
*L
) {
345 luaL_requiref(L
, "bin", luaopen_binlib
, 1);