cJSON.c 21.2 KB
Newer Older
D
Dave Gamble 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
/*
  Copyright (c) 2009 Dave Gamble

  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.
*/

23 24
/* cJSON */
/* JSON parser in C. */
D
Dave Gamble 已提交
25 26 27 28 29 30

#include <string.h>
#include <stdio.h>
#include <math.h>
#include <stdlib.h>
#include <float.h>
31
#include <limits.h>
D
Dave Gamble 已提交
32
#include <ctype.h>
D
Dave Gamble 已提交
33 34
#include "cJSON.h"

35 36
static const char *ep;

D
Dave Gamble 已提交
37
const char *cJSON_GetErrorPtr(void) {return ep;}
38

39 40
static int cJSON_strcasecmp(const char *s1,const char *s2)
{
41
	if (!s1) return (s1==s2)?0:1;if (!s2) return 1;
42 43 44
	for(; tolower(*s1) == tolower(*s2); ++s1, ++s2)	if(*s1 == 0)	return 0;
	return tolower(*(const unsigned char *)s1) - tolower(*(const unsigned char *)s2);
}
45

46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71
static void *(*cJSON_malloc)(size_t sz) = malloc;
static void (*cJSON_free)(void *ptr) = free;

static char* cJSON_strdup(const char* str)
{
      size_t len;
      char* copy;

      len = strlen(str) + 1;
      if (!(copy = (char*)cJSON_malloc(len))) return 0;
      memcpy(copy,str,len);
      return copy;
}

void cJSON_InitHooks(cJSON_Hooks* hooks)
{
    if (!hooks) { /* Reset hooks */
        cJSON_malloc = malloc;
        cJSON_free = free;
        return;
    }

	cJSON_malloc = (hooks->malloc_fn)?hooks->malloc_fn:malloc;
	cJSON_free	 = (hooks->free_fn)?hooks->free_fn:free;
}

72
/* Internal constructor. */
D
Dave Gamble 已提交
73
static cJSON *cJSON_New_Item(void)
74 75 76 77 78
{
	cJSON* node = (cJSON*)cJSON_malloc(sizeof(cJSON));
	if (node) memset(node,0,sizeof(cJSON));
	return node;
}
D
Dave Gamble 已提交
79

80
/* Delete a cJSON structure. */
D
Dave Gamble 已提交
81 82 83 84 85 86
void cJSON_Delete(cJSON *c)
{
	cJSON *next;
	while (c)
	{
		next=c->next;
D
Dave Gamble 已提交
87 88
		if (!(c->type&cJSON_IsReference) && c->child) cJSON_Delete(c->child);
		if (!(c->type&cJSON_IsReference) && c->valuestring) cJSON_free(c->valuestring);
89 90
		if (c->string) cJSON_free(c->string);
		cJSON_free(c);
D
Dave Gamble 已提交
91 92 93 94
		c=next;
	}
}

95
/* Parse the input text to generate a number, and populate the result into item. */
D
Dave Gamble 已提交
96 97 98 99
static const char *parse_number(cJSON *item,const char *num)
{
	double n=0,sign=1,scale=0;int subscale=0,signsubscale=1;

100 101 102 103
	/* Could use sscanf for this? */
	if (*num=='-') sign=-1,num++;	/* Has sign? */
	if (*num=='0') num++;			/* is zero */
	if (*num>='1' && *num<='9')	do	n=(n*10.0)+(*num++ -'0');	while (*num>='0' && *num<='9');	/* Number? */
D
Dave Gamble 已提交
104
	if (*num=='.' && num[1]>='0' && num[1]<='9') {num++;		do	n=(n*10.0)+(*num++ -'0'),scale--; while (*num>='0' && *num<='9');}	/* Fractional part? */
105 106 107
	if (*num=='e' || *num=='E')		/* Exponent? */
	{	num++;if (*num=='+') num++;	else if (*num=='-') signsubscale=-1,num++;		/* With sign? */
		while (*num>='0' && *num<='9') subscale=(subscale*10)+(*num++ - '0');	/* Number? */
D
Dave Gamble 已提交
108 109
	}

110
	n=sign*n*pow(10.0,(scale+subscale*signsubscale));	/* number = +/- number.fraction * 10^+/- exponent */
D
Dave Gamble 已提交
111 112 113 114 115 116 117
	
	item->valuedouble=n;
	item->valueint=(int)n;
	item->type=cJSON_Number;
	return num;
}

