Coverage Report

Created: 2025-06-13 06:35

/src/json-c/json_tokener.c
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * $Id: json_tokener.c,v 1.20 2006/07/25 03:24:50 mclark Exp $
3
 *
4
 * Copyright (c) 2004, 2005 Metaparadigm Pte. Ltd.
5
 * Michael Clark <michael@metaparadigm.com>
6
 *
7
 * This library is free software; you can redistribute it and/or modify
8
 * it under the terms of the MIT license. See COPYING for details.
9
 *
10
 *
11
 * Copyright (c) 2008-2009 Yahoo! Inc.  All rights reserved.
12
 * The copyrights to the contents of this file are licensed under the MIT License
13
 * (https://www.opensource.org/licenses/mit-license.php)
14
 */
15
16
#include "config.h"
17
18
#include "math_compat.h"
19
#include <assert.h>
20
#include <errno.h>
21
#include <limits.h>
22
#include <math.h>
23
#include <stddef.h>
24
#include <stdio.h>
25
#include <stdlib.h>
26
#include <string.h>
27
28
#include "debug.h"
29
#include "json_inttypes.h"
30
#include "json_object.h"
31
#include "json_object_private.h"
32
#include "json_tokener.h"
33
#include "json_util.h"
34
#include "printbuf.h"
35
#include "strdup_compat.h"
36
37
#ifdef HAVE_LOCALE_H
38
#include <locale.h>
39
#endif /* HAVE_LOCALE_H */
40
#ifdef HAVE_XLOCALE_H
41
#include <xlocale.h>
42
#endif
43
#ifdef HAVE_STRINGS_H
44
#include <strings.h>
45
#endif /* HAVE_STRINGS_H */
46
47
0
#define jt_hexdigit(x) (((x) <= '9') ? (x) - '0' : ((x)&7) + 9)
48
49
#if !HAVE_STRNCASECMP && defined(_MSC_VER)
50
/* MSC has the version as _strnicmp */
51
#define strncasecmp _strnicmp
52
#elif !HAVE_STRNCASECMP
53
#error You do not have strncasecmp on your system.
54
#endif /* HAVE_STRNCASECMP */
55
56
#if defined(_MSC_VER) && (_MSC_VER <= 1800)
57
/* VS2013 doesn't know about "inline" */
58
#define inline __inline
59
#elif defined(AIX_CC)
60
#define inline
61
#endif
62
63
/* The following helper functions are used to speed up parsing. They
64
 * are faster than their ctype counterparts because they assume that
65
 * the input is in ASCII and that the locale is set to "C". The
66
 * compiler will also inline these functions, providing an additional
67
 * speedup by saving on function calls.
68
 */
69
static inline int is_ws_char(char c)
70
0
{
71
0
  return c == ' '
72
0
      || c == '\t'
73
0
      || c == '\n'
74
0
      || c == '\r';
75
0
}
76
77
static inline int is_hex_char(char c)
78
0
{
79
0
  return (c >= '0' && c <= '9')
80
0
      || (c >= 'A' && c <= 'F')
81
0
      || (c >= 'a' && c <= 'f');
82
0
}
83
84
/* Use C99 NAN by default; if not available, nan("") should work too. */
85
#ifndef NAN
86
#define NAN nan("")
87
#endif /* !NAN */
88
89
static const char json_null_str[] = "null";
90
static const int json_null_str_len = sizeof(json_null_str) - 1;
91
static const char json_inf_str[] = "Infinity";
92
/* Swapped case "Infinity" to avoid need to call tolower() on input chars: */
93
static const char json_inf_str_invert[] = "iNFINITY";
94
static const unsigned int json_inf_str_len = sizeof(json_inf_str) - 1;
95
static const char json_nan_str[] = "NaN";
96
static const int json_nan_str_len = sizeof(json_nan_str) - 1;
97
static const char json_true_str[] = "true";
98
static const int json_true_str_len = sizeof(json_true_str) - 1;
99
static const char json_false_str[] = "false";
100
static const int json_false_str_len = sizeof(json_false_str) - 1;
101
102
/* clang-format off */
103
static const char *json_tokener_errors[] = {
104
  "success",
105
  "continue",
106
  "nesting too deep",
107
  "unexpected end of data",
108
  "unexpected character",
109
  "null expected",
110
  "boolean expected",
111
  "number expected",
112
  "array value separator ',' expected",
113
  "quoted object property name expected",
114
  "object property name separator ':' expected",
115
  "object value separator ',' expected",
116
  "invalid string sequence",
117
  "expected comment",
118
  "invalid utf-8 string",
119
  "buffer size overflow",
120
  "out of memory"
121
};
122
/* clang-format on */
123
124
/**
125
 * validete the utf-8 string in strict model.
126
 * if not utf-8 format, return err.
127
 */
128
static json_bool json_tokener_validate_utf8(const char c, unsigned int *nBytes);
129
130
static int json_tokener_parse_double(const char *buf, int len, double *retval);
131
132
const char *json_tokener_error_desc(enum json_tokener_error jerr)
133
0
{
134
0
  int jerr_int = (int)jerr;
135
0
  if (jerr_int < 0 ||
136
0
      jerr_int >= (int)(sizeof(json_tokener_errors) / sizeof(json_tokener_errors[0])))
137
0
    return "Unknown error, "
138
0
           "invalid json_tokener_error value passed to json_tokener_error_desc()";
139
0
  return json_tokener_errors[jerr];
140
0
}
141
142
enum json_tokener_error json_tokener_get_error(struct json_tokener *tok)
143
0
{
144
0
  return tok->err;
145
0
}
146
147
/* Stuff for decoding unicode sequences */
148
0
#define IS_HIGH_SURROGATE(uc) (((uc)&0xFC00) == 0xD800)
149
0
#define IS_LOW_SURROGATE(uc) (((uc)&0xFC00) == 0xDC00)
150
0
#define DECODE_SURROGATE_PAIR(hi, lo) ((((hi)&0x3FF) << 10) + ((lo)&0x3FF) + 0x10000)
151
static unsigned char utf8_replacement_char[3] = {0xEF, 0xBF, 0xBD};
152
153
struct json_tokener *json_tokener_new_ex(int depth)
154
0
{
155
0
  struct json_tokener *tok;
156
157
0
  if (depth < 1)
158
0
    return NULL;
159
160
0
  tok = (struct json_tokener *)calloc(1, sizeof(struct json_tokener));
161
0
  if (!tok)
162
0
    return NULL;
163
0
  tok->stack = (struct json_tokener_srec *)calloc(depth, sizeof(struct json_tokener_srec));
164
0
  if (!tok->stack)
165
0
  {
166
0
    free(tok);
167
0
    return NULL;
168
0
  }
169
0
  tok->pb = printbuf_new();
170
0
  if (!tok->pb)
171
0
  {
172
0
    free(tok->stack);
173
0
    free(tok);
174
0
    return NULL;
175
0
  }
176
0
  tok->max_depth = depth;
177
0
  json_tokener_reset(tok);
178
0
  return tok;
179
0
}
180
181
struct json_tokener *json_tokener_new(void)
182
0
{
183
0
  return json_tokener_new_ex(JSON_TOKENER_DEFAULT_DEPTH);
184
0
}
185
186
void json_tokener_free(struct json_tokener *tok)
187
0
{
188
0
  if (!tok)
189
0
    return;
190
0
  json_tokener_reset(tok);
191
0
  if (tok->pb)
192
0
    printbuf_free(tok->pb);
193
0
  free(tok->stack);
194
0
  free(tok);
195
0
}
196
197
static void json_tokener_reset_level(struct json_tokener *tok, int depth)
198
0
{
199
0
  tok->stack[depth].state = json_tokener_state_eatws;
200
0
  tok->stack[depth].saved_state = json_tokener_state_start;
201
0
  json_object_put(tok->stack[depth].current);
202
0
  tok->stack[depth].current = NULL;
203
0
  free(tok->stack[depth].obj_field_name);
204
0
  tok->stack[depth].obj_field_name = NULL;
205
0
}
206
207
void json_tokener_reset(struct json_tokener *tok)
208
0
{
209
0
  int i;
210
0
  if (!tok)
211
0
    return;
212
213
0
  for (i = tok->depth; i >= 0; i--)
214
0
    json_tokener_reset_level(tok, i);
215
0
  tok->depth = 0;
216
0
  tok->err = json_tokener_success;
217
0
}
218
219
struct json_object *json_tokener_parse(const char *str)
220
0
{
221
0
  enum json_tokener_error jerr_ignored;
222
0
  struct json_object *obj;
223
0
  obj = json_tokener_parse_verbose(str, &jerr_ignored);
224
0
  return obj;
225
0
}
226
227
struct json_object *json_tokener_parse_verbose(const char *str, enum json_tokener_error *error)
228
0
{
229
0
  struct json_tokener *tok;
230
0
  struct json_object *obj;
231
232
0
  tok = json_tokener_new();
233
0
  if (!tok)
234
0
  {
235
0
    *error = json_tokener_error_memory;
236
0
    return NULL;
237
0
  }
238
0
  obj = json_tokener_parse_ex(tok, str, -1);
239
0
  *error = tok->err;
240
0
  if (tok->err != json_tokener_success
241
#if 0
242
    /* This would be a more sensible default, and cause parsing
243
     * things like "null123" to fail when the caller can't know
244
     * where the parsing left off, but starting to fail would
245
     * be a notable behaviour change.  Save for a 1.0 release.
246
     */
247
      || json_tokener_get_parse_end(tok) != strlen(str)
248
#endif
249
0
  )
250
251
0
  {
252
0
    if (obj != NULL)
253
0
      json_object_put(obj);
254
0
    obj = NULL;
255
0
  }
256
257
0
  json_tokener_free(tok);
258
0
  return obj;
259
0
}
260
261
0
#define state tok->stack[tok->depth].state
262
0
#define saved_state tok->stack[tok->depth].saved_state
263
0
#define current tok->stack[tok->depth].current
264
0
#define obj_field_name tok->stack[tok->depth].obj_field_name
265
266
/* Optimization:
267
 * json_tokener_parse_ex() consumed a lot of CPU in its main loop,
268
 * iterating character-by character.  A large performance boost is
269
 * achieved by using tighter loops to locally handle units such as
270
 * comments and strings.  Loops that handle an entire token within
271
 * their scope also gather entire strings and pass them to
272
 * printbuf_memappend() in a single call, rather than calling
273
 * printbuf_memappend() one char at a time.
274
 *
275
 * PEEK_CHAR() and ADVANCE_CHAR() macros are used for code that is
276
 * common to both the main loop and the tighter loops.
277
 */
278
279
/* PEEK_CHAR(dest, tok) macro:
280
 *   Peeks at the current char and stores it in dest.
281
 *   Returns 1 on success, sets tok->err and returns 0 if no more chars.
282
 *   Implicit inputs:  str, len, nBytesp vars
283
 */
284
#define PEEK_CHAR(dest, tok)                                                 \
285
0
  (((tok)->char_offset == len)                                         \
286
0
       ? (((tok)->depth == 0 && state == json_tokener_state_eatws &&   \
287
0
           saved_state == json_tokener_state_finish)                   \
288
0
              ? (((tok)->err = json_tokener_success), 0)               \
289
0
              : (((tok)->err = json_tokener_continue), 0))             \
290
0
       : (((tok->flags & JSON_TOKENER_VALIDATE_UTF8) &&                \
291
0
           (!json_tokener_validate_utf8(*str, nBytesp)))               \
292
0
              ? ((tok->err = json_tokener_error_parse_utf8_string), 0) \
293
0
              : (((dest) = *str), 1)))
294
295
/* ADVANCE_CHAR() macro:
296
 *   Increments str & tok->char_offset.
297
 *   For convenience of existing conditionals, returns the old value of c (0 on eof).
298
 *   Implicit inputs:  c var
299
 */
300
0
#define ADVANCE_CHAR(str, tok) (++(str), ((tok)->char_offset)++, c)
301
302
/* printbuf_memappend_checked(p, s, l) macro:
303
 *   Add string s of length l to printbuffer p.
304
 *   If operation fails abort parse operation with memory error.
305
 */
306
#define printbuf_memappend_checked(p, s, l)                   \
307
0
  do {                                                  \
308
0
    if (printbuf_memappend((p), (s), (l)) < 0)    \
309
0
    {                                             \
310
0
      tok->err = json_tokener_error_memory; \
311
0
      goto out;                             \
312
0
    }                                             \
313
0
  } while (0)
314
315
/* End optimization macro defs */
316
317
struct json_object *json_tokener_parse_ex(struct json_tokener *tok, const char *str, int len)
318
0
{
319
0
  struct json_object *obj = NULL;
320
0
  char c = '\1';
321
0
  unsigned int nBytes = 0;
322
0
  unsigned int *nBytesp = &nBytes;
323
324
0
#ifdef HAVE_USELOCALE
325
0
  locale_t oldlocale = uselocale(NULL);
326
0
  locale_t newloc;
327
#elif defined(HAVE_SETLOCALE)
328
  char *oldlocale = NULL;
329
#endif
330
331
0
  tok->char_offset = 0;
332
0
  tok->err = json_tokener_success;
333
334
  /* this interface is presently not 64-bit clean due to the int len argument
335
   * and the internal printbuf interface that takes 32-bit int len arguments
336
   * so the function limits the maximum string size to INT32_MAX (2GB).
337
   * If the function is called with len == -1 then strlen is called to check
338
   * the string length is less than INT32_MAX (2GB)
339
   */
340
0
  if ((len < -1) || (len == -1 && strlen(str) > INT32_MAX))
341
0
  {
342
0
    tok->err = json_tokener_error_size;
343
0
    return NULL;
344
0
  }
345
346
0
#ifdef HAVE_USELOCALE
347
0
  {
348
0
#ifdef HAVE_DUPLOCALE
349
0
    locale_t duploc = duplocale(oldlocale);
350
0
    if (duploc == NULL && errno == ENOMEM)
351
0
    {
352
0
      tok->err = json_tokener_error_memory;
353
0
      return NULL;
354
0
    }
355
0
    newloc = newlocale(LC_NUMERIC_MASK, "C", duploc);
356
#else
357
    newloc = newlocale(LC_NUMERIC_MASK, "C", oldlocale);
358
#endif
359
0
    if (newloc == NULL)
360
0
    {
361
0
      tok->err = json_tokener_error_memory;
362
0
#ifdef HAVE_DUPLOCALE
363
0
      freelocale(duploc);
364
0
#endif
365
0
      return NULL;
366
0
    }
367
#ifdef NEWLOCALE_NEEDS_FREELOCALE
368
#ifdef HAVE_DUPLOCALE
369
    // Older versions of FreeBSD (<12.4) don't free the locale
370
    // passed to newlocale(), so do it here
371
    freelocale(duploc);
372
#endif
373
#endif
374
0
    uselocale(newloc);
375
0
  }
376
#elif defined(HAVE_SETLOCALE)
377
  {
378
    char *tmplocale;
379
    tmplocale = setlocale(LC_NUMERIC, NULL);
380
    if (tmplocale)
381
    {
382
      oldlocale = strdup(tmplocale);
383
      if (oldlocale == NULL)
384
      {
385
        tok->err = json_tokener_error_memory;
386
        return NULL;
387
      }
388
    }
389
    setlocale(LC_NUMERIC, "C");
390
  }
391
#endif
392
393
0
  while (PEEK_CHAR(c, tok)) // Note: c might be '\0' !
394
0
  {
395
396
0
  redo_char:
397
0
    switch (state)
398
0
    {
399
400
0
    case json_tokener_state_eatws:
401
      /* Advance until we change state */
402
0
      while (is_ws_char(c))
403
0
      {
404
0
        if ((!ADVANCE_CHAR(str, tok)) || (!PEEK_CHAR(c, tok)))
405
0
          goto out;
406
0
      }
407
0
      if (c == '/' && !(tok->flags & JSON_TOKENER_STRICT))
408
0
      {
409
0
        printbuf_reset(tok->pb);
410
0
        printbuf_memappend_checked(tok->pb, &c, 1);
411
0
        state = json_tokener_state_comment_start;
412
0
      }
413
0
      else
414
0
      {
415
0
        state = saved_state;
416
0
        goto redo_char;
417
0
      }
418
0
      break;
419
420
0
    case json_tokener_state_start:
421
0
      switch (c)
422
0
      {
423
0
      case '{':
424
0
        state = json_tokener_state_eatws;
425
0
        saved_state = json_tokener_state_object_field_start;
426
0
        current = json_object_new_object();
427
0
        if (current == NULL)
428
0
        {
429
0
          tok->err = json_tokener_error_memory;
430
0
          goto out;
431
0
        }
432
0
        break;
433
0
      case '[':
434
0
        state = json_tokener_state_eatws;
435
0
        saved_state = json_tokener_state_array;
436
0
        current = json_object_new_array();
437
0
        if (current == NULL)
438
0
        {
439
0
          tok->err = json_tokener_error_memory;
440
0
          goto out;
441
0
        }
442
0
        break;
443
0
      case 'I':
444
0
      case 'i':
445
0
        state = json_tokener_state_inf;
446
0
        printbuf_reset(tok->pb);
447
0
        tok->st_pos = 0;
448
0
        goto redo_char;
449
0
      case 'N':
450
0
      case 'n':
451
0
        state = json_tokener_state_null; // or NaN
452
0
        printbuf_reset(tok->pb);
453
0
        tok->st_pos = 0;
454
0
        goto redo_char;
455
0
      case '\'':
456
0
        if (tok->flags & JSON_TOKENER_STRICT)
457
0
        {
458
          /* in STRICT mode only double-quote are allowed */
459
0
          tok->err = json_tokener_error_parse_unexpected;
460
0
          goto out;
461
0
        }
462
        /* FALLTHRU */
463
0
      case '"':
464
0
        state = json_tokener_state_string;
465
0
        printbuf_reset(tok->pb);
466
0
        tok->quote_char = c;
467
0
        break;
468
0
      case 'T':
469
0
      case 't':
470
0
      case 'F':
471
0
      case 'f':
472
0
        state = json_tokener_state_boolean;
473
0
        printbuf_reset(tok->pb);
474
0
        tok->st_pos = 0;
475
0
        goto redo_char;
476
0
      case '0':
477
0
      case '1':
478
0
      case '2':
479
0
      case '3':
480
0
      case '4':
481
0
      case '5':
482
0
      case '6':
483
0
      case '7':
484
0
      case '8':
485
0
      case '9':
486
0
      case '-':
487
0
        state = json_tokener_state_number;
488
0
        printbuf_reset(tok->pb);
489
0
        tok->is_double = 0;
490
0
        goto redo_char;
491
0
      default: tok->err = json_tokener_error_parse_unexpected; goto out;
492
0
      }
493
0
      break;
494
495
0
    case json_tokener_state_finish:
496
0
      if (tok->depth == 0)
497
0
        goto out;
498
0
      obj = json_object_get(current);
499
0
      json_tokener_reset_level(tok, tok->depth);
500
0
      tok->depth--;
501
0
      goto redo_char;
502
503
0
    case json_tokener_state_inf: /* aka starts with 'i' (or 'I', or "-i", or "-I") */
504
0
    {
505
      /* If we were guaranteed to have len set, then we could (usually) handle
506
       * the entire "Infinity" check in a single strncmp (strncasecmp), but
507
       * since len might be -1 (i.e. "read until \0"), we need to check it
508
       * a character at a time.
509
       * Trying to handle it both ways would make this code considerably more
510
       * complicated with likely little performance benefit.
511
       */
512
0
      int is_negative = 0;
513
514
      /* Note: tok->st_pos must be 0 when state is set to json_tokener_state_inf */
515
0
      while (tok->st_pos < (int)json_inf_str_len)
516
0
      {
517
0
        char inf_char = *str;
518
0
        if (inf_char != json_inf_str[tok->st_pos] &&
519
0
            ((tok->flags & JSON_TOKENER_STRICT) ||
520
0
              inf_char != json_inf_str_invert[tok->st_pos])
521
0
           )
522
0
        {
523
0
          tok->err = json_tokener_error_parse_unexpected;
524
0
          goto out;
525
0
        }
526
0
        tok->st_pos++;
527
0
        (void)ADVANCE_CHAR(str, tok);
528
0
        if (!PEEK_CHAR(c, tok))
529
0
        {
530
          /* out of input chars, for now at least */
531
0
          goto out;
532
0
        }
533
0
      }
534
      /* We checked the full length of "Infinity", so create the object.
535
       * When handling -Infinity, the number parsing code will have dropped
536
       * the "-" into tok->pb for us, so check it now.
537
       */
538
0
      if (printbuf_length(tok->pb) > 0 && *(tok->pb->buf) == '-')
539
0
      {
540
0
        is_negative = 1;
541
0
      }
542
0
      current = json_object_new_double(is_negative ? -INFINITY : INFINITY);
543
0
      if (current == NULL)
544
0
      {
545
0
        tok->err = json_tokener_error_memory;
546
0
        goto out;
547
0
      }
548
0
      saved_state = json_tokener_state_finish;
549
0
      state = json_tokener_state_eatws;
550
0
      goto redo_char;
551
0
    }
552
0
    break;
553
0
    case json_tokener_state_null: /* aka starts with 'n' */
554
0
    {
555
0
      int size;
556
0
      int size_nan;
557
0
      printbuf_memappend_checked(tok->pb, &c, 1);
558
0
      size = json_min(tok->st_pos + 1, json_null_str_len);
559
0
      size_nan = json_min(tok->st_pos + 1, json_nan_str_len);
560
0
      if ((!(tok->flags & JSON_TOKENER_STRICT) &&
561
0
           strncasecmp(json_null_str, tok->pb->buf, size) == 0) ||
562
0
          (strncmp(json_null_str, tok->pb->buf, size) == 0))
563
0
      {
564
0
        if (tok->st_pos == json_null_str_len)
565
0
        {
566
0
          current = NULL;
567
0
          saved_state = json_tokener_state_finish;
568
0
          state = json_tokener_state_eatws;
569
0
          goto redo_char;
570
0
        }
571
0
      }
572
0
      else if ((!(tok->flags & JSON_TOKENER_STRICT) &&
573
0
                strncasecmp(json_nan_str, tok->pb->buf, size_nan) == 0) ||
574
0
               (strncmp(json_nan_str, tok->pb->buf, size_nan) == 0))
575
0
      {
576
0
        if (tok->st_pos == json_nan_str_len)
577
0
        {
578
0
          current = json_object_new_double(NAN);
579
0
          if (current == NULL)
580
0
          {
581
0
            tok->err = json_tokener_error_memory;
582
0
            goto out;
583
0
          }
584
0
          saved_state = json_tokener_state_finish;
585
0
          state = json_tokener_state_eatws;
586
0
          goto redo_char;
587
0
        }
588
0
      }
589
0
      else
590
0
      {
591
0
        tok->err = json_tokener_error_parse_null;
592
0
        goto out;
593
0
      }
594
0
      tok->st_pos++;
595
0
    }
596
0
    break;
597
598
0
    case json_tokener_state_comment_start:
599
0
      if (c == '*')
600
0
      {
601
0
        state = json_tokener_state_comment;
602
0
      }
603
0
      else if (c == '/')
604
0
      {
605
0
        state = json_tokener_state_comment_eol;
606
0
      }
607
0
      else
608
0
      {
609
0
        tok->err = json_tokener_error_parse_comment;
610
0
        goto out;
611
0
      }
612
0
      printbuf_memappend_checked(tok->pb, &c, 1);
613
0
      break;
614
615
0
    case json_tokener_state_comment:
616
0
    {
617
      /* Advance until we change state */
618
0
      const char *case_start = str;
619
0
      while (c != '*')
620
0
      {
621
0
        if (!ADVANCE_CHAR(str, tok) || !PEEK_CHAR(c, tok))
622
0
        {
623
0
          printbuf_memappend_checked(tok->pb, case_start,
624
0
                                     str - case_start);
625
0
          goto out;
626
0
        }
627
0
      }
628
0
      printbuf_memappend_checked(tok->pb, case_start, 1 + str - case_start);
629
0
      state = json_tokener_state_comment_end;
630
0
    }
631
0
    break;
632
633
0
    case json_tokener_state_comment_eol:
634
0
    {
635
      /* Advance until we change state */
636
0
      const char *case_start = str;
637
0
      while (c != '\n')
638
0
      {
639
0
        if (!ADVANCE_CHAR(str, tok) || !PEEK_CHAR(c, tok))
640
0
        {
641
0
          printbuf_memappend_checked(tok->pb, case_start,
642
0
                                     str - case_start);
643
0
          goto out;
644
0
        }
645
0
      }
646
0
      printbuf_memappend_checked(tok->pb, case_start, str - case_start);
647
0
      MC_DEBUG("json_tokener_comment: %s\n", tok->pb->buf);
648
0
      state = json_tokener_state_eatws;
649
0
    }
650
0
    break;
651
652
0
    case json_tokener_state_comment_end:
653
0
      printbuf_memappend_checked(tok->pb, &c, 1);
654
0
      if (c == '/')
655
0
      {
656
0
        MC_DEBUG("json_tokener_comment: %s\n", tok->pb->buf);
657
0
        state = json_tokener_state_eatws;
658
0
      }
659
0
      else
660
0
      {
661
0
        state = json_tokener_state_comment;
662
0
      }
663
0
      break;
664
665
0
    case json_tokener_state_string:
666
0
    {
667
      /* Advance until we change state */
668
0
      const char *case_start = str;
669
0
      while (1)
670
0
      {
671
0
        if (c == tok->quote_char)
672
0
        {
673
0
          printbuf_memappend_checked(tok->pb, case_start,
674
0
                                     str - case_start);
675
0
          current =
676
0
              json_object_new_string_len(tok->pb->buf, tok->pb->bpos);
677
0
          if (current == NULL)
678
0
          {
679
0
            tok->err = json_tokener_error_memory;
680
0
            goto out;
681
0
          }
682
0
          saved_state = json_tokener_state_finish;
683
0
          state = json_tokener_state_eatws;
684
0
          break;
685
0
        }
686
0
        else if (c == '\\')
687
0
        {
688
0
          printbuf_memappend_checked(tok->pb, case_start,
689
0
                                     str - case_start);
690
0
          saved_state = json_tokener_state_string;
691
0
          state = json_tokener_state_string_escape;
692
0
          break;
693
0
        }
694
0
        else if ((tok->flags & JSON_TOKENER_STRICT) && (unsigned char)c <= 0x1f)
695
0
        {
696
          // Disallow control characters in strict mode
697
0
          tok->err = json_tokener_error_parse_string;
698
0
          goto out;
699
0
        }
700
0
        if (!ADVANCE_CHAR(str, tok) || !PEEK_CHAR(c, tok))
701
0
        {
702
0
          printbuf_memappend_checked(tok->pb, case_start,
703
0
                                     str - case_start);
704
0
          goto out;
705
0
        }
706
0
      }
707
0
    }
708
0
    break;
709
710
0
    case json_tokener_state_string_escape:
711
0
      switch (c)
712
0
      {
713
0
      case '"':
714
0
      case '\\':
715
0
      case '/':
716
0
        printbuf_memappend_checked(tok->pb, &c, 1);
717
0
        state = saved_state;
718
0
        break;
719
0
      case 'b':
720
0
      case 'n':
721
0
      case 'r':
722
0
      case 't':
723
0
      case 'f':
724
0
        if (c == 'b')
725
0
          printbuf_memappend_checked(tok->pb, "\b", 1);
726
0
        else if (c == 'n')
727
0
          printbuf_memappend_checked(tok->pb, "\n", 1);
728
0
        else if (c == 'r')
729
0
          printbuf_memappend_checked(tok->pb, "\r", 1);
730
0
        else if (c == 't')
731
0
          printbuf_memappend_checked(tok->pb, "\t", 1);
732
0
        else if (c == 'f')
733
0
          printbuf_memappend_checked(tok->pb, "\f", 1);
734
0
        state = saved_state;
735
0
        break;
736
0
      case 'u':
737
0
        tok->ucs_char = 0;
738
0
        tok->st_pos = 0;
739
0
        state = json_tokener_state_escape_unicode;
740
0
        break;
741
0
      default: tok->err = json_tokener_error_parse_string; goto out;
742
0
      }
743
0
      break;
744
745
      // ===================================================
746
747
0
    case json_tokener_state_escape_unicode:
748
0
    {
749
      /* Handle a 4-byte \uNNNN sequence, or two sequences if a surrogate pair */
750
0
      while (1)
751
0
      {
752
0
        if (!c || !is_hex_char(c))
753
0
        {
754
0
          tok->err = json_tokener_error_parse_string;
755
0
          goto out;
756
0
        }
757
0
        tok->ucs_char |=
758
0
            ((unsigned int)jt_hexdigit(c) << ((3 - tok->st_pos) * 4));
759
0
        tok->st_pos++;
760
0
        if (tok->st_pos >= 4)
761
0
          break;
762
763
0
        (void)ADVANCE_CHAR(str, tok);
764
0
        if (!PEEK_CHAR(c, tok))
765
0
        {
766
          /*
767
           * We're out of characters in the current call to
768
           * json_tokener_parse(), but a subsequent call might
769
           * provide us with more, so leave our current state
770
           * as-is (including tok->high_surrogate) and return.
771
           */
772
0
          goto out;
773
0
        }
774
0
      }
775
0
      tok->st_pos = 0;
776
777
      /* Now, we have a full \uNNNN sequence in tok->ucs_char */
778
779
      /* If the *previous* sequence was a high surrogate ... */
780
0
      if (tok->high_surrogate)
781
0
      {
782
0
        if (IS_LOW_SURROGATE(tok->ucs_char))
783
0
        {
784
          /* Recalculate the ucs_char, then fall thru to process normally */
785
0
          tok->ucs_char = DECODE_SURROGATE_PAIR(tok->high_surrogate,
786
0
                                                tok->ucs_char);
787
0
        }
788
0
        else
789
0
        {
790
          /* High surrogate was not followed by a low surrogate
791
           * Replace the high and process the rest normally
792
           */
793
0
          printbuf_memappend_checked(tok->pb,
794
0
                                     (char *)utf8_replacement_char, 3);
795
0
        }
796
0
        tok->high_surrogate = 0;
797
0
      }
798
799
0
      if (tok->ucs_char < 0x80)
800
0
      {
801
0
        unsigned char unescaped_utf[1];
802
0
        unescaped_utf[0] = tok->ucs_char;
803
0
        printbuf_memappend_checked(tok->pb, (char *)unescaped_utf, 1);
804
0
      }
805
0
      else if (tok->ucs_char < 0x800)
806
0
      {
807
0
        unsigned char unescaped_utf[2];
808
0
        unescaped_utf[0] = 0xc0 | (tok->ucs_char >> 6);
809
0
        unescaped_utf[1] = 0x80 | (tok->ucs_char & 0x3f);
810
0
        printbuf_memappend_checked(tok->pb, (char *)unescaped_utf, 2);
811
0
      }
812
0
      else if (IS_HIGH_SURROGATE(tok->ucs_char))
813
0
      {
814
        /*
815
         * The next two characters should be \u, HOWEVER,
816
         * we can't simply peek ahead here, because the
817
         * characters we need might not be passed to us
818
         * until a subsequent call to json_tokener_parse.
819
         * Instead, transition through a couple of states.
820
         * (now):
821
         *   _escape_unicode => _unicode_need_escape
822
         * (see a '\\' char):
823
         *   _unicode_need_escape => _unicode_need_u
824
         * (see a 'u' char):
825
         *   _unicode_need_u => _escape_unicode
826
         *      ...and we'll end up back around here.
827
         */
828
0
        tok->high_surrogate = tok->ucs_char;
829
0
        tok->ucs_char = 0;
830
0
        state = json_tokener_state_escape_unicode_need_escape;
831
0
        break;
832
0
      }
833
0
      else if (IS_LOW_SURROGATE(tok->ucs_char))
834
0
      {
835
        /* Got a low surrogate not preceded by a high */
836
0
        printbuf_memappend_checked(tok->pb, (char *)utf8_replacement_char, 3);
837
0
      }
838
0
      else if (tok->ucs_char < 0x10000)
839
0
      {
840
0
        unsigned char unescaped_utf[3];
841
0
        unescaped_utf[0] = 0xe0 | (tok->ucs_char >> 12);
842
0
        unescaped_utf[1] = 0x80 | ((tok->ucs_char >> 6) & 0x3f);
843
0
        unescaped_utf[2] = 0x80 | (tok->ucs_char & 0x3f);
844
0
        printbuf_memappend_checked(tok->pb, (char *)unescaped_utf, 3);
845
0
      }
846
0
      else if (tok->ucs_char < 0x110000)
847
0
      {
848
0
        unsigned char unescaped_utf[4];
849
0
        unescaped_utf[0] = 0xf0 | ((tok->ucs_char >> 18) & 0x07);
850
0
        unescaped_utf[1] = 0x80 | ((tok->ucs_char >> 12) & 0x3f);
851
0
        unescaped_utf[2] = 0x80 | ((tok->ucs_char >> 6) & 0x3f);
852
0
        unescaped_utf[3] = 0x80 | (tok->ucs_char & 0x3f);
853
0
        printbuf_memappend_checked(tok->pb, (char *)unescaped_utf, 4);
854
0
      }
855
0
      else
856
0
      {
857
        /* Don't know what we got--insert the replacement char */
858
0
        printbuf_memappend_checked(tok->pb, (char *)utf8_replacement_char, 3);
859
0
      }
860
0
      state = saved_state; // i.e. _state_string or _state_object_field
861
0
    }
862
0
    break;
863
864
0
    case json_tokener_state_escape_unicode_need_escape:
865
      // We get here after processing a high_surrogate
866
      // require a '\\' char
867
0
      if (!c || c != '\\')
868
0
      {
869
        /* Got a high surrogate without another sequence following
870
         * it.  Put a replacement char in for the high surrogate
871
         * and pop back up to _state_string or _state_object_field.
872
         */
873
0
        printbuf_memappend_checked(tok->pb, (char *)utf8_replacement_char, 3);
874
0
        tok->high_surrogate = 0;
875
0
        tok->ucs_char = 0;
876
0
        tok->st_pos = 0;
877
0
        state = saved_state;
878
0
        goto redo_char;
879
0
      }
880
0
      state = json_tokener_state_escape_unicode_need_u;
881
0
      break;
882
883
0
    case json_tokener_state_escape_unicode_need_u:
884
      /* We already had a \ char, check that it's \u */
885
0
      if (!c || c != 'u')
886
0
      {
887
        /* Got a high surrogate with some non-unicode escape
888
         * sequence following it.
889
         * Put a replacement char in for the high surrogate
890
         * and handle the escape sequence normally.
891
         */
892
0
        printbuf_memappend_checked(tok->pb, (char *)utf8_replacement_char, 3);
893
0
        tok->high_surrogate = 0;
894
0
        tok->ucs_char = 0;
895
0
        tok->st_pos = 0;
896
0
        state = json_tokener_state_string_escape;
897
0
        goto redo_char;
898
0
      }
899
0
      state = json_tokener_state_escape_unicode;
900
0
      break;
901
902
      // ===================================================
903
904
0
    case json_tokener_state_boolean:
905
0
    {
906
0
      int size1, size2;
907
0
      printbuf_memappend_checked(tok->pb, &c, 1);
908
0
      size1 = json_min(tok->st_pos + 1, json_true_str_len);
909
0
      size2 = json_min(tok->st_pos + 1, json_false_str_len);
910
0
      if ((!(tok->flags & JSON_TOKENER_STRICT) &&
911
0
           strncasecmp(json_true_str, tok->pb->buf, size1) == 0) ||
912
0
          (strncmp(json_true_str, tok->pb->buf, size1) == 0))
913
0
      {
914
0
        if (tok->st_pos == json_true_str_len)
915
0
        {
916
0
          current = json_object_new_boolean(1);
917
0
          if (current == NULL)
918
0
          {
919
0
            tok->err = json_tokener_error_memory;
920
0
            goto out;
921
0
          }
922
0
          saved_state = json_tokener_state_finish;
923
0
          state = json_tokener_state_eatws;
924
0
          goto redo_char;
925
0
        }
926
0
      }
927
0
      else if ((!(tok->flags & JSON_TOKENER_STRICT) &&
928
0
                strncasecmp(json_false_str, tok->pb->buf, size2) == 0) ||
929
0
               (strncmp(json_false_str, tok->pb->buf, size2) == 0))
930
0
      {
931
0
        if (tok->st_pos == json_false_str_len)
932
0
        {
933
0
          current = json_object_new_boolean(0);
934
0
          if (current == NULL)
935
0
          {
936
0
            tok->err = json_tokener_error_memory;
937
0
            goto out;
938
0
          }
939
0
          saved_state = json_tokener_state_finish;
940
0
          state = json_tokener_state_eatws;
941
0
          goto redo_char;
942
0
        }
943
0
      }
944
0
      else
945
0
      {
946
0
        tok->err = json_tokener_error_parse_boolean;
947
0
        goto out;
948
0
      }
949
0
      tok->st_pos++;
950
0
    }
951
0
    break;
952
953
0
    case json_tokener_state_number:
954
0
    {
955
      /* Advance until we change state */
956
0
      const char *case_start = str;
957
0
      int case_len = 0;
958
0
      int is_exponent = 0;
959
0
      int neg_sign_ok = 1;
960
0
      int pos_sign_ok = 0;
961
0
      if (printbuf_length(tok->pb) > 0)
962
0
      {
963
        /* We don't save all state from the previous incremental parse
964
           so we need to re-generate it based on the saved string so far.
965
         */
966
0
        char *e_loc = strchr(tok->pb->buf, 'e');
967
0
        if (!e_loc)
968
0
          e_loc = strchr(tok->pb->buf, 'E');
969
0
        if (e_loc)
970
0
        {
971
0
          char *last_saved_char =
972
0
              &tok->pb->buf[printbuf_length(tok->pb) - 1];
973
0
          is_exponent = 1;
974
0
          pos_sign_ok = neg_sign_ok = 1;
975
          /* If the "e" isn't at the end, we can't start with a '-' */
976
0
          if (e_loc != last_saved_char)
977
0
          {
978
0
            neg_sign_ok = 0;
979
0
            pos_sign_ok = 0;
980
0
          }
981
          // else leave it set to 1, i.e. start of the new input
982
0
        }
983
0
      }
984
985
0
      while (c && ((c >= '0' && c <= '9') ||
986
0
                   (!is_exponent && (c == 'e' || c == 'E')) ||
987
0
                   (neg_sign_ok && c == '-') || (pos_sign_ok && c == '+') ||
988
0
                   (!tok->is_double && c == '.')))
989
0
      {
990
0
        pos_sign_ok = neg_sign_ok = 0;
991
0
        ++case_len;
992
993
        /* non-digit characters checks */
994
        /* note: since the main loop condition to get here was
995
         * an input starting with 0-9 or '-', we are
996
         * protected from input starting with '.' or
997
         * e/E.
998
         */
999
0
        switch (c)
1000
0
        {
1001
0
        case '.':
1002
0
          tok->is_double = 1;
1003
0
          pos_sign_ok = 1;
1004
0
          neg_sign_ok = 1;
1005
0
          break;
1006
0
        case 'e': /* FALLTHRU */
1007
0
        case 'E':
1008
0
          is_exponent = 1;
1009
0
          tok->is_double = 1;
1010
          /* the exponent part can begin with a negative sign */
1011
0
          pos_sign_ok = neg_sign_ok = 1;
1012
0
          break;
1013
0
        default: break;
1014
0
        }
1015
1016
0
        if (!ADVANCE_CHAR(str, tok) || !PEEK_CHAR(c, tok))
1017
0
        {
1018
0
          printbuf_memappend_checked(tok->pb, case_start, case_len);
1019
0
          goto out;
1020
0
        }
1021
0
      }
1022
      /*
1023
        Now we know c isn't a valid number char, but check whether
1024
        it might have been intended to be, and return a potentially
1025
        more understandable error right away.
1026
        However, if we're at the top-level, use the number as-is
1027
        because c can be part of a new object to parse on the
1028
        next call to json_tokener_parse().
1029
       */
1030
0
      if (tok->depth > 0 && c != ',' && c != ']' && c != '}' && c != '/' &&
1031
0
          c != 'I' && c != 'i' && !is_ws_char(c))
1032
0
      {
1033
0
        tok->err = json_tokener_error_parse_number;
1034
0
        goto out;
1035
0
      }
1036
0
      if (case_len > 0)
1037
0
        printbuf_memappend_checked(tok->pb, case_start, case_len);
1038
1039
      // Check for -Infinity
1040
0
      if (tok->pb->buf[0] == '-' && case_len <= 1 && (c == 'i' || c == 'I'))
1041
0
      {
1042
0
        state = json_tokener_state_inf;
1043
0
        tok->st_pos = 0;
1044
0
        goto redo_char;
1045
0
      }
1046
0
      if (tok->is_double && !(tok->flags & JSON_TOKENER_STRICT))
1047
0
      {
1048
        /* Trim some chars off the end, to allow things
1049
           like "123e+" to parse ok. */
1050
0
        while (printbuf_length(tok->pb) > 1)
1051
0
        {
1052
0
          char last_char = tok->pb->buf[printbuf_length(tok->pb) - 1];
1053
0
          if (last_char != 'e' && last_char != 'E' &&
1054
0
              last_char != '-' && last_char != '+')
1055
0
          {
1056
0
            break;
1057
0
          }
1058
0
          tok->pb->buf[printbuf_length(tok->pb) - 1] = '\0';
1059
0
          printbuf_length(tok->pb)--;
1060
0
        }
1061
0
      }
1062
0
    }
1063
0
      {
1064
0
        int64_t num64;
1065
0
        uint64_t numuint64;
1066
0
        double numd;
1067
0
        if (!tok->is_double && tok->pb->buf[0] == '-' &&
1068
0
            json_parse_int64(tok->pb->buf, &num64) == 0)
1069
0
        {
1070
0
          if (errno == ERANGE && (tok->flags & JSON_TOKENER_STRICT))
1071
0
          {
1072
0
            tok->err = json_tokener_error_parse_number;
1073
0
            goto out;
1074
0
          }
1075
0
          current = json_object_new_int64(num64);
1076
0
          if (current == NULL)
1077
0
          {
1078
0
            tok->err = json_tokener_error_memory;
1079
0
            goto out;
1080
0
          }
1081
0
        }
1082
0
        else if (!tok->is_double && tok->pb->buf[0] != '-' &&
1083
0
                 json_parse_uint64(tok->pb->buf, &numuint64) == 0)
1084
0
        {
1085
0
          if (errno == ERANGE && (tok->flags & JSON_TOKENER_STRICT))
1086
0
          {
1087
0
            tok->err = json_tokener_error_parse_number;
1088
0
            goto out;
1089
0
          }
1090
0
          if (numuint64 && tok->pb->buf[0] == '0' &&
1091
0
              (tok->flags & JSON_TOKENER_STRICT))
1092
0
          {
1093
0
            tok->err = json_tokener_error_parse_number;
1094
0
            goto out;
1095
0
          }
1096
0
          if (numuint64 <= INT64_MAX)
1097
0
          {
1098
0
            num64 = (uint64_t)numuint64;
1099
0
            current = json_object_new_int64(num64);
1100
0
            if (current == NULL)
1101
0
            {
1102
0
              tok->err = json_tokener_error_memory;
1103
0
              goto out;
1104
0
            }
1105
0
          }
1106
0
          else
1107
0
          {
1108
0
            current = json_object_new_uint64(numuint64);
1109
0
            if (current == NULL)
1110
0
            {
1111
0
              tok->err = json_tokener_error_memory;
1112
0
              goto out;
1113
0
            }
1114
0
          }
1115
0
        }
1116
0
        else if (tok->is_double &&
1117
0
                 json_tokener_parse_double(
1118
0
                     tok->pb->buf, printbuf_length(tok->pb), &numd) == 0)
1119
0
        {
1120
0
          current = json_object_new_double_s(numd, tok->pb->buf);
1121
0
          if (current == NULL)
1122
0
          {
1123
0
            tok->err = json_tokener_error_memory;
1124
0
            goto out;
1125
0
          }
1126
0
        }
1127
0
        else
1128
0
        {
1129
0
          tok->err = json_tokener_error_parse_number;
1130
0
          goto out;
1131
0
        }
1132
0
        saved_state = json_tokener_state_finish;
1133
0
        state = json_tokener_state_eatws;
1134
0
        goto redo_char;
1135
0
      }
1136
0
      break;
1137
1138
0
    case json_tokener_state_array_after_sep:
1139
0
    case json_tokener_state_array:
1140
0
      if (c == ']')
1141
0
      {
1142
        // Minimize memory usage; assume parsed objs are unlikely to be changed
1143
0
        json_object_array_shrink(current, 0);
1144
1145
0
        if (state == json_tokener_state_array_after_sep &&
1146
0
            (tok->flags & JSON_TOKENER_STRICT))
1147
0
        {
1148
0
          tok->err = json_tokener_error_parse_unexpected;
1149
0
          goto out;
1150
0
        }
1151
0
        saved_state = json_tokener_state_finish;
1152
0
        state = json_tokener_state_eatws;
1153
0
      }
1154
0
      else
1155
0
      {
1156
0
        if (tok->depth >= tok->max_depth - 1)
1157
0
        {
1158
0
          tok->err = json_tokener_error_depth;
1159
0
          goto out;
1160
0
        }
1161
0
        state = json_tokener_state_array_add;
1162
0
        tok->depth++;
1163
0
        json_tokener_reset_level(tok, tok->depth);
1164
0
        goto redo_char;
1165
0
      }
1166
0
      break;
1167
1168
0
    case json_tokener_state_array_add:
1169
0
      if (json_object_array_add(current, obj) != 0)
1170
0
      {
1171
0
        tok->err = json_tokener_error_memory;
1172
0
        goto out;
1173
0
      }
1174
0
      saved_state = json_tokener_state_array_sep;
1175
0
      state = json_tokener_state_eatws;
1176
0
      goto redo_char;
1177
1178
0
    case json_tokener_state_array_sep:
1179
0
      if (c == ']')
1180
0
      {
1181
        // Minimize memory usage; assume parsed objs are unlikely to be changed
1182
0
        json_object_array_shrink(current, 0);
1183
1184
0
        saved_state = json_tokener_state_finish;
1185
0
        state = json_tokener_state_eatws;
1186
0
      }
1187
0
      else if (c == ',')
1188
0
      {
1189
0
        saved_state = json_tokener_state_array_after_sep;
1190
0
        state = json_tokener_state_eatws;
1191
0
      }
1192
0
      else
1193
0
      {
1194
0
        tok->err = json_tokener_error_parse_array;
1195
0
        goto out;
1196
0
      }
1197
0
      break;
1198
1199
0
    case json_tokener_state_object_field_start:
1200
0
    case json_tokener_state_object_field_start_after_sep:
1201
0
      if (c == '}')
1202
0
      {
1203
0
        if (state == json_tokener_state_object_field_start_after_sep &&
1204
0
            (tok->flags & JSON_TOKENER_STRICT))
1205
0
        {
1206
0
          tok->err = json_tokener_error_parse_unexpected;
1207
0
          goto out;
1208
0
        }
1209
0
        saved_state = json_tokener_state_finish;
1210
0
        state = json_tokener_state_eatws;
1211
0
      }
1212
0
      else if (c == '"' || c == '\'')
1213
0
      {
1214
0
        tok->quote_char = c;
1215
0
        printbuf_reset(tok->pb);
1216
0
        state = json_tokener_state_object_field;
1217
0
      }
1218
0
      else
1219
0
      {
1220
0
        tok->err = json_tokener_error_parse_object_key_name;
1221
0
        goto out;
1222
0
      }
1223
0
      break;
1224
1225
0
    case json_tokener_state_object_field:
1226
0
    {
1227
      /* Advance until we change state */
1228
0
      const char *case_start = str;
1229
0
      while (1)
1230
0
      {
1231
0
        if (c == tok->quote_char)
1232
0
        {
1233
0
          printbuf_memappend_checked(tok->pb, case_start,
1234
0
                                     str - case_start);
1235
0
          obj_field_name = strdup(tok->pb->buf);
1236
0
          if (obj_field_name == NULL)
1237
0
          {
1238
0
            tok->err = json_tokener_error_memory;
1239
0
            goto out;
1240
0
          }
1241
0
          saved_state = json_tokener_state_object_field_end;
1242
0
          state = json_tokener_state_eatws;
1243
0
          break;
1244
0
        }
1245
0
        else if (c == '\\')
1246
0
        {
1247
0
          printbuf_memappend_checked(tok->pb, case_start,
1248
0
                                     str - case_start);
1249
0
          saved_state = json_tokener_state_object_field;
1250
0
          state = json_tokener_state_string_escape;
1251
0
          break;
1252
0
        }
1253
0
        else if ((tok->flags & JSON_TOKENER_STRICT) && (unsigned char)c <= 0x1f)
1254
0
        {
1255
          // Disallow control characters in strict mode
1256
0
          tok->err = json_tokener_error_parse_string;
1257
0
          goto out;
1258
0
        }
1259
0
        if (!ADVANCE_CHAR(str, tok) || !PEEK_CHAR(c, tok))
1260
0
        {
1261
0
          printbuf_memappend_checked(tok->pb, case_start,
1262
0
                                     str - case_start);
1263
0
          goto out;
1264
0
        }
1265
0
      }
1266
0
    }
1267
0
    break;
1268
1269
0
    case json_tokener_state_object_field_end:
1270
0
      if (c == ':')
1271
0
      {
1272
0
        saved_state = json_tokener_state_object_value;
1273
0
        state = json_tokener_state_eatws;
1274
0
      }
1275
0
      else
1276
0
      {
1277
0
        tok->err = json_tokener_error_parse_object_key_sep;
1278
0
        goto out;
1279
0
      }
1280
0
      break;
1281
1282
0
    case json_tokener_state_object_value:
1283
0
      if (tok->depth >= tok->max_depth - 1)
1284
0
      {
1285
0
        tok->err = json_tokener_error_depth;
1286
0
        goto out;
1287
0
      }
1288
0
      state = json_tokener_state_object_value_add;
1289
0
      tok->depth++;
1290
0
      json_tokener_reset_level(tok, tok->depth);
1291
0
      goto redo_char;
1292
1293
0
    case json_tokener_state_object_value_add:
1294
0
      if (json_object_object_add(current, obj_field_name, obj) != 0)
1295
0
      {
1296
0
        tok->err = json_tokener_error_memory;
1297
0
        goto out;
1298
0
      }
1299
0
      free(obj_field_name);
1300
0
      obj_field_name = NULL;
1301
0
      saved_state = json_tokener_state_object_sep;
1302
0
      state = json_tokener_state_eatws;
1303
0
      goto redo_char;
1304
1305
0
    case json_tokener_state_object_sep:
1306
      /* { */
1307
0
      if (c == '}')
1308
0
      {
1309
0
        saved_state = json_tokener_state_finish;
1310
0
        state = json_tokener_state_eatws;
1311
0
      }
1312
0
      else if (c == ',')
1313
0
      {
1314
0
        saved_state = json_tokener_state_object_field_start_after_sep;
1315
0
        state = json_tokener_state_eatws;
1316
0
      }
1317
0
      else
1318
0
      {
1319
0
        tok->err = json_tokener_error_parse_object_value_sep;
1320
0
        goto out;
1321
0
      }
1322
0
      break;
1323
0
    }
1324
0
    (void)ADVANCE_CHAR(str, tok);
1325
0
    if (!c) // This is the char *before* advancing
1326
0
      break;
1327
0
  } /* while(PEEK_CHAR) */
1328
1329
0
out:
1330
0
  if ((tok->flags & JSON_TOKENER_VALIDATE_UTF8) && (nBytes != 0))
1331
0
  {
1332
0
    tok->err = json_tokener_error_parse_utf8_string;
1333
0
  }
1334
0
  if (c && (state == json_tokener_state_finish) && (tok->depth == 0) &&
1335
0
      (tok->flags & (JSON_TOKENER_STRICT | JSON_TOKENER_ALLOW_TRAILING_CHARS)) ==
1336
0
          JSON_TOKENER_STRICT)
1337
0
  {
1338
    /* unexpected char after JSON data */
1339
0
    tok->err = json_tokener_error_parse_unexpected;
1340
0
  }
1341
0
  if (!c)
1342
0
  {
1343
    /* We hit an eof char (0) */
1344
0
    if (state != json_tokener_state_finish && saved_state != json_tokener_state_finish)
1345
0
      tok->err = json_tokener_error_parse_eof;
1346
0
  }
1347
1348
0
#ifdef HAVE_USELOCALE
1349
0
  uselocale(oldlocale);
1350
0
  freelocale(newloc);
1351
#elif defined(HAVE_SETLOCALE)
1352
  setlocale(LC_NUMERIC, oldlocale);
1353
  free(oldlocale);
1354
#endif
1355
1356
0
  if (tok->err == json_tokener_success)
1357
0
  {
1358
0
    json_object *ret = json_object_get(current);
1359
0
    int ii;
1360
1361
    /* Partially reset, so we parse additional objects on subsequent calls. */
1362
0
    for (ii = tok->depth; ii >= 0; ii--)
1363
0
      json_tokener_reset_level(tok, ii);
1364
0
    return ret;
1365
0
  }
1366
1367
0
  MC_DEBUG("json_tokener_parse_ex: error %s at offset %d\n", json_tokener_errors[tok->err],
1368
0
           tok->char_offset);
1369
0
  return NULL;
1370
0
}
1371
1372
static json_bool json_tokener_validate_utf8(const char c, unsigned int *nBytes)
1373
0
{
1374
0
  unsigned char chr = c;
1375
0
  if (*nBytes == 0)
1376
0
  {
1377
0
    if (chr >= 0x80)
1378
0
    {
1379
0
      if ((chr & 0xe0) == 0xc0)
1380
0
        *nBytes = 1;
1381
0
      else if ((chr & 0xf0) == 0xe0)
1382
0
        *nBytes = 2;
1383
0
      else if ((chr & 0xf8) == 0xf0)
1384
0
        *nBytes = 3;
1385
0
      else
1386
0
        return 0;
1387
0
    }
1388
0
  }
1389
0
  else
1390
0
  {
1391
0
    if ((chr & 0xC0) != 0x80)
1392
0
      return 0;
1393
0
    (*nBytes)--;
1394
0
  }
1395
0
  return 1;
1396
0
}
1397
1398
void json_tokener_set_flags(struct json_tokener *tok, int flags)
1399
0
{
1400
0
  tok->flags = flags;
1401
0
}
1402
1403
size_t json_tokener_get_parse_end(struct json_tokener *tok)
1404
0
{
1405
0
  assert(tok->char_offset >= 0); /* Drop this line when char_offset becomes a size_t */
1406
0
  return (size_t)tok->char_offset;
1407
0
}
1408
1409
static int json_tokener_parse_double(const char *buf, int len, double *retval)
1410
0
{
1411
0
  char *end;
1412
0
  *retval = strtod(buf, &end);
1413
0
  if (buf + len == end)
1414
0
    return 0; // It worked
1415
0
  return 1;
1416
0
}