msgpack/perl/pack.c

156 lines
3.8 KiB
C
Raw Normal View History

/*
* code is written by tokuhirom.
* buffer alocation technique is taken from JSON::XS. thanks to mlehmann.
*/
2009-04-15 12:55:41 +09:00
#ifdef __cplusplus
extern "C" {
#endif
#include "EXTERN.h"
#include "perl.h"
#include "XSUB.h"
#include "ppport.h"
#ifdef __cplusplus
};
#endif
#include "msgpack/pack_define.h"
#define msgpack_pack_inline_func(name) \
static inline void msgpack_pack ## name
#define msgpack_pack_inline_func_cint(name) \
static inline void msgpack_pack ## name
typedef struct {
char *cur; /* SvPVX (sv) + current output position */
char *end; /* SvEND (sv) */
SV *sv; /* result scalar */
} enc_t;
void need(enc_t *enc, STRLEN len);
2009-04-15 12:55:41 +09:00
#define msgpack_pack_user enc_t*
#define msgpack_pack_append_buffer(enc, buf, len) \
need(enc, len); \
memcpy(enc->cur, buf, len); \
enc->cur += len;
2009-04-15 12:55:41 +09:00
#include "msgpack/pack_template.h"
#define _PACK_WRAPPER(t) msgpack_pack_##t
#define PACK_WRAPPER(t) _PACK_WRAPPER(t)
#define INIT_SIZE 32 /* initial scalar size to be allocated */
void need(enc_t *enc, STRLEN len)
{
if (enc->cur + len >= enc->end) {
STRLEN cur = enc->cur - (char *)SvPVX (enc->sv);
SvGROW (enc->sv, cur + (len < (cur >> 2) ? cur >> 2 : len) + 1);
enc->cur = SvPVX (enc->sv) + cur;
enc->end = SvPVX (enc->sv) + SvLEN (enc->sv) - 1;
}
}
2009-04-15 12:55:41 +09:00
static void _msgpack_pack_sv(enc_t *enc, SV* val) {
2009-04-15 12:55:41 +09:00
if (val==NULL) {
msgpack_pack_nil(enc);
2009-04-15 12:55:41 +09:00
return;
}
switch (SvTYPE(val)) {
case SVt_NULL:
msgpack_pack_nil(enc);
2009-04-15 12:55:41 +09:00
break;
case SVt_IV:
if (SvIOK_UV(val)) {
msgpack_pack_uint32(enc, SvUV(val));
2009-04-15 12:55:41 +09:00
} else {
PACK_WRAPPER(IVTYPE)(enc, SvIV(val));
2009-04-15 12:55:41 +09:00
}
break;
case SVt_PVNV:
{
STRLEN len = 0;
char *pv = SvPV(val, len);
if (len == 1 && *pv == '1') {
msgpack_pack_true(enc);
2009-04-15 12:55:41 +09:00
} else if (len == 0 && *pv==0) {
msgpack_pack_false(enc);
2009-04-15 12:55:41 +09:00
} else {
msgpack_pack_nil(enc);
2009-04-15 12:55:41 +09:00
}
}
break;
case SVt_PV:
{
STRLEN len;
char * cval = SvPV(val, len);
msgpack_pack_raw(enc, len);
msgpack_pack_raw_body(enc, cval, len);
2009-04-15 12:55:41 +09:00
}
break;
case SVt_NV:
PACK_WRAPPER(NVTYPE)(enc, SvNV(val));
2009-04-15 12:55:41 +09:00
break;
case SVt_PVAV:
{
AV* ary = (AV*)val;
int len = av_len(ary) + 1;
int i;
msgpack_pack_array(enc, len);
2009-04-15 12:55:41 +09:00
for (i=0; i<len; i++) {
SV** svp = av_fetch(ary, i, 0);
if (svp) {
_msgpack_pack_sv(enc, *svp);
2009-04-15 12:55:41 +09:00
} else {
msgpack_pack_nil(enc);
2009-04-15 12:55:41 +09:00
}
}
}
break;
case SVt_PVHV:
{
HV* hval = (HV*)val;
int count = hv_iterinit(hval);
HE* he;
msgpack_pack_map(enc, count);
2009-04-15 12:55:41 +09:00
while (he = hv_iternext(hval)) {
_msgpack_pack_sv(enc, hv_iterkeysv(he));
_msgpack_pack_sv(enc, HeVAL(he));
2009-04-15 12:55:41 +09:00
}
}
break;
case SVt_RV:
_msgpack_pack_sv(enc, SvRV(val));
2009-04-15 12:55:41 +09:00
break;
default:
sv_dump(val);
Perl_croak(aTHX_ "msgpack for perl doesn't supported this type: %d\n", SvTYPE(val));
}
}
XS(xs_pack) {
dXSARGS;
2009-04-15 23:11:26 +09:00
if (items != 2) {
Perl_croak(aTHX_ "Usage: Data::MessagePack->pack($dat)");
}
2009-04-15 12:55:41 +09:00
SV* val = ST(1);
enc_t enc;
enc.sv = sv_2mortal(NEWSV(0, INIT_SIZE));
enc.cur = SvPVX(enc.sv);
enc.end = SvEND(enc.sv);
SvPOK_only(enc.sv);
_msgpack_pack_sv(&enc, val);
SvCUR_set(enc.sv, enc.cur - SvPVX (enc.sv));
*SvEND (enc.sv) = 0; /* many xs functions expect a trailing 0 for text strings */
2009-04-15 12:55:41 +09:00
ST(0) = enc.sv;
2009-04-15 12:55:41 +09:00
XSRETURN(1);
}