118
/* Render the number nicely from the given item into a string. */
D
Dave Gamble 已提交
119 120 121 122
static char *print_number(cJSON *item)
{
	char *str;
	double d=item->valuedouble;
123
	if (fabs(((double)item->valueint)-d)<=DBL_EPSILON && d<=INT_MAX && d>=INT_MIN)
D
Dave Gamble 已提交
124
	{
125
		str=(char*)cJSON_malloc(21);	/* 2^64+1 can be represented in 21 chars. */
126
		if (str) sprintf(str,"%d",item->valueint);
D
Dave Gamble 已提交
127 128 129
	}
	else
	{
130
		str=(char*)cJSON_malloc(64);	/* This is a nice tradeoff. */
131 132
		if (str)
		{
133 134 135
			if (fabs(floor(d)-d)<=DBL_EPSILON && fabs(d)<1.0e60)sprintf(str,"%.0f",d);
			else if (fabs(d)<1.0e-6 || fabs(d)>1.0e9)			sprintf(str,"%e",d);
			else												sprintf(str,"%f",d);
136
		}
D
Dave Gamble 已提交
137 138 139 140
	}
	return str;
}

141
/* Parse the input text into an unescaped cstring, and populate item. */
142
static const unsigned char firstByteMark[7] = { 0x00, 0x00, 0xC0, 0xE0, 0xF0, 0xF8, 0xFC };
D
Dave Gamble 已提交
143 144
static const char *parse_string(cJSON *item,const char *str)
{
145
	const char *ptr=str+1;char *ptr2;char *out;int len=0;unsigned uc,uc2;
146
	if (*str!='\"') {ep=str;return 0;}	/* not a string! */
D
Dave Gamble 已提交
147
	
D
Dave Gamble 已提交
148
	while (*ptr!='\"' && *ptr && ++len) if (*ptr++ == '\\') ptr++;	/* Skip escaped quotes. */
D
Dave Gamble 已提交
149
	
150
	out=(char*)cJSON_malloc(len+1);	/* This is how long we need for the string, roughly. */
151
	if (!out) return 0;
D
Dave Gamble 已提交
152 153
	
	ptr=str+1;ptr2=out;
D
Dave Gamble 已提交
154
	while (*ptr!='\"' && *ptr)
D
Dave Gamble 已提交
155 156 157 158 159 160 161 162 163 164 165 166
	{
		if (*ptr!='\\') *ptr2++=*ptr++;
		else
		{
			ptr++;
			switch (*ptr)
			{
				case 'b': *ptr2++='\b';	break;
				case 'f': *ptr2++='\f';	break;
				case 'n': *ptr2++='\n';	break;
				case 'r': *ptr2++='\r';	break;
				case 't': *ptr2++='\t';	break;
167 168 169
				case 'u':	 /* transcode utf16 to utf8. */
					sscanf(ptr+1,"%4x",&uc);ptr+=4;	/* get the unicode char. */

D
Dave Gamble 已提交
170
					if ((uc>=0xDC00 && uc<=0xDFFF) || uc==0)	break;	/* check for invalid.	*/
171

D
Dave Gamble 已提交
172
					if (uc>=0xD800 && uc<=0xDBFF)	/* UTF16 surrogate pairs.	*/
173
					{
D
Dave Gamble 已提交
174
						if (ptr[1]!='\\' || ptr[2]!='u')	break;	/* missing second-half of surrogate.	*/
175
						sscanf(ptr+3,"%4x",&uc2);ptr+=6;
D
Dave Gamble 已提交
176
						if (uc2<0xDC00 || uc2>0xDFFF)		break;	/* invalid second-half of surrogate.	*/
177
						uc=0x10000 + (((uc&0x3FF)<<10) | (uc2&0x3FF));
178 179 180
					}

					len=4;if (uc<0x80) len=1;else if (uc<0x800) len=2;else if (uc<0x10000) len=3; ptr2+=len;
D
Dave Gamble 已提交
181 182
					
					switch (len) {
183
						case 4: *--ptr2 =((uc | 0x80) & 0xBF); uc >>= 6;
D
Dave Gamble 已提交
184 185 186 187
						case 3: *--ptr2 =((uc | 0x80) & 0xBF); uc >>= 6;
						case 2: *--ptr2 =((uc | 0x80) & 0xBF); uc >>= 6;
						case 1: *--ptr2 =(uc | firstByteMark[len]);
					}
188
					ptr2+=len;
D
Dave Gamble 已提交
189 190 191 192 193 194 195 196 197 198 199 200 201
					break;
				default:  *ptr2++=*ptr; break;
			}
			ptr++;
		}
	}
	*ptr2=0;
	if (*ptr=='\"') ptr++;
	item->valuestring=out;
	item->type=cJSON_String;
	return ptr;
}

