radare2/shlr/java/dsojson.c

712 lines
20 KiB
C

/* radare - Apache - Copyright 2014,2015 dso, pancake */
#include "dsojson.h"
#include "class.h"
#include <r_types.h>
#include <r_util.h>
#include <stdint.h>
R_API char * dso_json_dict_entry_to_str (DsoJsonDictEntry * entry);
R_API char * dso_json_list_to_str (DsoJsonList *list);
R_API char * dso_json_dict_to_str (DsoJsonDict *list);
R_API char * dso_json_num_to_str (DsoJsonNum * num);
R_API char * dso_json_str_to_str (DsoJsonStr *str);
static int cmpDsoStr_to_str (DsoJsonStr *dsoStr1, char *dsoStr2);
static const DsoJsonInfo* get_type_info (unsigned int type);
static char * dso_json_get_str_data (DsoJsonObj *dso_obj);
static DsoJsonStr * dso_json_get_str (DsoJsonObj *dso_obj);
static DsoJsonInfo DSO_JSON_INFOS []= {
{DSO_JSON_NULL},
{DSO_JSON_NUM},//, dso_json_num_new, dso_json_free_num, dso_json_insert, dso_json_append, NULL, NULL, NULL, NULL},
{DSO_JSON_STR},//, dso_json_str_new, dso_json_str_new, dso_json_insert, dso_json_append, NULL, NULL, NULL, NULL},
{DSO_JSON_LIST},//, dso_json_list_new, dso_json_list_free, dso_json_insert, dso_json_list_append, NULL, NULL, NULL, NULL},
{DSO_JSON_DICT},//, dso_json_dict_new, dso_json_free_dict, dso_json_insert_dict, dso_json_append, NULL, NULL, NULL, NULL},
{DSO_JSON_DICT_ENTRY},//, dso_json_dict_entry_new, dso_json_dict_entry_free, dso_json_insert, dso_json_append, NULL, NULL, NULL, NULL},
{DSO_JSON_END},//, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL},
};
// TODO: remove useless calloc wrapper
static void * json_new0 (unsigned int sz) {
return calloc (sz, 1);
}
static RList * build_str_list_from_iterable (RList *the_list) {
RList * res = r_list_newf (free);
DsoJsonObj *json_obj;
RListIter *iter;
r_list_foreach (the_list, iter, json_obj) {
char *str = dso_json_obj_to_str (json_obj);
if (str && *str) {
r_list_append (res, str);
}
}
return res;
}
static char * build_str_from_str_list_for_iterable (RList *the_list, int is_array) {
char *res = NULL, *str;
int len = 3, pos = 1;
RListIter *iter = NULL;
RList *str_list = build_str_list_from_iterable (the_list);
r_list_foreach (str_list, iter, str) {
len += strlen (str) + 1;
}
res = json_new0 (len);
// TODO: use [ if needed
if (res) {
strcpy (res, is_array? "[": "{");
r_list_foreach (str_list, iter, str) {
pos += snprintf (res+pos, len - pos, "%s%s",
str, iter->n? ",": "");
}
strcat (res, is_array?"]": "}");
}
r_list_free (str_list);
return res;
}
static int get_type (DsoJsonObj *y) {
return (y && y->info)? y->info->type: -1;
}
static int dso_json_is_null (DsoJsonObj *y) {
return get_type (y) == DSO_JSON_NULL;
}
static int dso_json_is_list (DsoJsonObj *y) {
return get_type (y) == DSO_JSON_LIST;
}
static int dso_json_is_dict_entry (DsoJsonObj *y) {
return get_type (y) == DSO_JSON_DICT_ENTRY;
}
R_API ut8 dso_json_char_needs_hexing ( ut8 b) {
if (b < 0x20) {
return 1;
}
switch (b) {
case 0x7f:
case 0x81:
case 0x8F:
case 0x90:
case 0x9D:
case 0xA0:
case 0xAD:
return 1;
}
return 0;
}
R_API char * dso_json_obj_to_str (DsoJsonObj * dso_obj) {
if (dso_obj && dso_obj->info) {
switch (dso_obj->info->type) {
case DSO_JSON_NULL: return strdup ("null");
case DSO_JSON_NUM: return dso_json_num_to_str (dso_obj->val._num);
case DSO_JSON_STR: return dso_json_str_to_str (dso_obj->val._str);
case DSO_JSON_LIST: return dso_json_list_to_str (dso_obj->val._list);
case DSO_JSON_DICT: return dso_json_dict_to_str (dso_obj->val._dict);
case DSO_JSON_DICT_ENTRY: return dso_json_dict_entry_to_str (dso_obj->val._dict_entry);
}
}
return NULL;
}
R_API void dso_json_obj_del (DsoJsonObj *dso_obj) {
if (!dso_obj) return;
switch (dso_obj->info->type) {
case DSO_JSON_NULL: /*do nothing */ break;
case DSO_JSON_NUM: dso_json_num_free (dso_obj->val._num); break;
case DSO_JSON_STR: dso_json_str_free (dso_obj->val._str); break;
case DSO_JSON_LIST: dso_json_list_free (dso_obj); break; //->val._list); break;
case DSO_JSON_DICT: dso_json_dict_free (dso_obj->val._dict); break;
case DSO_JSON_DICT_ENTRY: dso_json_dict_entry_free (dso_obj->val._dict_entry); break;
default: break;
}
#if 0
//memset (dso_obj, 0, sizeof (DsoJsonObj));
free (dso_obj);
#endif
}
static const DsoJsonInfo* get_type_info (unsigned int type) {
unsigned int i = 0;
for (; DSO_JSON_INFOS[i].type != DSO_JSON_END; i++) {
if (DSO_JSON_INFOS[i].type == type) return &DSO_JSON_INFOS[i];
}
return NULL;
}
static int cmpDsoStr_to_str (DsoJsonStr *dsoStr1, char *dsoStr2) {
if (dsoStr1 && dsoStr1->data && dsoStr2)
return strcmp (dsoStr1->data, dsoStr2);
return -1;
}
static void allocDsoStr (DsoJsonStr *dsoStr, unsigned int sz) {
free (dsoStr->data);
if (sz > 0) dsoStr->data = json_new0 (sz);
else dsoStr->data = json_new0 (10);
dsoStr->len = sz;
}
static RList * dso_json_get_list (DsoJsonObj *dso_obj) {
RList *the_list = NULL;
if (dso_obj) {
switch (dso_obj->info->type) {
case DSO_JSON_LIST: the_list = dso_obj->val._list->json_list; break;
case DSO_JSON_DICT: the_list = dso_obj->val._dict->json_dict; break;
default: break;
}
}
return the_list;
}
static char * dso_json_get_str_data (DsoJsonObj *dso_obj) {
DsoJsonStr * str = dso_json_get_str (dso_obj);
if (str) return str->data;
return NULL;
}
static DsoJsonStr * dso_json_get_str (DsoJsonObj *dso_obj) {
DsoJsonStr * str = NULL;
if (dso_obj) {
switch (dso_obj->info->type) {
case DSO_JSON_STR: str = dso_obj->val._str; break;
case DSO_JSON_DICT_ENTRY: str = dso_json_get_str (dso_obj->val._dict_entry->key); break;
default: break;
}
}
return str;
}
R_API DsoJsonObj * dso_json_null_new () {
DsoJsonObj *x = json_new0 (sizeof (DsoJsonObj));
x->info = get_type_info (DSO_JSON_NULL);
return x;
}
R_API void dso_json_null_free (void *x) {
free (x);
}
R_API DsoJsonObj * dso_json_str_new () {
DsoJsonObj *x = dso_json_null_new ();
if (x) {
x->info = get_type_info (DSO_JSON_STR);
x->val._str = json_new0 (sizeof (DsoJsonStr));
}
return x;
}
R_API void dso_json_str_free (void *y) {
DsoJsonStr *x = (DsoJsonStr *)y;
if (x) {
free (x->data);
x->data = NULL;
free (x);
}
}
R_API DsoJsonObj * dso_json_dict_entry_new () {
DsoJsonObj *x = dso_json_null_new ();
x->info = get_type_info (DSO_JSON_DICT_ENTRY);
x->val._dict_entry = json_new0 (sizeof (DsoJsonDictEntry));
x->val._dict_entry->key = dso_json_str_new ();
x->val._dict_entry->value = dso_json_null_new ();
return x;
}
R_API DsoJsonObj * dso_json_dict_entry_new_from_key_obj_val_obj (DsoJsonObj *key, DsoJsonObj *value) {
DsoJsonObj *x = dso_json_dict_entry_new ();
dso_json_obj_del (x->val._dict_entry->key);
dso_json_obj_del (x->val._dict_entry->value);
x->val._dict_entry->key = key;
x->val._dict_entry->value = value;
return x;
}
R_API void dso_json_dict_entry_free (void *y) {
DsoJsonDictEntry *entry = (DsoJsonDictEntry *)y;
if (entry) {
dso_json_obj_del (entry->key);
dso_json_obj_del (entry->value);
entry->key = NULL;
entry->value = NULL;
}
free (entry);
}
R_API char * dso_json_dict_entry_to_str (DsoJsonDictEntry * entry) {
char *res = NULL;
if (entry) {
char *key = dso_json_obj_to_str (entry->key);
char *value = dso_json_obj_to_str (entry->value);
if (key) {
int len = 2 + 3 + strlen (key);
if (value) len += strlen (value);
res = json_new0 (len);
if (res) {
if (value) {
snprintf (res, len, "%s:%s", key, value);
} else {
snprintf (res, len, "%s:\"\"", key);
}
}
free (key);
}
free (value);
}
return res;
}
R_API int dso_json_dict_entry_set_key_str (DsoJsonObj * entry_obj, char *key) {
int res = false;
if (dso_json_is_dict_entry (entry_obj)) {
DsoJsonDictEntry *entry = (DsoJsonDictEntry *)entry_obj;
DsoJsonObj *o_key = dso_json_str_new_from_str (key);
if (entry->key) {
dso_json_obj_del (entry->key);
}
entry->key = o_key;
res = true;
}
return res;
}
R_API int dso_json_dict_entry_set_key_str_len (DsoJsonObj * entry_obj, char *key, unsigned int len) {
int res = false;
if (dso_json_is_dict_entry (entry_obj)) {
DsoJsonDictEntry *entry = (DsoJsonDictEntry *)entry_obj;
DsoJsonObj *o_key = dso_json_str_new_from_str_len (key, len);
if (entry->key) {
dso_json_obj_del (entry->key);
}
entry->key = o_key;
res = true;
}
return res;
}
R_API int dso_json_dict_entry_set_key_num (DsoJsonObj * entry_obj, st64 num) {
int res = false;
if (dso_json_is_dict_entry (entry_obj)) {
DsoJsonDictEntry *entry = (DsoJsonDictEntry *)entry_obj;
DsoJsonObj *o_key = dso_json_num_new_from_num (num);
if (entry->key) {
dso_json_obj_del (entry->key);
}
entry->key = o_key;
res = true;
}
return res;
}
R_API int dso_json_dict_entry_value_append_str (DsoJsonObj *entry_obj, char *str) {
int res = false;
if (dso_json_is_dict_entry (entry_obj)) {
DsoJsonObj *o_str = dso_json_str_new_from_str (str);
res = dso_json_dict_entry_value_append_obj (entry_obj, o_str);
if (!res) dso_json_obj_del (o_str);
}
return res;
}
R_API int dso_json_dict_entry_value_append_str_len (DsoJsonObj *entry_obj, char *str, unsigned int len) {
int res = false;
if (dso_json_is_dict_entry (entry_obj)) {
DsoJsonObj *o_str = dso_json_str_new_from_str_len (str, len);
res = dso_json_dict_entry_value_append_obj (entry_obj, o_str);
if (!res) dso_json_obj_del (o_str);
}
return res;
}
R_API int dso_json_dict_entry_value_append_num (DsoJsonObj *entry_obj, st64 num) {
int res = false;
if (dso_json_is_dict_entry (entry_obj)) {
DsoJsonObj *o_num = dso_json_num_new_from_num (num);
res = dso_json_dict_entry_value_append_obj (entry_obj, o_num);
if (!res) dso_json_obj_del (o_num);
}
return res;
}
R_API int dso_json_dict_entry_value_append_obj (DsoJsonObj *entry_obj, DsoJsonObj *obj) {
if (dso_json_is_dict_entry (entry_obj)) {
DsoJsonDictEntry *x = entry_obj->val._dict_entry;
// check to see if the object can be converted to a list
if (dso_json_is_null (x->value)) {
DsoJsonObj *new_list = dso_json_list_new ();
dso_json_obj_del (x->value);
x->value = new_list;
} else if (!dso_json_is_list (x->value)) {
DsoJsonObj *tmp = x->value;
x->value = dso_json_list_new ();
dso_json_list_append (x->value, tmp);
}
if (dso_json_is_list (x->value)) {
dso_json_list_append (x->value, obj);
return true;
}
}
return false;
}
R_API int dso_json_dict_entry_value_set_str (DsoJsonObj *entry_obj, char *str) {
int res = false;
if (dso_json_is_dict_entry (entry_obj)) {
DsoJsonObj *o_val = dso_json_str_new_from_str (str);
res = dso_json_dict_entry_value_set_obj (entry_obj, o_val);
}
return res;
}
R_API int dso_json_dict_entry_value_set_str_len (DsoJsonObj *entry_obj, char *str, unsigned int len) {
int res = false;
if (dso_json_is_dict_entry (entry_obj)) {
DsoJsonObj *o_val = dso_json_str_new_from_str_len (str, len);
res = dso_json_dict_entry_value_set_obj (entry_obj, o_val);
}
return res;
}
R_API int dso_json_dict_entry_value_set_num (DsoJsonObj *entry_obj, st64 num) {
int res = false;
if (dso_json_is_dict_entry (entry_obj)) {
DsoJsonObj *o_val = dso_json_num_new_from_num (num);
res = dso_json_dict_entry_value_set_obj (entry_obj, o_val);
}
return res;
}
R_API int dso_json_dict_entry_value_set_empty_dict (DsoJsonObj *entry_obj) {
int res = false;
if (dso_json_is_dict_entry (entry_obj)) {
DsoJsonObj *o_val = dso_json_dict_new ();
res = dso_json_dict_entry_value_set_obj (entry_obj, o_val);
}
return res;
}
R_API int dso_json_dict_entry_value_set_empty_list (DsoJsonObj *entry_obj) {
int res = false;
if (dso_json_is_dict_entry (entry_obj)) {
DsoJsonObj *o_val = dso_json_list_new ();
res = dso_json_dict_entry_value_set_obj (entry_obj, o_val);
}
return res;
}
R_API int dso_json_dict_entry_value_set_obj (DsoJsonObj *entry_obj, DsoJsonObj *obj) {
if (dso_json_is_dict_entry (entry_obj)) {
DsoJsonDictEntry *entry = entry_obj->val._dict_entry;
if (entry->value) dso_json_obj_del (entry->value);
entry->value = obj;
return true;
}
return false;
}
R_API DsoJsonObj * dso_json_list_new () {
DsoJsonObj *x = dso_json_null_new ();
if (x) {
x->info = get_type_info (DSO_JSON_LIST);
x->val._list = json_new0 (sizeof (DsoJsonList));
if (x->val._list) {
x->val._list->json_list = r_list_newf ((RListFree)dso_json_obj_del);
} else {
R_FREE (x);
}
}
return x;
}
R_API void dso_json_list_free (DsoJsonObj *x) {
if (!x) return;
if (x->val._list && x->val._list->json_list) {
r_list_free (x->val._list->json_list);
x->val._list->json_list = NULL;
}
}
R_API char * dso_json_list_to_str (DsoJsonList *list) {
if (list && list->json_list) {
return build_str_from_str_list_for_iterable (list->json_list, 1);
}
return strdup ("[]");
}
R_API int dso_json_list_append (DsoJsonObj *list_obj, DsoJsonObj *y) {
if (get_type (list_obj) == DSO_JSON_LIST) {
DsoJsonList * list = list_obj->val._list;
r_list_append (list->json_list, y);
return true;
}
return false;
}
R_API int dso_json_list_append_str (DsoJsonObj *list_obj, char *y) {
if (get_type (list_obj) == DSO_JSON_LIST) {
DsoJsonObj *val = dso_json_str_new_from_str (y);
int res = dso_json_list_append (list_obj, val);
if (!res) dso_json_obj_del (val);
return res;
}
return false;
}
R_API int dso_json_list_append_num (DsoJsonObj *list_obj, ut64 y) {
if (get_type (list_obj) == DSO_JSON_LIST) {
DsoJsonObj *val = dso_json_num_new_from_num (y);
int res = dso_json_list_append (list_obj, val);
if (!res) dso_json_obj_del (val);
return res;
}
return false;
}
R_API DsoJsonObj * dso_json_dict_new () {
DsoJsonObj *x = dso_json_null_new ();
if (x) {
x->info = get_type_info (DSO_JSON_DICT);
x->val._dict = json_new0 (sizeof (DsoJsonObj));
x->val._dict->json_dict = r_list_newf ((RListFree)dso_json_obj_del);
}
return x;
}
R_API void dso_json_dict_free (void *y) {
DsoJsonDict *x = (DsoJsonDict *)y;
if (x && x->json_dict) {
r_list_free (x->json_dict);
x->json_dict = NULL;
}
free (x);
}
R_API char * dso_json_dict_to_str (DsoJsonDict *dict) {
if (dict && dict->json_dict) {
return build_str_from_str_list_for_iterable (dict->json_dict, 0);
}
return strdup ("{}");
}
R_API int dso_json_dict_insert_str_key_obj (DsoJsonObj *dict, char *key, DsoJsonObj *val_obj) {
DsoJsonObj *key_obj = dso_json_str_new_from_str (key);
int res = dso_json_dict_insert_key_obj (dict, key_obj, val_obj);
if (!res) {
dso_json_obj_del (key_obj);
}
return res;
}
R_API int dso_json_dict_insert_str_key_num (DsoJsonObj *dict, char *key, int val) {
DsoJsonObj *key_obj = dso_json_str_new_from_str (key);
DsoJsonObj *val_obj = dso_json_num_new_from_num (val);
int res = dso_json_dict_insert_key_obj (dict, key_obj, val_obj);
if (!res) {
dso_json_obj_del (key_obj);
dso_json_obj_del (val_obj);
}
return res;
}
R_API int dso_json_dict_insert_str_key_str (DsoJsonObj *dict, char *key, char *val) {
DsoJsonObj *key_obj = dso_json_str_new_from_str (key),
*val_obj = dso_json_str_new_from_str (val);
int res = dso_json_dict_insert_key_obj (dict, key_obj, val_obj);
if (!res) {
dso_json_obj_del (key_obj);
dso_json_obj_del (val_obj);
}
return res;
}
// create keys from num value
R_API int dso_json_dict_insert_num_key_obj (DsoJsonObj *dict, int key, DsoJsonObj *val_obj) {
DsoJsonObj *key_obj = dso_json_str_new_from_num (key);
int res = dso_json_dict_insert_key_obj (dict, key_obj, val_obj);
if (!res) {
dso_json_obj_del (key_obj);
dso_json_obj_del (val_obj);
}
return res;
}
R_API int dso_json_dict_insert_num_key_num (DsoJsonObj *dict, int key, int val) {
DsoJsonObj *key_obj = dso_json_str_new_from_num (key),
*val_obj = dso_json_num_new_from_num (val);
int res = dso_json_dict_insert_key_obj (dict, key_obj, val_obj);
if (!res) {
dso_json_obj_del (key_obj);
dso_json_obj_del (val_obj);
}
return res;
}
R_API int dso_json_dict_insert_num_key_str (DsoJsonObj *dict, int key, char *val) {
DsoJsonObj *key_obj = dso_json_str_new_from_num (key),
*val_obj = dso_json_str_new_from_str (val);
int res = dso_json_dict_insert_key_obj (dict, key_obj, val_obj);
if (!res) {
dso_json_obj_del (key_obj);
dso_json_obj_del (val_obj);
}
return res;
}
// TODO inserting the dicts.
R_API int dso_json_dict_insert_key_obj (DsoJsonObj *dict, DsoJsonObj *key, DsoJsonObj *value) {
int res = false;
RList* the_list = dso_json_get_list (dict);
if (!the_list) return false;
if (get_type (key) != DSO_JSON_STR) return false;
if (!value) value = dso_json_null_new ();
if (value && key && !dso_json_dict_contains_key_obj (dict, key)) {
DsoJsonObj *entry = dso_json_dict_entry_new_from_key_obj_val_obj (key, value);
r_list_append (the_list, entry);
res = true;
//TODO implement the remove key
} else if (value && key && !dso_json_dict_remove_key_obj (dict, key)) {
DsoJsonObj *entry = dso_json_dict_entry_new_from_key_obj_val_obj (key, value);
r_list_append (the_list, entry);
res = true;
} else {
dso_json_dict_free (value);
}
return res;
}
R_API int dso_json_dict_remove_key_obj (DsoJsonObj *dict, DsoJsonObj *key) {
return dso_json_dict_remove_key_str (dict, dso_json_get_str_data (key));
}
R_API int dso_json_dict_remove_key_str (DsoJsonObj *dict, char *key) {
RListIter *iter;
DsoJsonObj *dso_obj;
int res = false;
RList* the_list = dso_json_get_list (dict);
if (the_list) {
r_list_foreach (the_list, iter, dso_obj) {
if (get_type (dso_obj) == DSO_JSON_DICT_ENTRY &&
get_type (dso_obj->val._dict_entry->key) == DSO_JSON_STR) {
if (!cmpDsoStr_to_str (dso_json_get_str (dso_obj), key)) {
res = true;
r_list_delete (the_list, iter);
break;
}
}
}
}
return res;
}
R_API int dso_json_dict_contains_key_obj (DsoJsonObj *dict, DsoJsonObj *key) {
return dso_json_dict_contains_key_str (dict, dso_json_get_str_data (key));
}
R_API int dso_json_dict_contains_key_str (DsoJsonObj *dict, char *key) {
RListIter *iter;
DsoJsonObj *dso_obj;
RList* the_list = dso_json_get_list (dict);
if (the_list) {
r_list_foreach (the_list, iter, dso_obj) {
if (get_type (dso_obj) == DSO_JSON_DICT_ENTRY &&
get_type (dso_obj->val._dict_entry->key) == DSO_JSON_STR) {
if (!cmpDsoStr_to_str (dso_json_get_str (dso_obj), key)) {
return true;
}
}
}
}
return false;
}
// TODO append value to key 1) check that key is valid, 2) if not create new entry and append it
R_API DsoJsonObj * dso_json_num_new () {
DsoJsonObj *x = dso_json_null_new ();
x->info = get_type_info (DSO_JSON_NUM);
x->val._num = json_new0 (sizeof (DsoJsonNum));
return x;
}
R_API void dso_json_num_free (void *y) {
DsoJsonNum *x = (DsoJsonNum *)y;
free (x);
}
R_API char * dso_json_num_to_str (DsoJsonNum * num) {
return r_str_newf ("%"PFMT64d, num->value);
}
R_API DsoJsonObj * dso_json_num_new_from_num (ut64 num) {
DsoJsonObj *x = dso_json_num_new ();
x->val._num->value = num;
return x;
}
R_API char * dso_json_convert_string (const char * bytes, ut32 len) {
ut32 idx = 0, pos = 1;
ut32 str_sz = 4*len+1+2;
char *cpy_buffer = len > 0 ? calloc (1, str_sz): NULL;
if (!cpy_buffer) return cpy_buffer;
// 4x is the increase from byte to \xHH where HH represents hexed byte
cpy_buffer[0] = '"';
while (idx < len) {
if (bytes[idx] == '"') {
sprintf (cpy_buffer+pos, "\\%c", bytes[idx]);
pos += 2;
} else if (dso_json_char_needs_hexing (bytes[idx])) {
sprintf (cpy_buffer+pos, "\\x%02x", bytes[idx]);
pos += 4;
} else {
cpy_buffer[pos] = bytes[idx];
pos++;
}
idx ++;
}
strcat (cpy_buffer, "\"");
return cpy_buffer;
}
R_API char * dso_json_str_to_str (DsoJsonStr *str) {
if (str && str->data && str->len> 0) {
return dso_json_convert_string (str->data, str->len);
}
return NULL;
}
R_API DsoJsonObj * dso_json_str_new_from_str (const char *str) {
DsoJsonObj *x = dso_json_str_new ();
DsoJsonStr * dsoStr = x->val._str;
allocDsoStr (dsoStr, strlen (str));
if (dsoStr->data) memcpy (dsoStr->data, str, dsoStr->len);
return x;
}
R_API DsoJsonObj * dso_json_str_new_from_str_len (const char *str, unsigned int len) {
DsoJsonObj *x = dso_json_str_new ();
DsoJsonStr * dsoStr = x->val._str;
allocDsoStr (dsoStr, len);
memcpy (dsoStr->data, str, dsoStr->len);
return x;
}
R_API DsoJsonObj * dso_json_str_new_from_num (long num) {
DsoJsonObj *x = dso_json_str_new ();
DsoJsonStr * dsoStr = x->val._str;
int len = snprintf (NULL, 0, "%lu", num);
allocDsoStr (dsoStr, len-1);
snprintf (dsoStr->data, dsoStr->len, "%lu", num);
return x;
}