cJSON.c 20.5 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 253 254
cJSON *cJSON_Parse(const char *value)
{
	cJSON *c=cJSON_New_Item();
D
Dave Gamble 已提交
255
	ep=0;
256 257 258 259 260 261
	if (!c) return 0;       /* memory fail */

	if (!parse_value(c,skip(value))) {cJSON_Delete(c);return 0;}
	return c;
}

262
/* Render a cJSON item/entity/structure to text. */
D
Dave Gamble 已提交
263 264
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 已提交
265

266
/* Parser core - when encountering text, process appropriately. */
D
Dave Gamble 已提交
267 268
static const char *parse_value(cJSON *item,const char *value)
{
269
	if (!value)						return 0;	/* Fail on null. */
D
Dave Gamble 已提交
270 271 272 273 274 275 276 277
	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); }

278
	ep=value;return 0;	/* failure. */
D
Dave Gamble 已提交
279 280
}

281
/* Render a value to text. */
D
Dave Gamble 已提交
282
static char *print_value(cJSON *item,int depth,int fmt)
D
Dave Gamble 已提交
283 284
{
	char *out=0;
D
Dave Gamble 已提交
285
	if (!item) return 0;
D
Dave Gamble 已提交
286
	switch ((item->type)&255)
D
Dave Gamble 已提交
287
	{
288 289 290
		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 已提交
291 292
		case cJSON_Number:	out=print_number(item);break;
		case cJSON_String:	out=print_string(item);break;
D
Dave Gamble 已提交
293 294
		case cJSON_Array:	out=print_array(item,depth,fmt);break;
		case cJSON_Object:	out=print_object(item,depth,fmt);break;
D
Dave Gamble 已提交
295 296 297 298
	}
	return out;
}

299
/* Build an array from input text. */
D
Dave Gamble 已提交
300 301 302
static const char *parse_array(cJSON *item,const char *value)
{
	cJSON *child;
303
	if (*value!='[')	{ep=value;return 0;}	/* not an array! */
D
Dave Gamble 已提交
304 305 306

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

	item->child=child=cJSON_New_Item();
310 311
	if (!item->child) return 0;		 /* memory fail */
	value=skip(parse_value(child,skip(value)));	/* skip any spacing, get the value. */
312
	if (!value) return 0;
D
Dave Gamble 已提交
313 314 315 316

	while (*value==',')
	{
		cJSON *new_item;
317
		if (!(new_item=cJSON_New_Item())) return 0; 	/* memory fail */
D
Dave Gamble 已提交
318 319
		child->next=new_item;new_item->prev=child;child=new_item;
		value=skip(parse_value(child,skip(value+1)));
320
		if (!value) return 0;	/* memory fail */
D
Dave Gamble 已提交
321 322
	}

323
	if (*value==']') return value+1;	/* end of array */
324
	ep=value;return 0;	/* malformed. */
D
Dave Gamble 已提交
325 326
}

327
/* Render an array to text */
D
Dave Gamble 已提交
328
static char *print_array(cJSON *item,int depth,int fmt)
D
Dave Gamble 已提交
329
{
330
	char **entries;
331
	char *out=0,*ptr,*ret;int len=5;
D
Dave Gamble 已提交
332
	cJSON *child=item->child;
333
	int numentries=0,i=0,fail=0;
D
Dave Gamble 已提交
334
	
335
	/* How many entries in the array? */
336
	while (child) numentries++,child=child->next;
337 338
	/* Explicitly handle numentries==0 */
	if (!numentries)
D
Dave Gamble 已提交
339
	{
340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355
		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 已提交
356
	}
357
	
358
	/* If we didn't fail, try to malloc the output string */
359
	if (!fail) out=(char*)cJSON_malloc(len);
360
	/* If that fails, we fail. */
361 362
	if (!out) fail=1;

363
	/* Handle failure. */
364 365 366
	if (fail)
	{
		for (i=0;i<numentries;i++) if (entries[i]) cJSON_free(entries[i]);
367
		cJSON_free(entries);
368 369 370
		return 0;
	}
	
371
	/* Compose the output array. */
372 373 374 375 376
	*out='[';
	ptr=out+1;*ptr=0;
	for (i=0;i<numentries;i++)
	{
		strcpy(ptr,entries[i]);ptr+=strlen(entries[i]);
D
Dave Gamble 已提交
377
		if (i!=numentries-1) {*ptr++=',';if(fmt)*ptr++=' ';*ptr=0;}
378 379
		cJSON_free(entries[i]);
	}
380
	cJSON_free(entries);
D
Dave Gamble 已提交
381 382 383 384
	*ptr++=']';*ptr++=0;
	return out;	
}

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