202
/* Render the cstring provided to an escaped version that can be printed. */
D
Dave Gamble 已提交
203 204
static char *print_string_ptr(const char *str)
{
D
Dave Gamble 已提交
205
	const char *ptr;char *ptr2,*out;int len=0;unsigned char token;
D
Dave Gamble 已提交
206
	
207
	if (!str) return cJSON_strdup("");
D
Dave Gamble 已提交
208
	ptr=str;while ((token=*ptr) && ++len) {if (strchr("\"\\\b\f\n\r\t",token)) len++; else if (token<32) len+=5;ptr++;}
D
Dave Gamble 已提交
209
	
210
	out=(char*)cJSON_malloc(len+3);
211 212
	if (!out) return 0;

D
Dave Gamble 已提交
213 214 215 216
	ptr2=out;ptr=str;
	*ptr2++='\"';
	while (*ptr)
	{
217
		if ((unsigned char)*ptr>31 && *ptr!='\"' && *ptr!='\\') *ptr2++=*ptr++;
D
Dave Gamble 已提交
218 219 220
		else
		{
			*ptr2++='\\';
D
Dave Gamble 已提交
221
			switch (token=*ptr++)
D
Dave Gamble 已提交
222 223 224 225 226 227 228 229
			{
				case '\\':	*ptr2++='\\';	break;
				case '\"':	*ptr2++='\"';	break;
				case '\b':	*ptr2++='b';	break;
				case '\f':	*ptr2++='f';	break;
				case '\n':	*ptr2++='n';	break;
				case '\r':	*ptr2++='r';	break;
				case '\t':	*ptr2++='t';	break;
D
Dave Gamble 已提交
230
				default: sprintf(ptr2,"u%04x",token);ptr2+=5;	break;	/* escape and print */
D
Dave Gamble 已提交
231 232 233 234 235 236
			}
		}
	}
	*ptr2++='\"';*ptr2++=0;
	return out;
}
237
/* Invote print_string_ptr (which is useful) on an item. */
D
Dave Gamble 已提交
238 239
static char *print_string(cJSON *item)	{return print_string_ptr(item->valuestring);}

240
/* Predeclare these prototypes. */
D
Dave Gamble 已提交
241
static const char *parse_value(cJSON *item,const char *value);
D
Dave Gamble 已提交
242
static char *print_value(cJSON *item,int depth,int fmt);
D
Dave Gamble 已提交
243
static const char *parse_array(cJSON *item,const char *value);
D
Dave Gamble 已提交
244
static char *print_array(cJSON *item,int depth,int fmt);
D
Dave Gamble 已提交
245
static const char *parse_object(cJSON *item,const char *value);
D
Dave Gamble 已提交
246
static char *print_object(cJSON *item,int depth,int fmt);
D
Dave Gamble 已提交
247

248
/* Utility to jump whitespace and cr/lf */
D
Dave Gamble 已提交
249
static const char *skip(const char *in) {while (in && *in && (unsigned char)*in<=32) in++; return in;}
250

