LCOV - code coverage report
Current view: top level - port - cpl_json_streaming_parser.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 415 423 98.1 %
Date: 2025-01-18 12:42:00 Functions: 23 24 95.8 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  CPL - Common Portability Library
       4             :  * Purpose:  JSon streaming parser
       5             :  * Author:   Even Rouault, even.rouault at spatialys.com
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2017, Even Rouault <even.rouault at spatialys.com>
       9             :  *
      10             :  * SPDX-License-Identifier: MIT
      11             :  ****************************************************************************/
      12             : 
      13             : /*! @cond Doxygen_Suppress */
      14             : 
      15             : #include <assert.h>
      16             : #include <ctype.h>   // isdigit...
      17             : #include <stdio.h>   // snprintf
      18             : #include <string.h>  // strlen
      19             : #include <vector>
      20             : #include <string>
      21             : 
      22             : #include "cpl_conv.h"
      23             : #include "cpl_string.h"
      24             : #include "cpl_json_streaming_parser.h"
      25             : 
      26             : /************************************************************************/
      27             : /*                       CPLJSonStreamingParser()                       */
      28             : /************************************************************************/
      29             : 
      30        2870 : CPLJSonStreamingParser::CPLJSonStreamingParser()
      31             : {
      32        2870 :     m_aState.push_back(INIT);
      33        2870 : }
      34             : 
      35             : /************************************************************************/
      36             : /*                      ~CPLJSonStreamingParser()                       */
      37             : /************************************************************************/
      38             : 
      39        2870 : CPLJSonStreamingParser::~CPLJSonStreamingParser()
      40             : {
      41        2870 : }
      42             : 
      43             : /************************************************************************/
      44             : /*                           SetMaxDepth()                              */
      45             : /************************************************************************/
      46             : 
      47           2 : void CPLJSonStreamingParser::SetMaxDepth(size_t nVal)
      48             : {
      49           2 :     m_nMaxDepth = nVal;
      50           2 : }
      51             : 
      52             : /************************************************************************/
      53             : /*                         SetMaxStringSize()                           */
      54             : /************************************************************************/
      55             : 
      56           1 : void CPLJSonStreamingParser::SetMaxStringSize(size_t nVal)
      57             : {
      58           1 :     m_nMaxStringSize = nVal;
      59           1 : }
      60             : 
      61             : /************************************************************************/
      62             : /*                                Reset()                               */
      63             : /************************************************************************/
      64             : 
      65          18 : void CPLJSonStreamingParser::Reset()
      66             : {
      67          18 :     m_bExceptionOccurred = false;
      68          18 :     m_bElementFound = false;
      69          18 :     m_nLastChar = 0;
      70          18 :     m_nLineCounter = 1;
      71          18 :     m_nCharCounter = 1;
      72          18 :     m_aState.clear();
      73          18 :     m_aState.push_back(INIT);
      74          18 :     m_osToken.clear();
      75          18 :     m_abArrayState.clear();
      76          18 :     m_aeObjectState.clear();
      77          18 :     m_bInStringEscape = false;
      78          18 :     m_bInUnicode = false;
      79          18 :     m_osUnicodeHex.clear();
      80          18 : }
      81             : 
      82             : /************************************************************************/
      83             : /*                              AdvanceChar()                           */
      84             : /************************************************************************/
      85             : 
      86    11217400 : void CPLJSonStreamingParser::AdvanceChar(const char *&pStr, size_t &nLength)
      87             : {
      88    11217400 :     if (*pStr == 13 && m_nLastChar != 10)
      89             :     {
      90          27 :         m_nLineCounter++;
      91          27 :         m_nCharCounter = 0;
      92             :     }
      93    11217400 :     else if (*pStr == 10 && m_nLastChar != 13)
      94             :     {
      95      199204 :         m_nLineCounter++;
      96      199204 :         m_nCharCounter = 0;
      97             :     }
      98    11217400 :     m_nLastChar = *pStr;
      99             : 
     100    11217400 :     pStr++;
     101    11217400 :     nLength--;
     102    11217400 :     m_nCharCounter++;
     103    11217400 : }
     104             : 
     105             : /************************************************************************/
     106             : /*                               SkipSpace()                            */
     107             : /************************************************************************/
     108             : 
     109     5193410 : void CPLJSonStreamingParser::SkipSpace(const char *&pStr, size_t &nLength)
     110             : {
     111     5193410 :     while (nLength > 0 && isspace(static_cast<unsigned char>(*pStr)))
     112             :     {
     113     3125840 :         AdvanceChar(pStr, nLength);
     114             :     }
     115     2067580 : }
     116             : 
     117             : /************************************************************************/
     118             : /*                             EmitException()                          */
     119             : /************************************************************************/
     120             : 
     121          60 : bool CPLJSonStreamingParser::EmitException(const char *pszMessage)
     122             : {
     123          60 :     m_bExceptionOccurred = true;
     124          60 :     CPLString osMsg;
     125             :     osMsg.Printf("At line %d, character %d: %s", m_nLineCounter, m_nCharCounter,
     126          60 :                  pszMessage);
     127          60 :     Exception(osMsg.c_str());
     128         120 :     return false;
     129             : }
     130             : 
     131             : /************************************************************************/
     132             : /*                             StopParsing()                            */
     133             : /************************************************************************/
     134             : 
     135        1894 : void CPLJSonStreamingParser::StopParsing()
     136             : {
     137        1894 :     m_bStopParsing = true;
     138        1894 : }
     139             : 
     140             : /************************************************************************/
     141             : /*                          EmitUnexpectedChar()                        */
     142             : /************************************************************************/
     143             : 
     144          33 : bool CPLJSonStreamingParser::EmitUnexpectedChar(char ch,
     145             :                                                 const char *pszExpecting)
     146             : {
     147             :     char szMessage[64];
     148          33 :     if (pszExpecting)
     149             :     {
     150           7 :         snprintf(szMessage, sizeof(szMessage),
     151             :                  "Unexpected character (%c). Expecting %s", ch, pszExpecting);
     152             :     }
     153             :     else
     154             :     {
     155          26 :         snprintf(szMessage, sizeof(szMessage), "Unexpected character (%c)", ch);
     156             :     }
     157          66 :     return EmitException(szMessage);
     158             : }
     159             : 
     160             : /************************************************************************/
     161             : /*                            IsValidNewToken()                         */
     162             : /************************************************************************/
     163             : 
     164      771373 : static bool IsValidNewToken(char ch)
     165             : {
     166      549820 :     return ch == '[' || ch == '{' || ch == '"' || ch == '-' || ch == '.' ||
     167      360586 :            isdigit(static_cast<unsigned char>(ch)) || ch == 't' || ch == 'f' ||
     168     1321190 :            ch == 'n' || ch == 'i' || ch == 'I' || ch == 'N';
     169             : }
     170             : 
     171             : /************************************************************************/
     172             : /*                             StartNewToken()                          */
     173             : /************************************************************************/
     174             : 
     175      771359 : bool CPLJSonStreamingParser::StartNewToken(const char *&pStr, size_t &nLength)
     176             : {
     177      771359 :     char ch = *pStr;
     178      771359 :     if (ch == '{')
     179             :     {
     180       22844 :         if (m_aState.size() == m_nMaxDepth)
     181             :         {
     182           1 :             return EmitException("Too many nested objects and/or arrays");
     183             :         }
     184       22843 :         StartObject();
     185       22843 :         m_aeObjectState.push_back(WAITING_KEY);
     186       22843 :         m_aState.push_back(OBJECT);
     187       22843 :         AdvanceChar(pStr, nLength);
     188             :     }
     189      748515 :     else if (ch == '"')
     190             :     {
     191       92906 :         m_aState.push_back(STRING);
     192       92906 :         AdvanceChar(pStr, nLength);
     193             :     }
     194      655609 :     else if (ch == '[')
     195             :     {
     196      221553 :         if (m_aState.size() == m_nMaxDepth)
     197             :         {
     198           1 :             return EmitException("Too many nested objects and/or arrays");
     199             :         }
     200      221552 :         StartArray();
     201      221552 :         m_abArrayState.push_back(ArrayState::INIT);
     202      221552 :         m_aState.push_back(ARRAY);
     203      221552 :         AdvanceChar(pStr, nLength);
     204             :     }
     205      434056 :     else if (ch == '-' || ch == '.' ||
     206      360572 :              isdigit(static_cast<unsigned char>(ch)) || ch == 'i' ||
     207        1541 :              ch == 'I' || ch == 'N')
     208             :     {
     209      432517 :         m_aState.push_back(NUMBER);
     210             :     }
     211        1539 :     else if (ch == 't')
     212             :     {
     213         162 :         m_aState.push_back(STATE_TRUE);
     214             :     }
     215        1377 :     else if (ch == 'f')
     216             :     {
     217          11 :         m_aState.push_back(STATE_FALSE);
     218             :     }
     219        1366 :     else if (ch == 'n')
     220             :     {
     221        1366 :         m_aState.push_back(STATE_NULL); /* might be nan */
     222             :     }
     223             :     else
     224             :     {
     225           0 :         assert(false);
     226             :     }
     227      771357 :     return true;
     228             : }
     229             : 
     230             : /************************************************************************/
     231             : /*                       CheckAndEmitTrueFalseOrNull()                  */
     232             : /************************************************************************/
     233             : 
     234        1528 : bool CPLJSonStreamingParser::CheckAndEmitTrueFalseOrNull(char ch)
     235             : {
     236        1528 :     State eCurState = currentState();
     237             : 
     238        1528 :     if (eCurState == STATE_TRUE)
     239             :     {
     240         159 :         if (m_osToken == "true")
     241             :         {
     242         158 :             Boolean(true);
     243             :         }
     244             :         else
     245             :         {
     246           1 :             return EmitUnexpectedChar(ch);
     247             :         }
     248             :     }
     249        1369 :     else if (eCurState == STATE_FALSE)
     250             :     {
     251           9 :         if (m_osToken == "false")
     252             :         {
     253           8 :             Boolean(false);
     254             :         }
     255             :         else
     256             :         {
     257           1 :             return EmitUnexpectedChar(ch);
     258             :         }
     259             :     }
     260             :     else /* if( eCurState == STATE_NULL ) */
     261             :     {
     262        1360 :         if (m_osToken == "null")
     263             :         {
     264        1359 :             Null();
     265             :         }
     266             :         else
     267             :         {
     268           1 :             return EmitUnexpectedChar(ch);
     269             :         }
     270             :     }
     271        1525 :     m_aState.pop_back();
     272        1525 :     m_osToken.clear();
     273        1525 :     return true;
     274             : }
     275             : 
     276             : /************************************************************************/
     277             : /*                           CheckStackEmpty()                          */
     278             : /************************************************************************/
     279             : 
     280          26 : bool CPLJSonStreamingParser::CheckStackEmpty()
     281             : {
     282          26 :     if (!m_aeObjectState.empty())
     283             :     {
     284           1 :         return EmitException("Unterminated object");
     285             :     }
     286          25 :     else if (!m_abArrayState.empty())
     287             :     {
     288           1 :         return EmitException("Unterminated array");
     289             :     }
     290          24 :     return true;
     291             : }
     292             : 
     293             : /************************************************************************/
     294             : /*                           IsHighSurrogate()                          */
     295             : /************************************************************************/
     296             : 
     297          21 : static bool IsHighSurrogate(unsigned uc)
     298             : {
     299          21 :     return (uc & 0xFC00) == 0xD800;
     300             : }
     301             : 
     302             : /************************************************************************/
     303             : /*                           IsLowSurrogate()                           */
     304             : /************************************************************************/
     305             : 
     306           8 : static bool IsLowSurrogate(unsigned uc)
     307             : {
     308           8 :     return (uc & 0xFC00) == 0xDC00;
     309             : }
     310             : 
     311             : /************************************************************************/
     312             : /*                         GetSurrogatePair()                           */
     313             : /************************************************************************/
     314             : 
     315           1 : static unsigned GetSurrogatePair(unsigned hi, unsigned lo)
     316             : {
     317           1 :     return ((hi & 0x3FF) << 10) + (lo & 0x3FF) + 0x10000;
     318             : }
     319             : 
     320             : /************************************************************************/
     321             : /*                            IsHexDigit()                              */
     322             : /************************************************************************/
     323             : 
     324          69 : static bool IsHexDigit(char ch)
     325             : {
     326          81 :     return (ch >= '0' && ch <= '9') || (ch >= 'a' && ch <= 'f') ||
     327          81 :            (ch >= 'A' && ch <= 'F');
     328             : }
     329             : 
     330             : /************************************************************************/
     331             : /*                           HexToDecimal()                             */
     332             : /************************************************************************/
     333             : 
     334         116 : static unsigned HexToDecimal(char ch)
     335             : {
     336         116 :     if (ch >= '0' && ch <= '9')
     337          93 :         return ch - '0';
     338          23 :     if (ch >= 'a' && ch <= 'f')
     339           8 :         return 10 + ch - 'a';
     340             :     // if (ch >= 'A' && ch <= 'F' )
     341          15 :     return 10 + ch - 'A';
     342             : }
     343             : 
     344             : /************************************************************************/
     345             : /*                            getUCSChar()                              */
     346             : /************************************************************************/
     347             : 
     348          29 : static unsigned getUCSChar(const std::string &unicode4HexChar)
     349             : {
     350          29 :     return (HexToDecimal(unicode4HexChar[0]) << 12) |
     351          29 :            (HexToDecimal(unicode4HexChar[1]) << 8) |
     352          29 :            (HexToDecimal(unicode4HexChar[2]) << 4) |
     353          29 :            (HexToDecimal(unicode4HexChar[3]));
     354             : }
     355             : 
     356             : /************************************************************************/
     357             : /*                           DecodeUnicode()                            */
     358             : /************************************************************************/
     359             : 
     360          13 : void CPLJSonStreamingParser::DecodeUnicode()
     361             : {
     362          13 :     constexpr char szReplacementUTF8[] = "\xEF\xBF\xBD";
     363             :     unsigned nUCSChar;
     364          13 :     if (m_osUnicodeHex.size() == 8)
     365             :     {
     366           2 :         unsigned nUCSHigh = getUCSChar(m_osUnicodeHex);
     367           2 :         assert(IsHighSurrogate(nUCSHigh));
     368           2 :         unsigned nUCSLow = getUCSChar(m_osUnicodeHex.substr(4));
     369           2 :         if (IsLowSurrogate(nUCSLow))
     370             :         {
     371           1 :             nUCSChar = GetSurrogatePair(nUCSHigh, nUCSLow);
     372             :         }
     373             :         else
     374             :         {
     375             :             /* Invalid code point. Insert the replacement char */
     376           1 :             nUCSChar = 0xFFFFFFFFU;
     377             :         }
     378             :     }
     379             :     else
     380             :     {
     381          11 :         assert(m_osUnicodeHex.size() == 4);
     382          11 :         nUCSChar = getUCSChar(m_osUnicodeHex);
     383             :     }
     384             : 
     385          13 :     if (nUCSChar < 0x80)
     386             :     {
     387           6 :         m_osToken += static_cast<char>(nUCSChar);
     388             :     }
     389           7 :     else if (nUCSChar < 0x800)
     390             :     {
     391           1 :         m_osToken += static_cast<char>(0xC0 | (nUCSChar >> 6));
     392           1 :         m_osToken += static_cast<char>(0x80 | (nUCSChar & 0x3F));
     393             :     }
     394           6 :     else if (IsLowSurrogate(nUCSChar) || IsHighSurrogate(nUCSChar))
     395             :     {
     396             :         /* Invalid code point. Insert the replacement char */
     397           4 :         m_osToken += szReplacementUTF8;
     398             :     }
     399           2 :     else if (nUCSChar < 0x10000)
     400             :     {
     401           0 :         m_osToken += static_cast<char>(0xE0 | (nUCSChar >> 12));
     402           0 :         m_osToken += static_cast<char>(0x80 | ((nUCSChar >> 6) & 0x3F));
     403           0 :         m_osToken += static_cast<char>(0x80 | (nUCSChar & 0x3F));
     404             :     }
     405           2 :     else if (nUCSChar < 0x110000)
     406             :     {
     407           1 :         m_osToken += static_cast<char>(0xF0 | ((nUCSChar >> 18) & 0x07));
     408           1 :         m_osToken += static_cast<char>(0x80 | ((nUCSChar >> 12) & 0x3F));
     409           1 :         m_osToken += static_cast<char>(0x80 | ((nUCSChar >> 6) & 0x3F));
     410           1 :         m_osToken += static_cast<char>(0x80 | (nUCSChar & 0x3F));
     411             :     }
     412             :     else
     413             :     {
     414             :         /* Invalid code point. Insert the replacement char */
     415           1 :         m_osToken += szReplacementUTF8;
     416             :     }
     417             : 
     418          13 :     m_bInUnicode = false;
     419          13 :     m_osUnicodeHex.clear();
     420          13 : }
     421             : 
     422             : /************************************************************************/
     423             : /*                              Parse()                                 */
     424             : /************************************************************************/
     425             : 
     426     2070740 : bool CPLJSonStreamingParser::Parse(const char *pStr, size_t nLength,
     427             :                                    bool bFinished)
     428             : {
     429             :     while (true)
     430             :     {
     431     2070740 :         if (m_bExceptionOccurred || m_bStopParsing)
     432        1895 :             return false;
     433     2068840 :         State eCurState = currentState();
     434     2068840 :         if (eCurState == INIT)
     435             :         {
     436        3803 :             SkipSpace(pStr, nLength);
     437        3803 :             if (nLength == 0)
     438        1020 :                 return true;
     439        2783 :             if (m_bElementFound || !IsValidNewToken(*pStr))
     440             :             {
     441           9 :                 return EmitUnexpectedChar(*pStr);
     442             :             }
     443        2774 :             if (!StartNewToken(pStr, nLength))
     444             :             {
     445           2 :                 return false;
     446             :             }
     447        2772 :             m_bElementFound = true;
     448             :         }
     449     2065040 :         else if (eCurState == NUMBER)
     450             :         {
     451     6542890 :             while (nLength)
     452             :             {
     453     6542690 :                 char ch = *pStr;
     454     6542690 :                 if (ch == '+' || ch == '-' ||
     455     6469190 :                     isdigit(static_cast<unsigned char>(ch)) || ch == '.' ||
     456      432638 :                     ch == 'e' || ch == 'E')
     457             :                 {
     458     6110050 :                     if (m_osToken.size() == 1024)
     459             :                     {
     460           0 :                         return EmitException("Too many characters in number");
     461             :                     }
     462     6110050 :                     m_osToken += ch;
     463             :                 }
     464      432636 :                 else if (isspace(static_cast<unsigned char>(ch)) || ch == ',' ||
     465       22119 :                          ch == '}' || ch == ']')
     466             :                 {
     467      432525 :                     SkipSpace(pStr, nLength);
     468      432525 :                     break;
     469             :                 }
     470             :                 else
     471             :                 {
     472         111 :                     CPLString extendedToken(m_osToken + ch);
     473         111 :                     if ((STARTS_WITH_CI("Infinity", extendedToken) &&
     474          47 :                          m_osToken.size() + 1 <= strlen("Infinity")) ||
     475          64 :                         (STARTS_WITH_CI("-Infinity", extendedToken) &&
     476         222 :                          m_osToken.size() + 1 <= strlen("-Infinity")) ||
     477          17 :                         (STARTS_WITH_CI("NaN", extendedToken) &&
     478          13 :                          m_osToken.size() + 1 <= strlen("NaN")))
     479             :                     {
     480         107 :                         m_osToken += ch;
     481             :                     }
     482             :                     else
     483             :                     {
     484           4 :                         return EmitUnexpectedChar(ch);
     485             :                     }
     486             :                 }
     487     6110160 :                 AdvanceChar(pStr, nLength);
     488             :             }
     489             : 
     490      432726 :             if (nLength != 0 || bFinished)
     491             :             {
     492      432515 :                 const char firstCh = m_osToken[0];
     493      432515 :                 if (firstCh == 'i' || firstCh == 'I')
     494             :                 {
     495           5 :                     if (!EQUAL(m_osToken.c_str(), "Infinity"))
     496             :                     {
     497           1 :                         return EmitException("Invalid number");
     498             :                     }
     499             :                 }
     500      432510 :                 else if (firstCh == '-')
     501             :                 {
     502       73483 :                     if (m_osToken[1] == 'i' || m_osToken[1] == 'I')
     503             :                     {
     504           5 :                         if (!EQUAL(m_osToken.c_str(), "-Infinity"))
     505             :                         {
     506           1 :                             return EmitException("Invalid number");
     507             :                         }
     508             :                     }
     509             :                 }
     510      359027 :                 else if (firstCh == 'n' || firstCh == 'N')
     511             :                 {
     512           5 :                     if (m_osToken[1] == 'a' || m_osToken[1] == 'A')
     513             :                     {
     514           5 :                         if (!EQUAL(m_osToken.c_str(), "NaN"))
     515             :                         {
     516           1 :                             return EmitException("Invalid number");
     517             :                         }
     518             :                     }
     519             :                 }
     520             : 
     521      432512 :                 Number(m_osToken.c_str(), m_osToken.size());
     522      432512 :                 m_osToken.clear();
     523      432512 :                 m_aState.pop_back();
     524             :             }
     525             : 
     526      432723 :             if (nLength == 0)
     527             :             {
     528         222 :                 if (bFinished)
     529             :                 {
     530          11 :                     return CheckStackEmpty();
     531             :                 }
     532         211 :                 return true;
     533             :             }
     534             :         }
     535     1632310 :         else if (eCurState == STRING)
     536             :         {
     537       93825 :             bool bEOS = false;
     538      871317 :             while (nLength)
     539             :             {
     540      870393 :                 if (m_osToken.size() == m_nMaxStringSize)
     541             :                 {
     542           1 :                     return EmitException("Too many characters in number");
     543             :                 }
     544             : 
     545      870392 :                 char ch = *pStr;
     546      870392 :                 if (m_bInUnicode)
     547             :                 {
     548          81 :                     if (m_osUnicodeHex.size() == 8)
     549             :                     {
     550           2 :                         DecodeUnicode();
     551             :                     }
     552          79 :                     else if (m_osUnicodeHex.size() == 4)
     553             :                     {
     554             :                         /* Start of next surrogate pair ? */
     555          13 :                         if (m_nLastChar == '\\')
     556             :                         {
     557           4 :                             if (ch == 'u')
     558             :                             {
     559           3 :                                 AdvanceChar(pStr, nLength);
     560           3 :                                 continue;
     561             :                             }
     562             :                             else
     563             :                             {
     564             :                                 /* will be replacement character */
     565           1 :                                 DecodeUnicode();
     566           1 :                                 m_bInStringEscape = true;
     567             :                             }
     568             :                         }
     569           9 :                         else if (m_nLastChar == 'u')
     570             :                         {
     571           3 :                             if (IsHexDigit(ch))
     572             :                             {
     573           2 :                                 m_osUnicodeHex += ch;
     574             :                             }
     575             :                             else
     576             :                             {
     577             :                                 char szMessage[64];
     578           1 :                                 snprintf(szMessage, sizeof(szMessage),
     579             :                                          "Illegal character in unicode "
     580             :                                          "sequence (\\%c)",
     581             :                                          ch);
     582           1 :                                 return EmitException(szMessage);
     583             :                             }
     584           2 :                             AdvanceChar(pStr, nLength);
     585           2 :                             continue;
     586             :                         }
     587           6 :                         else if (ch == '\\')
     588             :                         {
     589           4 :                             AdvanceChar(pStr, nLength);
     590           4 :                             continue;
     591             :                         }
     592             :                         else
     593             :                         {
     594             :                             /* will be replacement character */
     595           2 :                             DecodeUnicode();
     596             :                         }
     597             :                     }
     598             :                     else
     599             :                     {
     600          66 :                         if (IsHexDigit(ch))
     601             :                         {
     602          65 :                             m_osUnicodeHex += ch;
     603          79 :                             if (m_osUnicodeHex.size() == 4 &&
     604          14 :                                 !IsHighSurrogate(getUCSChar(m_osUnicodeHex)))
     605             :                             {
     606           8 :                                 DecodeUnicode();
     607             :                             }
     608             :                         }
     609             :                         else
     610             :                         {
     611             :                             char szMessage[64];
     612           1 :                             snprintf(szMessage, sizeof(szMessage),
     613             :                                      "Illegal character in unicode "
     614             :                                      "sequence (\\%c)",
     615             :                                      ch);
     616           1 :                             return EmitException(szMessage);
     617             :                         }
     618          65 :                         AdvanceChar(pStr, nLength);
     619          65 :                         continue;
     620             :                     }
     621             :                 }
     622             : 
     623      870316 :                 if (m_bInStringEscape)
     624             :                 {
     625         446 :                     if (ch == '"' || ch == '\\' || ch == '/')
     626         417 :                         m_osToken += ch;
     627          29 :                     else if (ch == 'b')
     628           2 :                         m_osToken += '\b';
     629          27 :                     else if (ch == 'f')
     630           2 :                         m_osToken += '\f';
     631          25 :                     else if (ch == 'n')
     632           2 :                         m_osToken += '\n';
     633          23 :                     else if (ch == 'r')
     634           2 :                         m_osToken += '\r';
     635          21 :                     else if (ch == 't')
     636           3 :                         m_osToken += '\t';
     637          18 :                     else if (ch == 'u')
     638             :                     {
     639          17 :                         m_bInUnicode = true;
     640             :                     }
     641             :                     else
     642             :                     {
     643             :                         char szMessage[32];
     644           1 :                         snprintf(szMessage, sizeof(szMessage),
     645             :                                  "Illegal escape sequence (\\%c)", ch);
     646           1 :                         return EmitException(szMessage);
     647             :                     }
     648         445 :                     m_bInStringEscape = false;
     649         445 :                     AdvanceChar(pStr, nLength);
     650         445 :                     continue;
     651             :                 }
     652      869870 :                 else if (ch == '\\')
     653             :                 {
     654         446 :                     m_bInStringEscape = true;
     655         446 :                     AdvanceChar(pStr, nLength);
     656         446 :                     continue;
     657             :                 }
     658      869424 :                 else if (ch == '"')
     659             :                 {
     660       92897 :                     bEOS = true;
     661       92897 :                     AdvanceChar(pStr, nLength);
     662       92897 :                     SkipSpace(pStr, nLength);
     663             : 
     664      185786 :                     if (!m_aeObjectState.empty() &&
     665       92889 :                         m_aeObjectState.back() == IN_KEY)
     666             :                     {
     667       67323 :                         StartObjectMember(m_osToken.c_str(), m_osToken.size());
     668             :                     }
     669             :                     else
     670             :                     {
     671       25574 :                         String(m_osToken.c_str(), m_osToken.size());
     672             :                     }
     673       92897 :                     m_osToken.clear();
     674       92897 :                     m_aState.pop_back();
     675             : 
     676       92897 :                     break;
     677             :                 }
     678             : 
     679      776527 :                 m_osToken += ch;
     680      776527 :                 AdvanceChar(pStr, nLength);
     681             :             }
     682             : 
     683       93821 :             if (nLength == 0)
     684             :             {
     685        1062 :                 if (bFinished)
     686             :                 {
     687          14 :                     if (!bEOS)
     688             :                     {
     689           5 :                         return EmitException("Unterminated string");
     690             :                     }
     691           9 :                     return CheckStackEmpty();
     692             :                 }
     693        1048 :                 return true;
     694             :             }
     695             :         }
     696     1538480 :         else if (eCurState == ARRAY)
     697             :         {
     698     1268110 :             SkipSpace(pStr, nLength);
     699     1268110 :             if (nLength == 0)
     700             :             {
     701         157 :                 if (bFinished)
     702             :                 {
     703           2 :                     return EmitException("Unterminated array");
     704             :                 }
     705         155 :                 return true;
     706             :             }
     707             : 
     708     1267950 :             char ch = *pStr;
     709     1267950 :             if (ch == ',')
     710             :             {
     711      412473 :                 if (m_abArrayState.back() != ArrayState::AFTER_VALUE)
     712             :                 {
     713           3 :                     return EmitUnexpectedChar(ch, "','");
     714             :                 }
     715      412470 :                 m_abArrayState.back() = ArrayState::AFTER_COMMA;
     716      412470 :                 AdvanceChar(pStr, nLength);
     717             :             }
     718      855479 :             else if (ch == ']')
     719             :             {
     720      221529 :                 if (m_abArrayState.back() == ArrayState::AFTER_COMMA)
     721             :                 {
     722           3 :                     return EmitException("Missing value");
     723             :                 }
     724             : 
     725      221526 :                 EndArray();
     726      221526 :                 AdvanceChar(pStr, nLength);
     727      221526 :                 m_abArrayState.pop_back();
     728      221526 :                 m_aState.pop_back();
     729             :             }
     730      633950 :             else if (IsValidNewToken(ch))
     731             :             {
     732      633949 :                 if (m_abArrayState.back() == ArrayState::AFTER_VALUE)
     733             :                 {
     734           1 :                     return EmitException(
     735           1 :                         "Unexpected state: ',' or ']' expected");
     736             :                 }
     737      633948 :                 m_abArrayState.back() = ArrayState::AFTER_VALUE;
     738             : 
     739      633948 :                 StartArrayMember();
     740      633948 :                 if (!StartNewToken(pStr, nLength))
     741             :                 {
     742           0 :                     return false;
     743             :                 }
     744             :             }
     745             :             else
     746             :             {
     747           1 :                 return EmitUnexpectedChar(ch);
     748             :             }
     749             :         }
     750      270373 :         else if (eCurState == OBJECT)
     751             :         {
     752      268704 :             SkipSpace(pStr, nLength);
     753      268704 :             if (nLength == 0)
     754             :             {
     755         464 :                 if (bFinished)
     756             :                 {
     757           2 :                     return EmitException("Unterminated object");
     758             :                 }
     759         462 :                 return true;
     760             :             }
     761             : 
     762      268240 :             char ch = *pStr;
     763      268240 :             if (ch == ',')
     764             :             {
     765       45360 :                 if (m_aeObjectState.back() != IN_VALUE)
     766             :                 {
     767           2 :                     return EmitUnexpectedChar(ch, "','");
     768             :                 }
     769             : 
     770       45358 :                 m_aeObjectState.back() = WAITING_KEY;
     771       45358 :                 AdvanceChar(pStr, nLength);
     772             :             }
     773      222880 :             else if (ch == ':')
     774             :             {
     775       67316 :                 if (m_aeObjectState.back() != IN_KEY)
     776             :                 {
     777           1 :                     return EmitUnexpectedChar(ch, "':'");
     778             :                 }
     779       67315 :                 m_aeObjectState.back() = KEY_FINISHED;
     780       67315 :                 AdvanceChar(pStr, nLength);
     781             :             }
     782      155564 :             else if (ch == '}')
     783             :             {
     784       40973 :                 if (m_aeObjectState.back() == WAITING_KEY ||
     785       20050 :                     m_aeObjectState.back() == IN_VALUE)
     786             :                 {
     787             :                     // nothing
     788             :                 }
     789             :                 else
     790             :                 {
     791           1 :                     return EmitException("Missing value");
     792             :                 }
     793             : 
     794       20922 :                 EndObject();
     795       20922 :                 AdvanceChar(pStr, nLength);
     796       20922 :                 m_aeObjectState.pop_back();
     797       20922 :                 m_aState.pop_back();
     798             :             }
     799      134641 :             else if (IsValidNewToken(ch))
     800             :             {
     801      134639 :                 if (m_aeObjectState.back() == WAITING_KEY)
     802             :                 {
     803       67324 :                     if (ch != '"')
     804             :                     {
     805           1 :                         return EmitUnexpectedChar(ch, "'\"'");
     806             :                     }
     807       67323 :                     m_aeObjectState.back() = IN_KEY;
     808             :                 }
     809       67315 :                 else if (m_aeObjectState.back() == KEY_FINISHED)
     810             :                 {
     811       67314 :                     m_aeObjectState.back() = IN_VALUE;
     812             :                 }
     813             :                 else
     814             :                 {
     815           1 :                     return EmitException("Unexpected state");
     816             :                 }
     817      134637 :                 if (!StartNewToken(pStr, nLength))
     818             :                 {
     819           0 :                     return false;
     820             :                 }
     821             :             }
     822             :             else
     823             :             {
     824           2 :                 return EmitUnexpectedChar(ch);
     825             :             }
     826             :         }
     827             :         else /* if( eCurState == STATE_TRUE || eCurState == STATE_FALSE ||
     828             :                     eCurState == STATE_NULL ) */
     829             :         {
     830        7817 :             while (nLength)
     831             :             {
     832        7697 :                 char ch = *pStr;
     833        7701 :                 if (eCurState == STATE_NULL && (ch == 'a' || ch == 'A') &&
     834           4 :                     m_osToken.size() == 1)
     835             :                 {
     836           4 :                     m_aState.back() = NUMBER;
     837           4 :                     break;
     838             :                 }
     839        7693 :                 if (isalpha(static_cast<unsigned char>(ch)))
     840             :                 {
     841        6154 :                     m_osToken += ch;
     842        7447 :                     if (eCurState == STATE_TRUE &&
     843         647 :                         (m_osToken.size() > strlen("true") ||
     844         646 :                          memcmp(m_osToken.c_str(), "true", m_osToken.size()) !=
     845             :                              0))
     846             :                     {
     847           2 :                         return EmitUnexpectedChar(*pStr);
     848             :                     }
     849        6261 :                     else if (eCurState == STATE_FALSE &&
     850          55 :                              (m_osToken.size() > strlen("false") ||
     851          54 :                               memcmp(m_osToken.c_str(), "false",
     852             :                                      m_osToken.size()) != 0))
     853             :                     {
     854           2 :                         return EmitUnexpectedChar(*pStr);
     855             :                     }
     856       17053 :                     else if (eCurState == STATE_NULL &&
     857        5452 :                              (m_osToken.size() > strlen("null") ||
     858        5451 :                               memcmp(m_osToken.c_str(), "null",
     859             :                                      m_osToken.size()) != 0))
     860             :                     {
     861           2 :                         return EmitUnexpectedChar(*pStr);
     862             :                     }
     863             :                 }
     864        1539 :                 else if (isspace(static_cast<unsigned char>(ch)) || ch == ',' ||
     865          51 :                          ch == '}' || ch == ']')
     866             :                 {
     867        1538 :                     SkipSpace(pStr, nLength);
     868        1538 :                     break;
     869             :                 }
     870             :                 else
     871             :                 {
     872           1 :                     return EmitUnexpectedChar(ch);
     873             :                 }
     874        6148 :                 AdvanceChar(pStr, nLength);
     875             :             }
     876        1662 :             if (m_aState.back() == NUMBER)
     877             :             {
     878           4 :                 continue;
     879             :             }
     880        1658 :             if (nLength == 0)
     881             :             {
     882         139 :                 if (bFinished)
     883             :                 {
     884           9 :                     if (!CheckAndEmitTrueFalseOrNull(0))
     885           3 :                         return false;
     886           6 :                     return CheckStackEmpty();
     887             :                 }
     888         130 :                 return true;
     889             :             }
     890             : 
     891        1519 :             if (!CheckAndEmitTrueFalseOrNull(*pStr))
     892           0 :                 return false;
     893             :         }
     894     2065730 :     }
     895             : }
     896             : 
     897             : /************************************************************************/
     898             : /*                       GetSerializedString()                          */
     899             : /************************************************************************/
     900             : 
     901       11174 : std::string CPLJSonStreamingParser::GetSerializedString(const char *pszStr)
     902             : {
     903       11174 :     std::string osStr("\"");
     904       87542 :     for (int i = 0; pszStr[i]; i++)
     905             :     {
     906       76368 :         char ch = pszStr[i];
     907       76368 :         if (ch == '\b')
     908           2 :             osStr += "\\b";
     909       76366 :         else if (ch == '\f')
     910           2 :             osStr += "\\f";
     911       76364 :         else if (ch == '\n')
     912           2 :             osStr += "\\n";
     913       76362 :         else if (ch == '\r')
     914           2 :             osStr += "\\r";
     915       76360 :         else if (ch == '\t')
     916           3 :             osStr += "\\t";
     917       76357 :         else if (ch == '"')
     918           2 :             osStr += "\\\"";
     919       76355 :         else if (ch == '\\')
     920           2 :             osStr += "\\\\";
     921       76353 :         else if (static_cast<unsigned char>(ch) < ' ')
     922           3 :             osStr += CPLSPrintf("\\u%04X", ch);
     923             :         else
     924       76350 :             osStr += ch;
     925             :     }
     926       11174 :     osStr += "\"";
     927       11174 :     return osStr;
     928             : }
     929             : 
     930             : /*! @endcond */

Generated by: LCOV version 1.14