421
/* Render an object to text. */
D
Dave Gamble 已提交
422
static char *print_object(cJSON *item,int depth,int fmt)
D
Dave Gamble 已提交
423
{
424 425
	char **entries=0,**names=0;
	char *out=0,*ptr,*ret,*str;int len=7,i=0,j;
D
Dave Gamble 已提交
426
	cJSON *child=item->child;
427
	int numentries=0,fail=0;
428
	/* Count the number of entries. */
429
	while (child) numentries++,child=child->next;
430 431 432 433 434 435 436 437 438 439
	/* 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;
	}
440
	/* Allocate space for the names and the objects */
441 442 443 444 445 446 447
	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);

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

462
	/* Handle failure */
463 464
	if (fail)
	{
465 466
		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);
467 468 469
		return 0;
	}
	
470
	/* Compose the output: */
D
Dave Gamble 已提交
471
	*out='{';ptr=out+1;if (fmt)*ptr++='\n';*ptr=0;
472 473
	for (i=0;i<numentries;i++)
	{
D
Dave Gamble 已提交
474
		if (fmt) for (j=0;j<depth;j++) *ptr++='\t';
475
		strcpy(ptr,names[i]);ptr+=strlen(names[i]);
D
Dave Gamble 已提交
476
		*ptr++=':';if (fmt) *ptr++='\t';
477 478
		strcpy(ptr,entries[i]);ptr+=strlen(entries[i]);
		if (i!=numentries-1) *ptr++=',';
D
Dave Gamble 已提交
479
		if (fmt) *ptr++='\n';*ptr=0;
480
		cJSON_free(names[i]);cJSON_free(entries[i]);
D
Dave Gamble 已提交
481
	}
482 483
	
	cJSON_free(names);cJSON_free(entries);
D
Dave Gamble 已提交
484
	if (fmt) for (i=0;i<depth-1;i++) *ptr++='\t';
D
Dave Gamble 已提交
485 486 487 488
	*ptr++='}';*ptr++=0;
	return out;	
}

489
/* Get Array size/item / object item. */
D
Dave Gamble 已提交
490
int    cJSON_GetArraySize(cJSON *array)							{cJSON *c=array->child;int i=0;while(c)i++,c=c->next;return i;}
491
cJSON *cJSON_GetArrayItem(cJSON *array,int item)				{cJSON *c=array->child;  while (c && item>0) item--,c=c->next; return c;}
492
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 已提交
493

494
/* Utility for array list handling. */
D
Dave Gamble 已提交
495
static void suffix_object(cJSON *prev,cJSON *item) {prev->next=item;item->prev=prev;}
496
/* Utility for handling references. */
497
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 已提交
498

499
/* Add item to array/object. */
500
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);}}
501
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 已提交
502 503 504
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 已提交
505
cJSON *cJSON_DetachItemFromArray(cJSON *array,int which)			{cJSON *c=array->child;while (c && which>0) c=c->next,which--;if (!c) return 0;
506
	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;}
507
void   cJSON_DeleteItemFromArray(cJSON *array,int which)			{cJSON_Delete(cJSON_DetachItemFromArray(array,which));}
508
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;}
509
void   cJSON_DeleteItemFromObject(cJSON *object,const char *string) {cJSON_Delete(cJSON_DetachItemFromObject(object,string));}
D
Dave Gamble 已提交
510

511
/* Replace array/object items with new ones. */
D
Dave Gamble 已提交
512 513
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;
514
	if (c==array->child) array->child=newitem; else newitem->prev->next=newitem;c->next=c->prev=0;cJSON_Delete(c);}
515
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 已提交
516

517
/* Create basic types: */
D
Dave Gamble 已提交
518 519 520
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 已提交
521
cJSON *cJSON_CreateBool(int b)					{cJSON *item=cJSON_New_Item();if(item)item->type=b?cJSON_True:cJSON_False;return item;}
522 523
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 已提交
524 525
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 已提交
526

527
/* Create Arrays: */
528 529 530 531
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;}
532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558

/* 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;
	if (item->valuestring)	newitem->valuestring=cJSON_strdup(item->valuestring);
	if (item->string)		newitem->string=cJSON_strdup(item->string);
	/* 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 */
		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;
}