251
/* Parse an object - create a new root, and populate. */
252
cJSON *cJSON_ParseWithOpts(const char *value,const char **return_parse_end,int require_null_terminated)
253
{
254
	const char *end=0;
255
	cJSON *c=cJSON_New_Item();
D
Dave Gamble 已提交
256
	ep=0;
257 258
	if (!c) return 0;       /* memory fail */

259 260 261 262 263 264
	end=parse_value(c,skip(value));
	if (!end)	{cJSON_Delete(c);return 0;}	/* parse failure. ep is set. */

	/* if we require null-terminated JSON without appended garbage, skip and then check for a null terminator */
	if (require_null_terminated) {end=skip(end);if (*end) {cJSON_Delete(c);ep=end;return 0;}}
	if (return_parse_end) *return_parse_end=end;
265 266
	return c;
}
267 268
/* Default options for cJSON_Parse */
cJSON *cJSON_Parse(const char *value) {return cJSON_ParseWithOpts(value,0,0);}
269

270
/* Render a cJSON item/entity/structure to text. */
D
Dave Gamble 已提交
271 272
char *cJSON_Print(cJSON *item)				{return print_value(item,0,1);}
char *cJSON_PrintUnformatted(cJSON *item)	{return print_value(item,0,0);}
D
Dave Gamble 已提交
273

274
/* Parser core - when encountering text, process appropriately. */
D
Dave Gamble 已提交
275 276
static const char *parse_value(cJSON *item,const char *value)
{
277
	if (!value)						return 0;	/* Fail on null. */
D
Dave Gamble 已提交
278 279 280 281 282 283 284 285
	if (!strncmp(value,"null",4))	{ item->type=cJSON_NULL;  return value+4; }
	if (!strncmp(value,"false",5))	{ item->type=cJSON_False; return value+5; }
	if (!strncmp(value,"true",4))	{ item->type=cJSON_True; item->valueint=1;	return value+4; }
	if (*value=='\"')				{ return parse_string(item,value); }
	if (*value=='-' || (*value>='0' && *value<='9'))	{ return parse_number(item,value); }
	if (*value=='[')				{ return parse_array(item,value); }
	if (*value=='{')				{ return parse_object(item,value); }

286
	ep=value;return 0;	/* failure. */
D
Dave Gamble 已提交
287 288
}

289
/* Render a value to text. */
D
Dave Gamble 已提交
290
static char *print_value(cJSON *item,int depth,int fmt)
D
Dave Gamble 已提交
291 292
{
	char *out=0;
D
Dave Gamble 已提交
293
	if (!item) return 0;
D
Dave Gamble 已提交
294
	switch ((item->type)&255)
D
Dave Gamble 已提交
295
	{
296 297 298
		case cJSON_NULL:	out=cJSON_strdup("null");	break;
		case cJSON_False:	out=cJSON_strdup("false");break;
		case cJSON_True:	out=cJSON_strdup("true"); break;
D
Dave Gamble 已提交
299 300
		case cJSON_Number:	out=print_number(item);break;
		case cJSON_String:	out=print_string(item);break;
D
Dave Gamble 已提交
301 302
		case cJSON_Array:	out=print_array(item,depth,fmt);break;
		case cJSON_Object:	out=print_object(item,depth,fmt);break;
D
Dave Gamble 已提交
303 304 305 306
	}
	return out;
}

307
/* Build an array from input text. */
D
Dave Gamble 已提交
308 309 310
static const char *parse_array(cJSON *item,const char *value)
{
	cJSON *child;
311
	if (*value!='[')	{ep=value;return 0;}	/* not an array! */
D
Dave Gamble 已提交
312 313 314

	item->type=cJSON_Array;
	value=skip(value+1);
315
	if (*value==']') return value+1;	/* empty array. */
D
Dave Gamble 已提交
316 317

	item->child=child=cJSON_New_Item();
318 319
	if (!item->child) return 0;		 /* memory fail */
	value=skip(parse_value(child,skip(value)));	/* skip any spacing, get the value. */
320
	if (!value) return 0;
D
Dave Gamble 已提交
321 322 323 324

	while (*value==',')
	{
		cJSON *new_item;
325
		if (!(new_item=cJSON_New_Item())) return 0; 	/* memory fail */
D
Dave Gamble 已提交
326 327
		child->next=new_item;new_item->prev=child;child=new_item;
		value=skip(parse_value(child,skip(value+1)));
328
		if (!value) return 0;	/* memory fail */
D
Dave Gamble 已提交
329 330
	}

331
	if (*value==']') return value+1;	/* end of array */
332
	ep=value;return 0;	/* malformed. */
D
Dave Gamble 已提交
333 334
}

