12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454 |
- /* Lua CJSON - JSON support for Lua
- *
- * Copyright (c) 2010-2012 Mark Pulford <mark@kyne.com.au>
- *
- * Permission is hereby granted, free of charge, to any person obtaining
- * a copy of this software and associated documentation files (the
- * "Software"), to deal in the Software without restriction, including
- * without limitation the rights to use, copy, modify, merge, publish,
- * distribute, sublicense, and/or sell copies of the Software, and to
- * permit persons to whom the Software is furnished to do so, subject to
- * the following conditions:
- *
- * The above copyright notice and this permission notice shall be
- * included in all copies or substantial portions of the Software.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
- * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
- * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
- * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
- * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
- */
- /* Caveats:
- * - JSON "null" values are represented as lightuserdata since Lua
- * tables cannot contain "nil". Compare with cjson.null.
- * - Invalid UTF-8 characters are not detected and will be passed
- * untouched. If required, UTF-8 error checking should be done
- * outside this library.
- * - Javascript comments are not part of the JSON spec, and are not
- * currently supported.
- *
- * Note: Decoding is slower than encoding. Lua spends significant
- * time (30%) managing tables when parsing JSON since it is
- * difficult to know object/array sizes ahead of time.
- */
- #include <assert.h>
- #include <string.h>
- #include <math.h>
- #include <limits.h>
- #include <lua.h>
- #include <lauxlib.h>
- #include "strbuf.h"
- #include "fpconv.h"
- #ifndef CJSON_MODNAME
- #define CJSON_MODNAME "cjson"
- #endif
- #ifndef CJSON_VERSION
- #define CJSON_VERSION "2.1devel"
- #endif
- /* Workaround for Solaris platforms missing isinf() */
- #if !defined(isinf) && (defined(USE_INTERNAL_ISINF) || defined(MISSING_ISINF))
- #define isinf(x) (!isnan(x) && isnan((x) - (x)))
- #endif
- #define DEFAULT_SPARSE_CONVERT 0
- #define DEFAULT_SPARSE_RATIO 2
- #define DEFAULT_SPARSE_SAFE 10
- #define DEFAULT_ENCODE_MAX_DEPTH 1000
- #define DEFAULT_DECODE_MAX_DEPTH 1000
- #define DEFAULT_ENCODE_INVALID_NUMBERS 0
- #define DEFAULT_DECODE_INVALID_NUMBERS 1
- #define DEFAULT_ENCODE_KEEP_BUFFER 1
- #define DEFAULT_ENCODE_NUMBER_PRECISION 14
- #ifdef DISABLE_INVALID_NUMBERS
- #undef DEFAULT_DECODE_INVALID_NUMBERS
- #define DEFAULT_DECODE_INVALID_NUMBERS 0
- #endif
- typedef enum {
- T_OBJ_BEGIN,
- T_OBJ_END,
- T_ARR_BEGIN,
- T_ARR_END,
- T_STRING,
- T_NUMBER,
- T_INTEGER,
- T_BOOLEAN,
- T_NULL,
- T_COLON,
- T_COMMA,
- T_END,
- T_WHITESPACE,
- T_ERROR,
- T_UNKNOWN
- } json_token_type_t;
- static const char *json_token_type_name[] = {
- "T_OBJ_BEGIN",
- "T_OBJ_END",
- "T_ARR_BEGIN",
- "T_ARR_END",
- "T_STRING",
- "T_NUMBER",
- "T_INTEGER",
- "T_BOOLEAN",
- "T_NULL",
- "T_COLON",
- "T_COMMA",
- "T_END",
- "T_WHITESPACE",
- "T_ERROR",
- "T_UNKNOWN",
- NULL
- };
- typedef struct {
- json_token_type_t ch2token[256];
- char escape2char[256]; /* Decoding */
- /* encode_buf is only allocated and used when
- * encode_keep_buffer is set */
- strbuf_t encode_buf;
- int encode_sparse_convert;
- int encode_sparse_ratio;
- int encode_sparse_safe;
- int encode_max_depth;
- int encode_invalid_numbers; /* 2 => Encode as "null" */
- int encode_number_precision;
- int encode_keep_buffer;
- int decode_invalid_numbers;
- int decode_max_depth;
- } json_config_t;
- typedef struct {
- const char *data;
- const char *ptr;
- strbuf_t *tmp; /* Temporary storage for strings */
- json_config_t *cfg;
- int current_depth;
- } json_parse_t;
- typedef struct {
- json_token_type_t type;
- int index;
- union {
- const char *string;
- double number;
- lua_Integer integer;
- int boolean;
- } value;
- int string_len;
- } json_token_t;
- static const char *char2escape[256] = {
- "\\u0000", "\\u0001", "\\u0002", "\\u0003",
- "\\u0004", "\\u0005", "\\u0006", "\\u0007",
- "\\b", "\\t", "\\n", "\\u000b",
- "\\f", "\\r", "\\u000e", "\\u000f",
- "\\u0010", "\\u0011", "\\u0012", "\\u0013",
- "\\u0014", "\\u0015", "\\u0016", "\\u0017",
- "\\u0018", "\\u0019", "\\u001a", "\\u001b",
- "\\u001c", "\\u001d", "\\u001e", "\\u001f",
- NULL, NULL, "\\\"", NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, "\\/",
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, "\\\\", NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, "\\u007f",
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
- };
- /* ===== CONFIGURATION ===== */
- static json_config_t *json_fetch_config(lua_State *l)
- {
- json_config_t *cfg;
- cfg = lua_touserdata(l, lua_upvalueindex(1));
- if (!cfg)
- luaL_error(l, "BUG: Unable to fetch CJSON configuration");
- return cfg;
- }
- /* Ensure the correct number of arguments have been provided.
- * Pad with nil to allow other functions to simply check arg[i]
- * to find whether an argument was provided */
- static json_config_t *json_arg_init(lua_State *l, int args)
- {
- luaL_argcheck(l, lua_gettop(l) <= args, args + 1,
- "found too many arguments");
- while (lua_gettop(l) < args)
- lua_pushnil(l);
- return json_fetch_config(l);
- }
- /* Process integer options for configuration functions */
- static int json_integer_option(lua_State *l, int optindex, int *setting,
- int min, int max)
- {
- char errmsg[64];
- int value;
- if (!lua_isnil(l, optindex)) {
- value = luaL_checkinteger(l, optindex);
- snprintf(errmsg, sizeof(errmsg), "expected integer between %d and %d", min, max);
- luaL_argcheck(l, min <= value && value <= max, 1, errmsg);
- *setting = value;
- }
- lua_pushinteger(l, *setting);
- return 1;
- }
- /* Process enumerated arguments for a configuration function */
- static int json_enum_option(lua_State *l, int optindex, int *setting,
- const char **options, int bool_true)
- {
- static const char *bool_options[] = { "off", "on", NULL };
- if (!options) {
- options = bool_options;
- bool_true = 1;
- }
- if (!lua_isnil(l, optindex)) {
- if (bool_true && lua_isboolean(l, optindex))
- *setting = lua_toboolean(l, optindex) * bool_true;
- else
- *setting = luaL_checkoption(l, optindex, NULL, options);
- }
- if (bool_true && (*setting == 0 || *setting == bool_true))
- lua_pushboolean(l, *setting);
- else
- lua_pushstring(l, options[*setting]);
- return 1;
- }
- /* Configures handling of extremely sparse arrays:
- * convert: Convert extremely sparse arrays into objects? Otherwise error.
- * ratio: 0: always allow sparse; 1: never allow sparse; >1: use ratio
- * safe: Always use an array when the max index <= safe */
- static int json_cfg_encode_sparse_array(lua_State *l)
- {
- json_config_t *cfg = json_arg_init(l, 3);
- json_enum_option(l, 1, &cfg->encode_sparse_convert, NULL, 1);
- json_integer_option(l, 2, &cfg->encode_sparse_ratio, 0, INT_MAX);
- json_integer_option(l, 3, &cfg->encode_sparse_safe, 0, INT_MAX);
- return 3;
- }
- /* Configures the maximum number of nested arrays/objects allowed when
- * encoding */
- static int json_cfg_encode_max_depth(lua_State *l)
- {
- json_config_t *cfg = json_arg_init(l, 1);
- return json_integer_option(l, 1, &cfg->encode_max_depth, 1, INT_MAX);
- }
- /* Configures the maximum number of nested arrays/objects allowed when
- * encoding */
- static int json_cfg_decode_max_depth(lua_State *l)
- {
- json_config_t *cfg = json_arg_init(l, 1);
- return json_integer_option(l, 1, &cfg->decode_max_depth, 1, INT_MAX);
- }
- /* Configures number precision when converting doubles to text */
- static int json_cfg_encode_number_precision(lua_State *l)
- {
- json_config_t *cfg = json_arg_init(l, 1);
- return json_integer_option(l, 1, &cfg->encode_number_precision, 1, 14);
- }
- /* Configures JSON encoding buffer persistence */
- static int json_cfg_encode_keep_buffer(lua_State *l)
- {
- json_config_t *cfg = json_arg_init(l, 1);
- int old_value;
- old_value = cfg->encode_keep_buffer;
- json_enum_option(l, 1, &cfg->encode_keep_buffer, NULL, 1);
- /* Init / free the buffer if the setting has changed */
- if (old_value ^ cfg->encode_keep_buffer) {
- if (cfg->encode_keep_buffer)
- strbuf_init(&cfg->encode_buf, 0);
- else
- strbuf_free(&cfg->encode_buf);
- }
- return 1;
- }
- #if defined(DISABLE_INVALID_NUMBERS) && !defined(USE_INTERNAL_FPCONV)
- void json_verify_invalid_number_setting(lua_State *l, int *setting)
- {
- if (*setting == 1) {
- *setting = 0;
- luaL_error(l, "Infinity, NaN, and/or hexadecimal numbers are not supported.");
- }
- }
- #else
- #define json_verify_invalid_number_setting(l, s) do { } while(0)
- #endif
- static int json_cfg_encode_invalid_numbers(lua_State *l)
- {
- static const char *options[] = { "off", "on", "null", NULL };
- json_config_t *cfg = json_arg_init(l, 1);
- json_enum_option(l, 1, &cfg->encode_invalid_numbers, options, 1);
- json_verify_invalid_number_setting(l, &cfg->encode_invalid_numbers);
- return 1;
- }
- static int json_cfg_decode_invalid_numbers(lua_State *l)
- {
- json_config_t *cfg = json_arg_init(l, 1);
- json_enum_option(l, 1, &cfg->decode_invalid_numbers, NULL, 1);
- json_verify_invalid_number_setting(l, &cfg->encode_invalid_numbers);
- return 1;
- }
- static int json_destroy_config(lua_State *l)
- {
- json_config_t *cfg;
- cfg = lua_touserdata(l, 1);
- if (cfg)
- strbuf_free(&cfg->encode_buf);
- cfg = NULL;
- return 0;
- }
- static void json_create_config(lua_State *l)
- {
- json_config_t *cfg;
- int i;
- cfg = lua_newuserdata(l, sizeof(*cfg));
- /* Create GC method to clean up strbuf */
- lua_newtable(l);
- lua_pushcfunction(l, json_destroy_config);
- lua_setfield(l, -2, "__gc");
- lua_setmetatable(l, -2);
- cfg->encode_sparse_convert = DEFAULT_SPARSE_CONVERT;
- cfg->encode_sparse_ratio = DEFAULT_SPARSE_RATIO;
- cfg->encode_sparse_safe = DEFAULT_SPARSE_SAFE;
- cfg->encode_max_depth = DEFAULT_ENCODE_MAX_DEPTH;
- cfg->decode_max_depth = DEFAULT_DECODE_MAX_DEPTH;
- cfg->encode_invalid_numbers = DEFAULT_ENCODE_INVALID_NUMBERS;
- cfg->decode_invalid_numbers = DEFAULT_DECODE_INVALID_NUMBERS;
- cfg->encode_keep_buffer = DEFAULT_ENCODE_KEEP_BUFFER;
- cfg->encode_number_precision = DEFAULT_ENCODE_NUMBER_PRECISION;
- #if DEFAULT_ENCODE_KEEP_BUFFER > 0
- strbuf_init(&cfg->encode_buf, 0);
- #endif
- /* Decoding init */
- /* Tag all characters as an error */
- for (i = 0; i < 256; i++)
- cfg->ch2token[i] = T_ERROR;
- /* Set tokens that require no further processing */
- cfg->ch2token['{'] = T_OBJ_BEGIN;
- cfg->ch2token['}'] = T_OBJ_END;
- cfg->ch2token['['] = T_ARR_BEGIN;
- cfg->ch2token[']'] = T_ARR_END;
- cfg->ch2token[','] = T_COMMA;
- cfg->ch2token[':'] = T_COLON;
- cfg->ch2token['\0'] = T_END;
- cfg->ch2token[' '] = T_WHITESPACE;
- cfg->ch2token['\t'] = T_WHITESPACE;
- cfg->ch2token['\n'] = T_WHITESPACE;
- cfg->ch2token['\r'] = T_WHITESPACE;
- /* Update characters that require further processing */
- cfg->ch2token['f'] = T_UNKNOWN; /* false? */
- cfg->ch2token['i'] = T_UNKNOWN; /* inf, ininity? */
- cfg->ch2token['I'] = T_UNKNOWN;
- cfg->ch2token['n'] = T_UNKNOWN; /* null, nan? */
- cfg->ch2token['N'] = T_UNKNOWN;
- cfg->ch2token['t'] = T_UNKNOWN; /* true? */
- cfg->ch2token['"'] = T_UNKNOWN; /* string? */
- cfg->ch2token['+'] = T_UNKNOWN; /* number? */
- cfg->ch2token['-'] = T_UNKNOWN;
- for (i = 0; i < 10; i++)
- cfg->ch2token['0' + i] = T_UNKNOWN;
- /* Lookup table for parsing escape characters */
- for (i = 0; i < 256; i++)
- cfg->escape2char[i] = 0; /* String error */
- cfg->escape2char['"'] = '"';
- cfg->escape2char['\\'] = '\\';
- cfg->escape2char['/'] = '/';
- cfg->escape2char['b'] = '\b';
- cfg->escape2char['t'] = '\t';
- cfg->escape2char['n'] = '\n';
- cfg->escape2char['f'] = '\f';
- cfg->escape2char['r'] = '\r';
- cfg->escape2char['u'] = 'u'; /* Unicode parsing required */
- }
- /* ===== ENCODING ===== */
- static void json_encode_exception(lua_State *l, json_config_t *cfg, strbuf_t *json, int lindex,
- const char *reason)
- {
- if (!cfg->encode_keep_buffer)
- strbuf_free(json);
- luaL_error(l, "Cannot serialise %s: %s",
- lua_typename(l, lua_type(l, lindex)), reason);
- }
- /* json_append_string args:
- * - lua_State
- * - JSON strbuf
- * - String (Lua stack index)
- *
- * Returns nothing. Doesn't remove string from Lua stack */
- static void json_append_string(lua_State *l, strbuf_t *json, int lindex)
- {
- const char *escstr;
- int i;
- const char *str;
- size_t len;
- str = lua_tolstring(l, lindex, &len);
- /* Worst case is len * 6 (all unicode escapes).
- * This buffer is reused constantly for small strings
- * If there are any excess pages, they won't be hit anyway.
- * This gains ~5% speedup. */
- strbuf_ensure_empty_length(json, len * 6 + 2);
- strbuf_append_char_unsafe(json, '\"');
- for (i = 0; i < len; i++) {
- escstr = char2escape[(unsigned char)str[i]];
- if (escstr)
- strbuf_append_string(json, escstr);
- else
- strbuf_append_char_unsafe(json, str[i]);
- }
- strbuf_append_char_unsafe(json, '\"');
- }
- /* Find the size of the array on the top of the Lua stack
- * -1 object (not a pure array)
- * >=0 elements in array
- */
- static int lua_array_length(lua_State *l, json_config_t *cfg, strbuf_t *json)
- {
- double k;
- int max;
- int items;
- max = 0;
- items = 0;
- lua_pushnil(l);
- /* table, startkey */
- while (lua_next(l, -2) != 0) {
- /* table, key, value */
- if (lua_type(l, -2) == LUA_TNUMBER &&
- (k = lua_tonumber(l, -2))) {
- /* Integer >= 1 ? */
- if (floor(k) == k && k >= 1) {
- if (k > max)
- max = k;
- items++;
- lua_pop(l, 1);
- continue;
- }
- }
- /* Must not be an array (non integer key) */
- lua_pop(l, 2);
- return -1;
- }
- /* Encode excessively sparse arrays as objects (if enabled) */
- if (cfg->encode_sparse_ratio > 0 &&
- max > items * cfg->encode_sparse_ratio &&
- max > cfg->encode_sparse_safe) {
- if (!cfg->encode_sparse_convert)
- json_encode_exception(l, cfg, json, -1, "excessively sparse array");
- return -1;
- }
- return max;
- }
- static void json_check_encode_depth(lua_State *l, json_config_t *cfg,
- int current_depth, strbuf_t *json)
- {
- /* Ensure there are enough slots free to traverse a table (key,
- * value) and push a string for a potential error message.
- *
- * Unlike "decode", the key and value are still on the stack when
- * lua_checkstack() is called. Hence an extra slot for luaL_error()
- * below is required just in case the next check to lua_checkstack()
- * fails.
- *
- * While this won't cause a crash due to the EXTRA_STACK reserve
- * slots, it would still be an improper use of the API. */
- if (current_depth <= cfg->encode_max_depth && lua_checkstack(l, 3))
- return;
- if (!cfg->encode_keep_buffer)
- strbuf_free(json);
- luaL_error(l, "Cannot serialise, excessive nesting (%d)",
- current_depth);
- }
- static void json_append_data(lua_State *l, json_config_t *cfg,
- int current_depth, strbuf_t *json);
- /* json_append_array args:
- * - lua_State
- * - JSON strbuf
- * - Size of passwd Lua array (top of stack) */
- static void json_append_array(lua_State *l, json_config_t *cfg, int current_depth,
- strbuf_t *json, int array_length)
- {
- int comma, i;
- strbuf_append_char(json, '[');
- comma = 0;
- for (i = 1; i <= array_length; i++) {
- if (comma)
- strbuf_append_char(json, ',');
- else
- comma = 1;
- lua_rawgeti(l, -1, i);
- json_append_data(l, cfg, current_depth, json);
- lua_pop(l, 1);
- }
- strbuf_append_char(json, ']');
- }
- static void json_append_number(lua_State *l, json_config_t *cfg,
- strbuf_t *json, int lindex)
- {
- int len;
- #if LUA_VERSION_NUM >= 503
- if (lua_isinteger(l, lindex)) {
- lua_Integer num = lua_tointeger(l, lindex);
- strbuf_ensure_empty_length(json, FPCONV_G_FMT_BUFSIZE); /* max length of int64 is 19 */
- len = sprintf(strbuf_empty_ptr(json), LUA_INTEGER_FMT, num);
- strbuf_extend_length(json, len);
- return;
- }
- #endif
- double num = lua_tonumber(l, lindex);
- if (cfg->encode_invalid_numbers == 0) {
- /* Prevent encoding invalid numbers */
- if (isinf(num) || isnan(num))
- json_encode_exception(l, cfg, json, lindex,
- "must not be NaN or Infinity");
- } else if (cfg->encode_invalid_numbers == 1) {
- /* Encode NaN/Infinity separately to ensure Javascript compatible
- * values are used. */
- if (isnan(num)) {
- strbuf_append_mem(json, "NaN", 3);
- return;
- }
- if (isinf(num)) {
- if (num < 0)
- strbuf_append_mem(json, "-Infinity", 9);
- else
- strbuf_append_mem(json, "Infinity", 8);
- return;
- }
- } else {
- /* Encode invalid numbers as "null" */
- if (isinf(num) || isnan(num)) {
- strbuf_append_mem(json, "null", 4);
- return;
- }
- }
- strbuf_ensure_empty_length(json, FPCONV_G_FMT_BUFSIZE);
- len = fpconv_g_fmt(strbuf_empty_ptr(json), num, cfg->encode_number_precision);
- strbuf_extend_length(json, len);
- }
- static void json_append_object(lua_State *l, json_config_t *cfg,
- int current_depth, strbuf_t *json)
- {
- int comma, keytype;
- /* Object */
- strbuf_append_char(json, '{');
- lua_pushnil(l);
- /* table, startkey */
- comma = 0;
- while (lua_next(l, -2) != 0) {
- if (comma)
- strbuf_append_char(json, ',');
- else
- comma = 1;
- /* table, key, value */
- keytype = lua_type(l, -2);
- if (keytype == LUA_TNUMBER) {
- strbuf_append_char(json, '"');
- json_append_number(l, cfg, json, -2);
- strbuf_append_mem(json, "\":", 2);
- } else if (keytype == LUA_TSTRING) {
- json_append_string(l, json, -2);
- strbuf_append_char(json, ':');
- } else {
- json_encode_exception(l, cfg, json, -2,
- "table key must be a number or string");
- /* never returns */
- }
- /* table, key, value */
- json_append_data(l, cfg, current_depth, json);
- lua_pop(l, 1);
- /* table, key */
- }
- strbuf_append_char(json, '}');
- }
- /* Serialise Lua data into JSON string. */
- static void json_append_data(lua_State *l, json_config_t *cfg,
- int current_depth, strbuf_t *json)
- {
- int len;
- switch (lua_type(l, -1)) {
- case LUA_TSTRING:
- json_append_string(l, json, -1);
- break;
- case LUA_TNUMBER:
- json_append_number(l, cfg, json, -1);
- break;
- case LUA_TBOOLEAN:
- if (lua_toboolean(l, -1))
- strbuf_append_mem(json, "true", 4);
- else
- strbuf_append_mem(json, "false", 5);
- break;
- case LUA_TTABLE:
- current_depth++;
- json_check_encode_depth(l, cfg, current_depth, json);
- len = lua_array_length(l, cfg, json);
- if (len > 0)
- json_append_array(l, cfg, current_depth, json, len);
- else
- json_append_object(l, cfg, current_depth, json);
- break;
- case LUA_TNIL:
- strbuf_append_mem(json, "null", 4);
- break;
- case LUA_TLIGHTUSERDATA:
- if (lua_touserdata(l, -1) == NULL) {
- strbuf_append_mem(json, "null", 4);
- break;
- }
- default:
- /* Remaining types (LUA_TFUNCTION, LUA_TUSERDATA, LUA_TTHREAD,
- * and LUA_TLIGHTUSERDATA) cannot be serialised */
- json_encode_exception(l, cfg, json, -1, "type not supported");
- /* never returns */
- }
- }
- static int json_encode(lua_State *l)
- {
- json_config_t *cfg = json_fetch_config(l);
- strbuf_t local_encode_buf;
- strbuf_t *encode_buf;
- char *json;
- int len;
- luaL_argcheck(l, lua_gettop(l) == 1, 1, "expected 1 argument");
- if (!cfg->encode_keep_buffer) {
- /* Use private buffer */
- encode_buf = &local_encode_buf;
- strbuf_init(encode_buf, 0);
- } else {
- /* Reuse existing buffer */
- encode_buf = &cfg->encode_buf;
- strbuf_reset(encode_buf);
- }
- json_append_data(l, cfg, 0, encode_buf);
- json = strbuf_string(encode_buf, &len);
- lua_pushlstring(l, json, len);
- if (!cfg->encode_keep_buffer)
- strbuf_free(encode_buf);
- return 1;
- }
- /* ===== DECODING ===== */
- static void json_process_value(lua_State *l, json_parse_t *json,
- json_token_t *token);
- static int hexdigit2int(char hex)
- {
- if ('0' <= hex && hex <= '9')
- return hex - '0';
- /* Force lowercase */
- hex |= 0x20;
- if ('a' <= hex && hex <= 'f')
- return 10 + hex - 'a';
- return -1;
- }
- static int decode_hex4(const char *hex)
- {
- int digit[4];
- int i;
- /* Convert ASCII hex digit to numeric digit
- * Note: this returns an error for invalid hex digits, including
- * NULL */
- for (i = 0; i < 4; i++) {
- digit[i] = hexdigit2int(hex[i]);
- if (digit[i] < 0) {
- return -1;
- }
- }
- return (digit[0] << 12) +
- (digit[1] << 8) +
- (digit[2] << 4) +
- digit[3];
- }
- /* Converts a Unicode codepoint to UTF-8.
- * Returns UTF-8 string length, and up to 4 bytes in *utf8 */
- static int codepoint_to_utf8(char *utf8, int codepoint)
- {
- /* 0xxxxxxx */
- if (codepoint <= 0x7F) {
- utf8[0] = codepoint;
- return 1;
- }
- /* 110xxxxx 10xxxxxx */
- if (codepoint <= 0x7FF) {
- utf8[0] = (codepoint >> 6) | 0xC0;
- utf8[1] = (codepoint & 0x3F) | 0x80;
- return 2;
- }
- /* 1110xxxx 10xxxxxx 10xxxxxx */
- if (codepoint <= 0xFFFF) {
- utf8[0] = (codepoint >> 12) | 0xE0;
- utf8[1] = ((codepoint >> 6) & 0x3F) | 0x80;
- utf8[2] = (codepoint & 0x3F) | 0x80;
- return 3;
- }
- /* 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx */
- if (codepoint <= 0x1FFFFF) {
- utf8[0] = (codepoint >> 18) | 0xF0;
- utf8[1] = ((codepoint >> 12) & 0x3F) | 0x80;
- utf8[2] = ((codepoint >> 6) & 0x3F) | 0x80;
- utf8[3] = (codepoint & 0x3F) | 0x80;
- return 4;
- }
- return 0;
- }
- /* Called when index pointing to beginning of UTF-16 code escape: \uXXXX
- * \u is guaranteed to exist, but the remaining hex characters may be
- * missing.
- * Translate to UTF-8 and append to temporary token string.
- * Must advance index to the next character to be processed.
- * Returns: 0 success
- * -1 error
- */
- static int json_append_unicode_escape(json_parse_t *json)
- {
- char utf8[4]; /* Surrogate pairs require 4 UTF-8 bytes */
- int codepoint;
- int surrogate_low;
- int len;
- int escape_len = 6;
- /* Fetch UTF-16 code unit */
- codepoint = decode_hex4(json->ptr + 2);
- if (codepoint < 0)
- return -1;
- /* UTF-16 surrogate pairs take the following 2 byte form:
- * 11011 x yyyyyyyyyy
- * When x = 0: y is the high 10 bits of the codepoint
- * x = 1: y is the low 10 bits of the codepoint
- *
- * Check for a surrogate pair (high or low) */
- if ((codepoint & 0xF800) == 0xD800) {
- /* Error if the 1st surrogate is not high */
- if (codepoint & 0x400)
- return -1;
- /* Ensure the next code is a unicode escape */
- if (*(json->ptr + escape_len) != '\\' ||
- *(json->ptr + escape_len + 1) != 'u') {
- return -1;
- }
- /* Fetch the next codepoint */
- surrogate_low = decode_hex4(json->ptr + 2 + escape_len);
- if (surrogate_low < 0)
- return -1;
- /* Error if the 2nd code is not a low surrogate */
- if ((surrogate_low & 0xFC00) != 0xDC00)
- return -1;
- /* Calculate Unicode codepoint */
- codepoint = (codepoint & 0x3FF) << 10;
- surrogate_low &= 0x3FF;
- codepoint = (codepoint | surrogate_low) + 0x10000;
- escape_len = 12;
- }
- /* Convert codepoint to UTF-8 */
- len = codepoint_to_utf8(utf8, codepoint);
- if (!len)
- return -1;
- /* Append bytes and advance parse index */
- strbuf_append_mem_unsafe(json->tmp, utf8, len);
- json->ptr += escape_len;
- return 0;
- }
- static void json_set_token_error(json_token_t *token, json_parse_t *json,
- const char *errtype)
- {
- token->type = T_ERROR;
- token->index = json->ptr - json->data;
- token->value.string = errtype;
- }
- static void json_next_string_token(json_parse_t *json, json_token_t *token)
- {
- char *escape2char = json->cfg->escape2char;
- char ch;
- /* Caller must ensure a string is next */
- assert(*json->ptr == '"');
- /* Skip " */
- json->ptr++;
- /* json->tmp is the temporary strbuf used to accumulate the
- * decoded string value.
- * json->tmp is sized to handle JSON containing only a string value.
- */
- strbuf_reset(json->tmp);
- while ((ch = *json->ptr) != '"') {
- if (!ch) {
- /* Premature end of the string */
- json_set_token_error(token, json, "unexpected end of string");
- return;
- }
- /* Handle escapes */
- if (ch == '\\') {
- /* Fetch escape character */
- ch = *(json->ptr + 1);
- /* Translate escape code and append to tmp string */
- ch = escape2char[(unsigned char)ch];
- if (ch == 'u') {
- if (json_append_unicode_escape(json) == 0)
- continue;
- json_set_token_error(token, json,
- "invalid unicode escape code");
- return;
- }
- if (!ch) {
- json_set_token_error(token, json, "invalid escape code");
- return;
- }
- /* Skip '\' */
- json->ptr++;
- }
- /* Append normal character or translated single character
- * Unicode escapes are handled above */
- strbuf_append_char_unsafe(json->tmp, ch);
- json->ptr++;
- }
- json->ptr++; /* Eat final quote (") */
- strbuf_ensure_null(json->tmp);
- token->type = T_STRING;
- token->value.string = strbuf_string(json->tmp, &token->string_len);
- }
- /* JSON numbers should take the following form:
- * -?(0|[1-9]|[1-9][0-9]+)(.[0-9]+)?([eE][-+]?[0-9]+)?
- *
- * json_next_number_token() uses strtod() which allows other forms:
- * - numbers starting with '+'
- * - NaN, -NaN, infinity, -infinity
- * - hexadecimal numbers
- * - numbers with leading zeros
- *
- * json_is_invalid_number() detects "numbers" which may pass strtod()'s
- * error checking, but should not be allowed with strict JSON.
- *
- * json_is_invalid_number() may pass numbers which cause strtod()
- * to generate an error.
- */
- static int json_is_invalid_number(json_parse_t *json)
- {
- const char *p = json->ptr;
- /* Reject numbers starting with + */
- if (*p == '+')
- return 1;
- /* Skip minus sign if it exists */
- if (*p == '-')
- p++;
- /* Reject numbers starting with 0x, or leading zeros */
- if (*p == '0') {
- int ch2 = *(p + 1);
- if ((ch2 | 0x20) == 'x' || /* Hex */
- ('0' <= ch2 && ch2 <= '9')) /* Leading zero */
- return 1;
- return 0;
- } else if (*p <= '9') {
- return 0; /* Ordinary number */
- }
- /* Reject inf/nan */
- if (!strncasecmp(p, "inf", 3))
- return 1;
- if (!strncasecmp(p, "nan", 3))
- return 1;
- /* Pass all other numbers which may still be invalid, but
- * strtod() will catch them. */
- return 0;
- }
- static void json_next_number_token(json_parse_t *json, json_token_t *token)
- {
- char *endptr;
- token->value.integer = strtoll(json->ptr, &endptr, 0);
- if (json->ptr == endptr) {
- json_set_token_error(token, json, "invalid number");
- return;
- }
- if (*endptr == '.' || *endptr == 'e' || *endptr == 'E') {
- token->type = T_NUMBER;
- token->value.number = fpconv_strtod(json->ptr, &endptr);
- } else {
- token->type = T_INTEGER;
- }
- json->ptr = endptr; /* Skip the processed number */
- return;
- }
- /* Fills in the token struct.
- * T_STRING will return a pointer to the json_parse_t temporary string
- * T_ERROR will leave the json->ptr pointer at the error.
- */
- static void json_next_token(json_parse_t *json, json_token_t *token)
- {
- const json_token_type_t *ch2token = json->cfg->ch2token;
- int ch;
- /* Eat whitespace. */
- while (1) {
- ch = (unsigned char)*(json->ptr);
- token->type = ch2token[ch];
- if (token->type != T_WHITESPACE)
- break;
- json->ptr++;
- }
- /* Store location of new token. Required when throwing errors
- * for unexpected tokens (syntax errors). */
- token->index = json->ptr - json->data;
- /* Don't advance the pointer for an error or the end */
- if (token->type == T_ERROR) {
- json_set_token_error(token, json, "invalid token");
- return;
- }
- if (token->type == T_END) {
- return;
- }
- /* Found a known single character token, advance index and return */
- if (token->type != T_UNKNOWN) {
- json->ptr++;
- return;
- }
- /* Process characters which triggered T_UNKNOWN
- *
- * Must use strncmp() to match the front of the JSON string.
- * JSON identifier must be lowercase.
- * When strict_numbers if disabled, either case is allowed for
- * Infinity/NaN (since we are no longer following the spec..) */
- if (ch == '"') {
- json_next_string_token(json, token);
- return;
- } else if (ch == '-' || ('0' <= ch && ch <= '9')) {
- if (!json->cfg->decode_invalid_numbers && json_is_invalid_number(json)) {
- json_set_token_error(token, json, "invalid number");
- return;
- }
- json_next_number_token(json, token);
- return;
- } else if (!strncmp(json->ptr, "true", 4)) {
- token->type = T_BOOLEAN;
- token->value.boolean = 1;
- json->ptr += 4;
- return;
- } else if (!strncmp(json->ptr, "false", 5)) {
- token->type = T_BOOLEAN;
- token->value.boolean = 0;
- json->ptr += 5;
- return;
- } else if (!strncmp(json->ptr, "null", 4)) {
- token->type = T_NULL;
- json->ptr += 4;
- return;
- } else if (json->cfg->decode_invalid_numbers &&
- json_is_invalid_number(json)) {
- /* When decode_invalid_numbers is enabled, only attempt to process
- * numbers we know are invalid JSON (Inf, NaN, hex)
- * This is required to generate an appropriate token error,
- * otherwise all bad tokens will register as "invalid number"
- */
- json_next_number_token(json, token);
- return;
- }
- /* Token starts with t/f/n but isn't recognised above. */
- json_set_token_error(token, json, "invalid token");
- }
- /* This function does not return.
- * DO NOT CALL WITH DYNAMIC MEMORY ALLOCATED.
- * The only supported exception is the temporary parser string
- * json->tmp struct.
- * json and token should exist on the stack somewhere.
- * luaL_error() will long_jmp and release the stack */
- static void json_throw_parse_error(lua_State *l, json_parse_t *json,
- const char *exp, json_token_t *token)
- {
- const char *found;
- strbuf_free(json->tmp);
- if (token->type == T_ERROR)
- found = token->value.string;
- else
- found = json_token_type_name[token->type];
- /* Note: token->index is 0 based, display starting from 1 */
- luaL_error(l, "Expected %s but found %s at character %d",
- exp, found, token->index + 1);
- }
- static inline void json_decode_ascend(json_parse_t *json)
- {
- json->current_depth--;
- }
- static void json_decode_descend(lua_State *l, json_parse_t *json, int slots)
- {
- json->current_depth++;
- if (json->current_depth <= json->cfg->decode_max_depth &&
- lua_checkstack(l, slots)) {
- return;
- }
- strbuf_free(json->tmp);
- luaL_error(l, "Found too many nested data structures (%d) at character %d",
- json->current_depth, json->ptr - json->data);
- }
- static void json_parse_object_context(lua_State *l, json_parse_t *json)
- {
- json_token_t token;
- /* 3 slots required:
- * .., table, key, value */
- json_decode_descend(l, json, 3);
- lua_newtable(l);
- json_next_token(json, &token);
- /* Handle empty objects */
- if (token.type == T_OBJ_END) {
- json_decode_ascend(json);
- return;
- }
- while (1) {
- if (token.type != T_STRING)
- json_throw_parse_error(l, json, "object key string", &token);
- /* Push key */
- lua_pushlstring(l, token.value.string, token.string_len);
- json_next_token(json, &token);
- if (token.type != T_COLON)
- json_throw_parse_error(l, json, "colon", &token);
- /* Fetch value */
- json_next_token(json, &token);
- json_process_value(l, json, &token);
- /* Set key = value */
- lua_rawset(l, -3);
- json_next_token(json, &token);
- if (token.type == T_OBJ_END) {
- json_decode_ascend(json);
- return;
- }
- if (token.type != T_COMMA)
- json_throw_parse_error(l, json, "comma or object end", &token);
- json_next_token(json, &token);
- }
- }
- /* Handle the array context */
- static void json_parse_array_context(lua_State *l, json_parse_t *json)
- {
- json_token_t token;
- int i;
- /* 2 slots required:
- * .., table, value */
- json_decode_descend(l, json, 2);
- lua_newtable(l);
- json_next_token(json, &token);
- /* Handle empty arrays */
- if (token.type == T_ARR_END) {
- json_decode_ascend(json);
- return;
- }
- for (i = 1; ; i++) {
- json_process_value(l, json, &token);
- lua_rawseti(l, -2, i); /* arr[i] = value */
- json_next_token(json, &token);
- if (token.type == T_ARR_END) {
- json_decode_ascend(json);
- return;
- }
- if (token.type != T_COMMA)
- json_throw_parse_error(l, json, "comma or array end", &token);
- json_next_token(json, &token);
- }
- }
- /* Handle the "value" context */
- static void json_process_value(lua_State *l, json_parse_t *json,
- json_token_t *token)
- {
- switch (token->type) {
- case T_STRING:
- lua_pushlstring(l, token->value.string, token->string_len);
- break;;
- case T_NUMBER:
- lua_pushnumber(l, token->value.number);
- break;;
- case T_INTEGER:
- lua_pushinteger(l, token->value.integer);
- break;;
- case T_BOOLEAN:
- lua_pushboolean(l, token->value.boolean);
- break;;
- case T_OBJ_BEGIN:
- json_parse_object_context(l, json);
- break;;
- case T_ARR_BEGIN:
- json_parse_array_context(l, json);
- break;;
- case T_NULL:
- /* In Lua, setting "t[k] = nil" will delete k from the table.
- * Hence a NULL pointer lightuserdata object is used instead */
- lua_pushlightuserdata(l, NULL);
- break;;
- default:
- json_throw_parse_error(l, json, "value", token);
- }
- }
- static int json_decode(lua_State *l)
- {
- json_parse_t json;
- json_token_t token;
- size_t json_len;
- luaL_argcheck(l, lua_gettop(l) == 1, 1, "expected 1 argument");
- json.cfg = json_fetch_config(l);
- json.data = luaL_checklstring(l, 1, &json_len);
- json.current_depth = 0;
- json.ptr = json.data;
- /* Detect Unicode other than UTF-8 (see RFC 4627, Sec 3)
- *
- * CJSON can support any simple data type, hence only the first
- * character is guaranteed to be ASCII (at worst: '"'). This is
- * still enough to detect whether the wrong encoding is in use. */
- if (json_len >= 2 && (!json.data[0] || !json.data[1]))
- luaL_error(l, "JSON parser does not support UTF-16 or UTF-32");
- /* Ensure the temporary buffer can hold the entire string.
- * This means we no longer need to do length checks since the decoded
- * string must be smaller than the entire json string */
- json.tmp = strbuf_new(json_len);
- json_next_token(&json, &token);
- json_process_value(l, &json, &token);
- /* Ensure there is no more input left */
- json_next_token(&json, &token);
- if (token.type != T_END)
- json_throw_parse_error(l, &json, "the end", &token);
- strbuf_free(json.tmp);
- return 1;
- }
- /* ===== INITIALISATION ===== */
- #if !defined(LUA_VERSION_NUM) || LUA_VERSION_NUM < 502
- /* Compatibility for Lua 5.1.
- *
- * luaL_setfuncs() is used to create a module table where the functions have
- * json_config_t as their first upvalue. Code borrowed from Lua 5.2 source. */
- static void luaL_setfuncs (lua_State *l, const luaL_Reg *reg, int nup)
- {
- int i;
- luaL_checkstack(l, nup, "too many upvalues");
- for (; reg->name != NULL; reg++) { /* fill the table with given functions */
- for (i = 0; i < nup; i++) /* copy upvalues to the top */
- lua_pushvalue(l, -nup);
- lua_pushcclosure(l, reg->func, nup); /* closure with those upvalues */
- lua_setfield(l, -(nup + 2), reg->name);
- }
- lua_pop(l, nup); /* remove upvalues */
- }
- #endif
- /* Call target function in protected mode with all supplied args.
- * Assumes target function only returns a single non-nil value.
- * Convert and return thrown errors as: nil, "error message" */
- static int json_protect_conversion(lua_State *l)
- {
- int err;
- /* Deliberately throw an error for invalid arguments */
- luaL_argcheck(l, lua_gettop(l) == 1, 1, "expected 1 argument");
- /* pcall() the function stored as upvalue(1) */
- lua_pushvalue(l, lua_upvalueindex(1));
- lua_insert(l, 1);
- err = lua_pcall(l, 1, 1, 0);
- if (!err)
- return 1;
- if (err == LUA_ERRRUN) {
- lua_pushnil(l);
- lua_insert(l, -2);
- return 2;
- }
- /* Since we are not using a custom error handler, the only remaining
- * errors are memory related */
- return luaL_error(l, "Memory allocation error in CJSON protected call");
- }
- /* Return cjson module table */
- static int lua_cjson_new(lua_State *l)
- {
- luaL_Reg reg[] = {
- { "encode", json_encode },
- { "decode", json_decode },
- { "encode_sparse_array", json_cfg_encode_sparse_array },
- { "encode_max_depth", json_cfg_encode_max_depth },
- { "decode_max_depth", json_cfg_decode_max_depth },
- { "encode_number_precision", json_cfg_encode_number_precision },
- { "encode_keep_buffer", json_cfg_encode_keep_buffer },
- { "encode_invalid_numbers", json_cfg_encode_invalid_numbers },
- { "decode_invalid_numbers", json_cfg_decode_invalid_numbers },
- { "new", lua_cjson_new },
- { NULL, NULL }
- };
- /* Initialise number conversions */
- fpconv_init();
- /* cjson module table */
- lua_newtable(l);
- /* Register functions with config data as upvalue */
- json_create_config(l);
- luaL_setfuncs(l, reg, 1);
- /* Set cjson.null */
- lua_pushlightuserdata(l, NULL);
- lua_setfield(l, -2, "null");
- /* Set module name / version fields */
- lua_pushliteral(l, CJSON_MODNAME);
- lua_setfield(l, -2, "_NAME");
- lua_pushliteral(l, CJSON_VERSION);
- lua_setfield(l, -2, "_VERSION");
- return 1;
- }
- /* Return cjson.safe module table */
- static int lua_cjson_safe_new(lua_State *l)
- {
- const char *func[] = { "decode", "encode", NULL };
- int i;
- lua_cjson_new(l);
- /* Fix new() method */
- lua_pushcfunction(l, lua_cjson_safe_new);
- lua_setfield(l, -2, "new");
- for (i = 0; func[i]; i++) {
- lua_getfield(l, -1, func[i]);
- lua_pushcclosure(l, json_protect_conversion, 1);
- lua_setfield(l, -2, func[i]);
- }
- return 1;
- }
- int luaopen_cjson(lua_State *l)
- {
- lua_cjson_new(l);
- #ifdef ENABLE_CJSON_GLOBAL
- /* Register a global "cjson" table. */
- lua_pushvalue(l, -1);
- lua_setglobal(l, CJSON_MODNAME);
- #endif
- /* Return cjson table */
- return 1;
- }
- int luaopen_cjson_safe(lua_State *l)
- {
- lua_cjson_safe_new(l);
- /* Return cjson.safe table */
- return 1;
- }
- /* vi:ai et sw=4 ts=4:
- */
|