335
/* Render an array to text */
D
Dave Gamble 已提交
336
static char *print_array(cJSON *item,int depth,int fmt)
D
Dave Gamble 已提交
337
{
338
	char **entries;
339
	char *out=0,*ptr,*ret;int len=5;
D
Dave Gamble 已提交
340
	cJSON *child=item->child;
341
	int numentries=0,i=0,fail=0;
D
Dave Gamble 已提交
342
	
343
	/* How many entries in the array? */
344
	while (child) numentries++,child=child->next;
345 346
	/* Explicitly handle numentries==0 */
	if (!numentries)
D
Dave Gamble 已提交
347
	{
348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363
		out=(char*)cJSON_malloc(3);
		if (out) strcpy(out,"[]");
		return out;
	}
	/* Allocate an array to hold the values for each */
	entries=(char**)cJSON_malloc(numentries*sizeof(char*));
	if (!entries) return 0;
	memset(entries,0,numentries*sizeof(char*));
	/* Retrieve all the results: */
	child=item->child;
	while (child && !fail)
	{
		ret=print_value(child,depth+1,fmt);
		entries[i++]=ret;
		if (ret) len+=strlen(ret)+2+(fmt?1:0); else fail=1;
		child=child->next;
D
Dave Gamble 已提交
364
	}
365
	
366
	/* If we didn't fail, try to malloc the output string */
367
	if (!fail) out=(char*)cJSON_malloc(len);
368
	/* If that fails, we fail. */
369 370
	if (!out) fail=1;

371
	/* Handle failure. */
372 373 374
	if (fail)
	{
		for (i=0;i<numentries;i++) if (entries[i]) cJSON_free(entries[i]);
375
		cJSON_free(entries);
376 377 378
		return 0;
	}
	
379
	/* Compose the output array. */
380 381 382 383 384
	*out='[';
	ptr=out+1;*ptr=0;
	for (i=0;i<numentries;i++)
	{
		strcpy(ptr,entries[i]);ptr+=strlen(entries[i]);
D
Dave Gamble 已提交
385
		if (i!=numentries-1) {*ptr++=',';if(fmt)*ptr++=' ';*ptr=0;}
386 387
		cJSON_free(entries[i]);
	}
388
	cJSON_free(entries);
D
Dave Gamble 已提交
389 390 391 392
	*ptr++=']';*ptr++=0;
	return out;	
}

393
/* Build an object from the text. */
D
Dave Gamble 已提交
394 395 396
static const char *parse_object(cJSON *item,const char *value)
{
	cJSON *child;
397
	if (*value!='{')	{ep=value;return 0;}	/* not an object! */
D
Dave Gamble 已提交
398 399 400
	
	item->type=cJSON_Object;
	value=skip(value+1);
401
	if (*value=='}') return value+1;	/* empty array. */
D
Dave Gamble 已提交
402 403
	
	item->child=child=cJSON_New_Item();
404
	if (!item->child) return 0;
D
Dave Gamble 已提交
405
	value=skip(parse_string(child,skip(value)));
406
	if (!value) return 0;
D
Dave Gamble 已提交
407
	child->string=child->valuestring;child->valuestring=0;
408
	if (*value!=':') {ep=value;return 0;}	/* fail! */
409
	value=skip(parse_value(child,skip(value+1)));	/* skip any spacing, get the value. */
410
	if (!value) return 0;
D
Dave Gamble 已提交
411 412 413 414
	
	while (*value==',')
	{
		cJSON *new_item;
415
		if (!(new_item=cJSON_New_Item()))	return 0; /* memory fail */
D
Dave Gamble 已提交
416 417
		child->next=new_item;new_item->prev=child;child=new_item;
		value=skip(parse_string(child,skip(value+1)));
418
		if (!value) return 0;
D
Dave Gamble 已提交
419
		child->string=child->valuestring;child->valuestring=0;
420
		if (*value!=':') {ep=value;return 0;}	/* fail! */
421
		value=skip(parse_value(child,skip(value+1)));	/* skip any spacing, get the value. */
422
		if (!value) return 0;
D
Dave Gamble 已提交
423 424
	}
	
425
	if (*value=='}') return value+1;	/* end of array */
426
	ep=value;return 0;	/* malformed. */
D
Dave Gamble 已提交
427 428
}

429
/* Render an object to text. */
D
Dave Gamble 已提交
430
static char *print_object(cJSON *item,int depth,int fmt)
D
Dave Gamble 已提交
431
{
432 433
	char **entries=0,**names=0;
	char *out=0,*ptr,*ret,*str;int len=7,i=0,j;
D
Dave Gamble 已提交
434
	cJSON *child=item->child;
435
	int numentries=0,fail=0;
436
	/* Count the number of entries. */
437
	while (child) numentries++,child=child->next;
438 439 440 441 442 443 444 445 446 447
	/* Explicitly handle empty object case */
	if (!numentries)
	{
		out=cJSON_malloc(fmt?depth+3:3);
		if (!out)	return 0;
		ptr=out;*ptr++='{';
		if (fmt) {*ptr++='\n';for (i=0;i<depth-1;i++) *ptr++='\t';}
		*ptr++='}';*ptr++=0;
		return out;
	}
448
	/* Allocate space for the names and the objects */
449 450 451 452 453 454 455
	entries=(char**)cJSON_malloc(numentries*sizeof(char*));
	if (!entries) return 0;
	names=(char**)cJSON_malloc(numentries*sizeof(char*));
	if (!names) {cJSON_free(entries);return 0;}
	memset(entries,0,sizeof(char*)*numentries);
	memset(names,0,sizeof(char*)*numentries);

456
	/* Collect all the results into our arrays: */
D
Dave Gamble 已提交
457
	child=item->child;depth++;if (fmt) len+=depth;
D
Dave Gamble 已提交
458 459
	while (child)
	{
460
		names[i]=str=print_string_ptr(child->string);
D
Dave Gamble 已提交
461 462
		entries[i++]=ret=print_value(child,depth,fmt);
		if (str && ret) len+=strlen(ret)+strlen(str)+2+(fmt?2+depth:0); else fail=1;
463 464 465
		child=child->next;
	}
	
466
	/* Try to allocate the output string */
467 468 469
	if (!fail) out=(char*)cJSON_malloc(len);
	if (!out) fail=1;

470
	/* Handle failure */
471 472
	if (fail)
	{
473 474
		for (i=0;i<numentries;i++) {if (names[i]) cJSON_free(names[i]);if (entries[i]) cJSON_free(entries[i]);}
		cJSON_free(names);cJSON_free(entries);
475 476 477
		return 0;
	}
	
478
	/* Compose the output: */
D
Dave Gamble 已提交
479
	*out='{';ptr=out+1;if (fmt)*ptr++='\n';*ptr=0;
480 481
	for (i=0;i<numentries;i++)
	{
D
Dave Gamble 已提交
482
		if (fmt) for (j=0;j<depth;j++) *ptr++='\t';
483
		strcpy(ptr,names[i]);ptr+=strlen(names[i]);
D
Dave Gamble 已提交
484
		*ptr++=':';if (fmt) *ptr++='\t';
485 486
		strcpy(ptr,entries[i]);ptr+=strlen(entries[i]);
		if (i!=numentries-1) *ptr++=',';
D
Dave Gamble 已提交
487
		if (fmt) *ptr++='\n';*ptr=0;
488
		cJSON_free(names[i]);cJSON_free(entries[i]);
D
Dave Gamble 已提交
489
	}
490 491
	
	cJSON_free(names);cJSON_free(entries);
D
Dave Gamble 已提交
492
	if (fmt) for (i=0;i<depth-1;i++) *ptr++='\t';
D
Dave Gamble 已提交
493 494 495 496
	*ptr++='}';*ptr++=0;
	return out;	
}

497
/* Get Array size/item / object item. */
D
Dave Gamble 已提交
498
int    cJSON_GetArraySize(cJSON *array)							{cJSON *c=array->child;int i=0;while(c)i++,c=c->next;return i;}
499
cJSON *cJSON_GetArrayItem(cJSON *array,int item)				{cJSON *c=array->child;  while (c && item>0) item--,c=c->next; return c;}
500
cJSON *cJSON_GetObjectItem(cJSON *object,const char *string)	{cJSON *c=object->child; while (c && cJSON_strcasecmp(c->string,string)) c=c->next; return c;}
D
Dave Gamble 已提交
501

502
/* Utility for array list handling. */
D
Dave Gamble 已提交
503
static void suffix_object(cJSON *prev,cJSON *item) {prev->next=item;item->prev=prev;}
504
/* Utility for handling references. */
505
static cJSON *create_reference(cJSON *item) {cJSON *ref=cJSON_New_Item();if (!ref) return 0;memcpy(ref,item,sizeof(cJSON));ref->string=0;ref->type|=cJSON_IsReference;ref->next=ref->prev=0;return ref;}
D
Dave Gamble 已提交
506

507
/* Add item to array/object. */
508
void   cJSON_AddItemToArray(cJSON *array, cJSON *item)						{cJSON *c=array->child;if (!item) return; if (!c) {array->child=item;} else {while (c && c->next) c=c->next; suffix_object(c,item);}}
509
void   cJSON_AddItemToObject(cJSON *object,const char *string,cJSON *item)	{if (!item) return; if (item->string) cJSON_free(item->string);item->string=cJSON_strdup(string);cJSON_AddItemToArray(object,item);}
D
Dave Gamble 已提交
510 511 512
void	cJSON_AddItemReferenceToArray(cJSON *array, cJSON *item)						{cJSON_AddItemToArray(array,create_reference(item));}
void	cJSON_AddItemReferenceToObject(cJSON *object,const char *string,cJSON *item)	{cJSON_AddItemToObject(object,string,create_reference(item));}

D
Dave Gamble 已提交
513
cJSON *cJSON_DetachItemFromArray(cJSON *array,int which)			{cJSON *c=array->child;while (c && which>0) c=c->next,which--;if (!c) return 0;
514
	if (c->prev) c->prev->next=c->next;if (c->next) c->next->prev=c->prev;if (c==array->child) array->child=c->next;c->prev=c->next=0;return c;}
515
void   cJSON_DeleteItemFromArray(cJSON *array,int which)			{cJSON_Delete(cJSON_DetachItemFromArray(array,which));}
516
cJSON *cJSON_DetachItemFromObject(cJSON *object,const char *string) {int i=0;cJSON *c=object->child;while (c && cJSON_strcasecmp(c->string,string)) i++,c=c->next;if (c) return cJSON_DetachItemFromArray(object,i);return 0;}
517
void   cJSON_DeleteItemFromObject(cJSON *object,const char *string) {cJSON_Delete(cJSON_DetachItemFromObject(object,string));}
D
Dave Gamble 已提交
518

519
/* Replace array/object items with new ones. */
D
Dave Gamble 已提交
520 521
void   cJSON_ReplaceItemInArray(cJSON *array,int which,cJSON *newitem)		{cJSON *c=array->child;while (c && which>0) c=c->next,which--;if (!c) return;
	newitem->next=c->next;newitem->prev=c->prev;if (newitem->next) newitem->next->prev=newitem;
522
	if (c==array->child) array->child=newitem; else newitem->prev->next=newitem;c->next=c->prev=0;cJSON_Delete(c);}
523
void   cJSON_ReplaceItemInObject(cJSON *object,const char *string,cJSON *newitem){int i=0;cJSON *c=object->child;while(c && cJSON_strcasecmp(c->string,string))i++,c=c->next;if(c){newitem->string=cJSON_strdup(string);cJSON_ReplaceItemInArray(object,i,newitem);}}
D
Dave Gamble 已提交
524

525
/* Create basic types: */
D
Dave Gamble 已提交
526 527 528
cJSON *cJSON_CreateNull(void)					{cJSON *item=cJSON_New_Item();if(item)item->type=cJSON_NULL;return item;}
cJSON *cJSON_CreateTrue(void)					{cJSON *item=cJSON_New_Item();if(item)item->type=cJSON_True;return item;}
cJSON *cJSON_CreateFalse(void)					{cJSON *item=cJSON_New_Item();if(item)item->type=cJSON_False;return item;}
D
Dave Gamble 已提交
529
cJSON *cJSON_CreateBool(int b)					{cJSON *item=cJSON_New_Item();if(item)item->type=b?cJSON_True:cJSON_False;return item;}
530 531
cJSON *cJSON_CreateNumber(double num)			{cJSON *item=cJSON_New_Item();if(item){item->type=cJSON_Number;item->valuedouble=num;item->valueint=(int)num;}return item;}
cJSON *cJSON_CreateString(const char *string)	{cJSON *item=cJSON_New_Item();if(item){item->type=cJSON_String;item->valuestring=cJSON_strdup(string);}return item;}
D
Dave Gamble 已提交
532 533
cJSON *cJSON_CreateArray(void)					{cJSON *item=cJSON_New_Item();if(item)item->type=cJSON_Array;return item;}
cJSON *cJSON_CreateObject(void)					{cJSON *item=cJSON_New_Item();if(item)item->type=cJSON_Object;return item;}
D
Dave Gamble 已提交
534

535
/* Create Arrays: */
536 537 538 539
cJSON *cJSON_CreateIntArray(int *numbers,int count)				{int i;cJSON *n=0,*p=0,*a=cJSON_CreateArray();for(i=0;a && i<count;i++){n=cJSON_CreateNumber(numbers[i]);if(!i)a->child=n;else suffix_object(p,n);p=n;}return a;}
cJSON *cJSON_CreateFloatArray(float *numbers,int count)			{int i;cJSON *n=0,*p=0,*a=cJSON_CreateArray();for(i=0;a && i<count;i++){n=cJSON_CreateNumber(numbers[i]);if(!i)a->child=n;else suffix_object(p,n);p=n;}return a;}
cJSON *cJSON_CreateDoubleArray(double *numbers,int count)		{int i;cJSON *n=0,*p=0,*a=cJSON_CreateArray();for(i=0;a && i<count;i++){n=cJSON_CreateNumber(numbers[i]);if(!i)a->child=n;else suffix_object(p,n);p=n;}return a;}
cJSON *cJSON_CreateStringArray(const char **strings,int count)	{int i;cJSON *n=0,*p=0,*a=cJSON_CreateArray();for(i=0;a && i<count;i++){n=cJSON_CreateString(strings[i]);if(!i)a->child=n;else suffix_object(p,n);p=n;}return a;}
540 541 542 543 544 545 546 547 548 549 550 551

/* Duplication */
cJSON *cJSON_Duplicate(cJSON *item,int recurse)
{
	cJSON *newitem,*cptr,*nptr=0,*newchild;
	/* Bail on bad ptr */
	if (!item) return 0;
	/* Create new item */
	newitem=cJSON_New_Item();
	if (!newitem) return 0;
	/* Copy over all vars */
	newitem->type=item->type&(~cJSON_IsReference),newitem->valueint=item->valueint,newitem->valuedouble=item->valuedouble;
552 553
	if (item->valuestring)	{newitem->valuestring=cJSON_strdup(item->valuestring);	if (!newitem->valuestring)	{cJSON_Delete(newitem);return 0;}}
	if (item->string)		{newitem->string=cJSON_strdup(item->string);			if (!newitem->string)		{cJSON_Delete(newitem);return 0;}}
554 555 556 557 558 559 560
	/* If non-recursive, then we're done! */
	if (!recurse) return newitem;
	/* Walk the ->next chain for the child. */
	cptr=item->child;
	while (cptr)
	{
		newchild=cJSON_Duplicate(cptr,1);		/* Duplicate (with recurse) each item in the ->next chain */
561
		if (!newchild) {cJSON_Delete(newitem);return 0;}
562 563 564 565 566 567
		if (nptr)	{nptr->next=newchild,newchild->prev=nptr;nptr=newchild;}	/* If newitem->child already set, then crosswire ->prev and ->next and move on */
		else		{newitem->child=newchild;nptr=newchild;}					/* Set newitem->child and move to it */
		cptr=cptr->next;
	}
	return newitem;
}