/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- * vim: set ts=8 sts=2 et sw=2 tw=80: * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "vm/JSONParser.h" #include "mozilla/Assertions.h" // MOZ_ASSERT #include "mozilla/Attributes.h" // MOZ_STACK_CLASS #include "mozilla/Range.h" // mozilla::Range #include "mozilla/RangedPtr.h" // mozilla::RangedPtr #include "mozilla/Sprintf.h" // SprintfLiteral #include "mozilla/TextUtils.h" // mozilla::AsciiAlphanumericToNumber, mozilla::IsAsciiDigit, mozilla::IsAsciiHexDigit #include // size_t #include // uint32_t #include // std::move #include "jsnum.h" // ParseDecimalNumber, GetFullInteger, FullStringToDouble #include "builtin/Array.h" // NewDenseCopiedArray #include "ds/IdValuePair.h" // IdValuePair #include "gc/Allocator.h" // CanGC #include "gc/Tracer.h" // JS::TraceRoot #include "js/AllocPolicy.h" // ReportOutOfMemory #include "js/ErrorReport.h" // JS_ReportErrorNumberASCII #include "js/friend/ErrorMessages.h" // js::GetErrorMessage, JSMSG_* #include "js/GCVector.h" // JS::GCVector #include "js/Id.h" // jsid #include "js/JSON.h" // JS::IsValidJSON #include "js/RootingAPI.h" // JS::Handle, JS::MutableHandle, MutableWrappedPtrOperations #include "js/TypeDecls.h" // Latin1Char #include "js/Utility.h" // js_delete #include "js/Value.h" // JS::Value, JS::BooleanValue, JS::NullValue, JS::NumberValue, JS::StringValue #include "js/Vector.h" // Vector #include "util/StringBuffer.h" // JSStringBuilder #include "vm/ArrayObject.h" // ArrayObject #include "vm/ErrorReporting.h" // ReportCompileErrorLatin1, ErrorMetadata #include "vm/JSAtom.h" // AtomizeChars #include "vm/JSContext.h" // JSContext #include "vm/PlainObject.h" // NewPlainObjectWithMaybeDuplicateKeys #include "vm/StringType.h" // JSString, JSAtom, JSLinearString, NewStringCopyN, NameToId #include "vm/JSAtom-inl.h" // AtomToId using namespace js; using mozilla::AsciiAlphanumericToNumber; using mozilla::IsAsciiDigit; using mozilla::IsAsciiHexDigit; using mozilla::RangedPtr; template template JSONToken JSONTokenizer::stringToken( const CharPtr start, size_t length) { if (!parser->handler.template setStringValue(start, length)) { return JSONToken::OOM; } return JSONToken::String; } template template JSONToken JSONTokenizer::stringToken( StringBuilderT& builder) { if (!parser->handler.template setStringValue(builder)) { return JSONToken::OOM; } return JSONToken::String; } template JSONToken JSONTokenizer::numberToken(double d) { parser->handler.setNumberValue(d); return JSONToken::Number; } template template JSONToken JSONTokenizer::readString() { MOZ_ASSERT(current < end); MOZ_ASSERT(*current == '"'); /* * JSONString: * /^"([^\u0000-\u001F"\\]|\\(["/\\bfnrt]|u[0-9a-fA-F]{4}))*"$/ */ if (++current == end) { error("unterminated string literal"); return token(JSONToken::Error); } /* * Optimization: if the source contains no escaped characters, create the * string directly from the source text. */ CharPtr start = current; for (; current < end; current++) { if (*current == '"') { size_t length = current - start; current++; return stringToken(start, length); } if (*current == '\\') { break; } if (*current <= 0x001F) { error("bad control character in string literal"); return token(JSONToken::Error); } } /* * Slow case: string contains escaped characters. Copy a maximal sequence * of unescaped characters into a temporary buffer, then an escaped * character, and repeat until the entire string is consumed. */ StringBuilderT builder(parser->handler.context()); do { if (start < current && !builder.append(start.get(), current.get())) { return token(JSONToken::OOM); } if (current >= end) { break; } char16_t c = *current++; if (c == '"') { return stringToken(builder); } if (c != '\\') { --current; error("bad character in string literal"); return token(JSONToken::Error); } if (current >= end) { break; } switch (*current++) { case '"': c = '"'; break; case '/': c = '/'; break; case '\\': c = '\\'; break; case 'b': c = '\b'; break; case 'f': c = '\f'; break; case 'n': c = '\n'; break; case 'r': c = '\r'; break; case 't': c = '\t'; break; case 'u': if (end - current < 4 || !(IsAsciiHexDigit(current[0]) && IsAsciiHexDigit(current[1]) && IsAsciiHexDigit(current[2]) && IsAsciiHexDigit(current[3]))) { // Point to the first non-hexadecimal character (which may be // missing). if (current == end || !IsAsciiHexDigit(current[0])) { ; // already at correct location } else if (current + 1 == end || !IsAsciiHexDigit(current[1])) { current += 1; } else if (current + 2 == end || !IsAsciiHexDigit(current[2])) { current += 2; } else if (current + 3 == end || !IsAsciiHexDigit(current[3])) { current += 3; } else { MOZ_CRASH("logic error determining first erroneous character"); } error("bad Unicode escape"); return token(JSONToken::Error); } c = (AsciiAlphanumericToNumber(current[0]) << 12) | (AsciiAlphanumericToNumber(current[1]) << 8) | (AsciiAlphanumericToNumber(current[2]) << 4) | (AsciiAlphanumericToNumber(current[3])); current += 4; break; default: current--; error("bad escaped character"); return token(JSONToken::Error); } if (!builder.append(c)) { return token(JSONToken::OOM); } start = current; for (; current < end; current++) { if (*current == '"' || *current == '\\' || *current <= 0x001F) { break; } } } while (current < end); error("unterminated string"); return token(JSONToken::Error); } template JSONToken JSONTokenizer::readNumber() { MOZ_ASSERT(current < end); MOZ_ASSERT(IsAsciiDigit(*current) || *current == '-'); /* * JSONNumber: * /^-?(0|[1-9][0-9]+)(\.[0-9]+)?([eE][\+\-]?[0-9]+)?$/ */ bool negative = *current == '-'; /* -? */ if (negative && ++current == end) { error("no number after minus sign"); return token(JSONToken::Error); } const CharPtr digitStart = current; /* 0|[1-9][0-9]+ */ if (!IsAsciiDigit(*current)) { error("unexpected non-digit"); return token(JSONToken::Error); } if (*current++ != '0') { for (; current < end; current++) { if (!IsAsciiDigit(*current)) { break; } } } /* Fast path: no fractional or exponent part. */ if (current == end || (*current != '.' && *current != 'e' && *current != 'E')) { mozilla::Range chars(digitStart.get(), current - digitStart); if (chars.length() < strlen("9007199254740992")) { // If the decimal number is shorter than the length of 2**53, (the // largest number a double can represent with integral precision), // parse it using a decimal-only parser. This comparison is // conservative but faster than a fully-precise check. double d = ParseDecimalNumber(chars); return numberToken(negative ? -d : d); } double d; if (!GetFullInteger(digitStart.get(), current.get(), 10, IntegerSeparatorHandling::None, &d)) { parser->outOfMemory(); return token(JSONToken::OOM); } return numberToken(negative ? -d : d); } /* (\.[0-9]+)? */ if (current < end && *current == '.') { if (++current == end) { error("missing digits after decimal point"); return token(JSONToken::Error); } if (!IsAsciiDigit(*current)) { error("unterminated fractional number"); return token(JSONToken::Error); } while (++current < end) { if (!IsAsciiDigit(*current)) { break; } } } /* ([eE][\+\-]?[0-9]+)? */ if (current < end && (*current == 'e' || *current == 'E')) { if (++current == end) { error("missing digits after exponent indicator"); return token(JSONToken::Error); } if (*current == '+' || *current == '-') { if (++current == end) { error("missing digits after exponent sign"); return token(JSONToken::Error); } } if (!IsAsciiDigit(*current)) { error("exponent part is missing a number"); return token(JSONToken::Error); } while (++current < end) { if (!IsAsciiDigit(*current)) { break; } } } double d = FullStringToDouble(digitStart.get(), current.get()); return numberToken(negative ? -d : d); } static inline bool IsJSONWhitespace(char16_t c) { return c == '\t' || c == '\r' || c == '\n' || c == ' '; } template bool JSONTokenizer::consumeTrailingWhitespaces() { for (; current < end; current++) { if (!IsJSONWhitespace(*current)) { return false; } } return true; } template JSONToken JSONTokenizer::advance() { while (current < end && IsJSONWhitespace(*current)) { current++; } if (current >= end) { error("unexpected end of data"); return token(JSONToken::Error); } switch (*current) { case '"': return readString(); case '-': case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9': return readNumber(); case 't': if (end - current < 4 || current[1] != 'r' || current[2] != 'u' || current[3] != 'e') { error("unexpected keyword"); return token(JSONToken::Error); } current += 4; return token(JSONToken::True); case 'f': if (end - current < 5 || current[1] != 'a' || current[2] != 'l' || current[3] != 's' || current[4] != 'e') { error("unexpected keyword"); return token(JSONToken::Error); } current += 5; return token(JSONToken::False); case 'n': if (end - current < 4 || current[1] != 'u' || current[2] != 'l' || current[3] != 'l') { error("unexpected keyword"); return token(JSONToken::Error); } current += 4; return token(JSONToken::Null); case '[': current++; return token(JSONToken::ArrayOpen); case ']': current++; return token(JSONToken::ArrayClose); case '{': current++; return token(JSONToken::ObjectOpen); case '}': current++; return token(JSONToken::ObjectClose); case ',': current++; return token(JSONToken::Comma); case ':': current++; return token(JSONToken::Colon); default: error("unexpected character"); return token(JSONToken::Error); } } template JSONToken JSONTokenizer::advancePropertyName() { MOZ_ASSERT(current[-1] == ','); while (current < end && IsJSONWhitespace(*current)) { current++; } if (current >= end) { error("end of data when property name was expected"); return token(JSONToken::Error); } if (*current == '"') { return readString(); } error("expected double-quoted property name"); return token(JSONToken::Error); } template JSONToken JSONTokenizer::advancePropertyColon() { MOZ_ASSERT(current[-1] == '"'); while (current < end && IsJSONWhitespace(*current)) { current++; } if (current >= end) { error("end of data after property name when ':' was expected"); return token(JSONToken::Error); } if (*current == ':') { current++; return token(JSONToken::Colon); } error("expected ':' after property name in object"); return token(JSONToken::Error); } template static inline void AssertPastValue(const RangedPtr current) { /* * We're past an arbitrary JSON value, so the previous character is * *somewhat* constrained, even if this assertion is pretty broad. Don't * knock it till you tried it: this assertion *did* catch a bug once. */ MOZ_ASSERT((current[-1] == 'l' && current[-2] == 'l' && current[-3] == 'u' && current[-4] == 'n') || (current[-1] == 'e' && current[-2] == 'u' && current[-3] == 'r' && current[-4] == 't') || (current[-1] == 'e' && current[-2] == 's' && current[-3] == 'l' && current[-4] == 'a' && current[-5] == 'f') || current[-1] == '}' || current[-1] == ']' || current[-1] == '"' || IsAsciiDigit(current[-1])); } template JSONToken JSONTokenizer::advanceAfterProperty() { AssertPastValue(current); while (current < end && IsJSONWhitespace(*current)) { current++; } if (current >= end) { error("end of data after property value in object"); return token(JSONToken::Error); } if (*current == ',') { current++; return token(JSONToken::Comma); } if (*current == '}') { current++; return token(JSONToken::ObjectClose); } error("expected ',' or '}' after property value in object"); return token(JSONToken::Error); } template JSONToken JSONTokenizer::advanceAfterObjectOpen() { MOZ_ASSERT(current[-1] == '{'); while (current < end && IsJSONWhitespace(*current)) { current++; } if (current >= end) { error("end of data while reading object contents"); return token(JSONToken::Error); } if (*current == '"') { return readString(); } if (*current == '}') { current++; return token(JSONToken::ObjectClose); } error("expected property name or '}'"); return token(JSONToken::Error); } template JSONToken JSONTokenizer::advanceAfterArrayElement() { AssertPastValue(current); while (current < end && IsJSONWhitespace(*current)) { current++; } if (current >= end) { error("end of data when ',' or ']' was expected"); return token(JSONToken::Error); } if (*current == ',') { current++; return token(JSONToken::Comma); } if (*current == ']') { current++; return token(JSONToken::ArrayClose); } error("expected ',' or ']' after array element"); return token(JSONToken::Error); } template void JSONTokenizer::error(const char* msg) { parser->error(msg); } template void JSONTokenizer::getTextPosition( uint32_t* column, uint32_t* line) { CharPtr ptr = begin; uint32_t col = 1; uint32_t row = 1; for (; ptr < current; ptr++) { if (*ptr == '\n' || *ptr == '\r') { ++row; col = 1; // \r\n is treated as a single newline. if (ptr + 1 < current && *ptr == '\r' && *(ptr + 1) == '\n') { ++ptr; } } else { ++col; } } *column = col; *line = row; } JSONFullParseHandlerAnyChar::~JSONFullParseHandlerAnyChar() { for (size_t i = 0; i < freeElements.length(); i++) { js_delete(freeElements[i]); } for (size_t i = 0; i < freeProperties.length(); i++) { js_delete(freeProperties[i]); } } void JSONFullParseHandlerAnyChar::trace(JSTracer* trc) { JS::TraceRoot(trc, &v, "JSONFullParseHandlerAnyChar current value"); } inline void JSONFullParseHandlerAnyChar::freeStackEntry(StackEntry& entry) { if (entry.state == JSONParserState::FinishArrayElement) { js_delete(&entry.elements()); } else { js_delete(&entry.properties()); } } template void JSONParser::trace(JSTracer* trc) { this->handler.trace(trc); for (auto& elem : this->stack) { if (elem.state == JSONParserState::FinishArrayElement) { elem.elements().trace(trc); } else { elem.properties().trace(trc); } } } inline void JSONFullParseHandlerAnyChar::setNumberValue(double d) { v = JS::NumberValue(d); } template template inline bool JSONFullParseHandler::setStringValue(CharPtr start, size_t length) { JSLinearString* str = (ST == JSONStringType::PropertyName) ? AtomizeChars(cx, start.get(), length) : NewStringCopyN(cx, start.get(), length); if (!str) { return false; } v = JS::StringValue(str); return true; } template template inline bool JSONFullParseHandler::setStringValue( StringBuilder& builder) { JSLinearString* str = (ST == JSONStringType::PropertyName) ? builder.buffer.finishAtom() : builder.buffer.finishString(); if (!str) { return false; } v = JS::StringValue(str); return true; } inline bool JSONFullParseHandlerAnyChar::objectOpen( Vector& stack, PropertyVector** properties) { if (!freeProperties.empty()) { *properties = freeProperties.popCopy(); (*properties)->clear(); } else { (*properties) = cx->new_(cx); if (!*properties) { return false; } } if (!stack.append(*properties)) { js_delete(*properties); return false; } return true; } inline bool JSONFullParseHandlerAnyChar::objectPropertyName( Vector& stack, bool* isProtoInEval) { *isProtoInEval = false; jsid id = AtomToId(atomValue()); if (parseType == ParseType::AttemptForEval) { // In |JSON.parse|, "__proto__" is a property like any other and may // appear multiple times. In object literal syntax, "__proto__" is // prototype mutation and can appear at most once. |JSONParser| only // supports the former semantics, so if this parse attempt is for // |eval|, return true (without reporting an error) to indicate the // JSON parse attempt was unsuccessful. if (id == NameToId(cx->names().proto)) { *isProtoInEval = true; return true; } } PropertyVector& properties = stack.back().properties(); if (!properties.emplaceBack(id)) { return false; } return true; } inline void JSONFullParseHandlerAnyChar::finishObjectMember( Vector& stack, JS::Handle value, PropertyVector** properties) { *properties = &stack.back().properties(); (*properties)->back().value = value; } inline bool JSONFullParseHandlerAnyChar::finishObject( Vector& stack, JS::MutableHandle vp, PropertyVector& properties) { MOZ_ASSERT(&properties == &stack.back().properties()); JSObject* obj = NewPlainObjectWithMaybeDuplicateKeys(cx, properties.begin(), properties.length()); if (!obj) { return false; } vp.setObject(*obj); if (!freeProperties.append(&properties)) { return false; } stack.popBack(); return true; } inline bool JSONFullParseHandlerAnyChar::arrayOpen( Vector& stack, ElementVector** elements) { if (!freeElements.empty()) { *elements = freeElements.popCopy(); (*elements)->clear(); } else { (*elements) = cx->new_(cx); if (!*elements) { return false; } } if (!stack.append(*elements)) { js_delete(*elements); return false; } return true; } inline bool JSONFullParseHandlerAnyChar::arrayElement( Vector& stack, JS::Handle value, ElementVector** elements) { *elements = &stack.back().elements(); return (*elements)->append(value.get()); } inline bool JSONFullParseHandlerAnyChar::finishArray( Vector& stack, JS::MutableHandle vp, ElementVector& elements) { MOZ_ASSERT(&elements == &stack.back().elements()); ArrayObject* obj = NewDenseCopiedArray(cx, elements.length(), elements.begin()); if (!obj) { return false; } vp.setObject(*obj); if (!freeElements.append(&elements)) { return false; } stack.popBack(); return true; } template void JSONFullParseHandler::reportError(const char* msg, const char* lineString, const char* columnString) { JS_ReportErrorNumberASCII(cx, GetErrorMessage, nullptr, JSMSG_JSON_BAD_PARSE, msg, lineString, columnString); } template void JSONPerHandlerParser::outOfMemory() { ReportOutOfMemory(handler.context()); } template void JSONPerHandlerParser::error(const char* msg) { if (handler.ignoreError()) { return; } uint32_t column = 1, line = 1; tokenizer.getTextPosition(&column, &line); const size_t MaxWidth = sizeof("4294967295"); char columnNumber[MaxWidth]; SprintfLiteral(columnNumber, "%" PRIu32, column); char lineNumber[MaxWidth]; SprintfLiteral(lineNumber, "%" PRIu32, line); handler.reportError(msg, lineNumber, columnNumber); } template bool JSONFullParseHandler::StringBuilder::append(char16_t c) { return buffer.append(c); } template bool JSONFullParseHandler::StringBuilder::append(const CharT* begin, const CharT* end) { return buffer.append(begin, end); } template JSONPerHandlerParser::~JSONPerHandlerParser() { for (size_t i = 0; i < stack.length(); i++) { handler.freeStackEntry(stack[i]); } } template class js::JSONPerHandlerParser>; template class js::JSONPerHandlerParser>; template template bool JSONPerHandlerParser::parseImpl(TempValueT& value, ResultSetter setResult) { MOZ_ASSERT(stack.empty()); JSONToken token; JSONParserState state = JSONParserState::JSONValue; while (true) { switch (state) { case JSONParserState::FinishObjectMember: { typename HandlerT::PropertyVector* properties; handler.finishObjectMember(stack, value, &properties); token = tokenizer.advanceAfterProperty(); if (token == JSONToken::ObjectClose) { if (!handler.finishObject(stack, &value, *properties)) { return false; } break; } if (token != JSONToken::Comma) { if (token == JSONToken::OOM) { return false; } if (token != JSONToken::Error) { error( "expected ',' or '}' after property-value pair in object " "literal"); } return handler.errorReturn(); } token = tokenizer.advancePropertyName(); /* FALL THROUGH */ } JSONMember: if (token == JSONToken::String) { bool isProtoInEval; if (!handler.objectPropertyName(stack, &isProtoInEval)) { return false; } if (isProtoInEval) { // See JSONFullParseHandlerAnyChar::objectPropertyName. return true; } token = tokenizer.advancePropertyColon(); if (token != JSONToken::Colon) { MOZ_ASSERT(token == JSONToken::Error); return handler.errorReturn(); } goto JSONValue; } if (token == JSONToken::OOM) { return false; } if (token != JSONToken::Error) { error("property names must be double-quoted strings"); } return handler.errorReturn(); case JSONParserState::FinishArrayElement: { typename HandlerT::ElementVector* elements; if (!handler.arrayElement(stack, value, &elements)) { return false; } token = tokenizer.advanceAfterArrayElement(); if (token == JSONToken::Comma) { goto JSONValue; } if (token == JSONToken::ArrayClose) { if (!handler.finishArray(stack, &value, *elements)) { return false; } break; } MOZ_ASSERT(token == JSONToken::Error); return handler.errorReturn(); } JSONValue: case JSONParserState::JSONValue: token = tokenizer.advance(); JSONValueSwitch: switch (token) { case JSONToken::String: value = handler.stringValue(); break; case JSONToken::Number: value = handler.numberValue(); break; case JSONToken::True: value = handler.booleanValue(true); break; case JSONToken::False: value = handler.booleanValue(false); break; case JSONToken::Null: value = handler.nullValue(); break; case JSONToken::ArrayOpen: { typename HandlerT::ElementVector* elements; if (!handler.arrayOpen(stack, &elements)) { return false; } token = tokenizer.advance(); if (token == JSONToken::ArrayClose) { if (!handler.finishArray(stack, &value, *elements)) { return false; } break; } goto JSONValueSwitch; } case JSONToken::ObjectOpen: { typename HandlerT::PropertyVector* properties; if (!handler.objectOpen(stack, &properties)) { return false; } token = tokenizer.advanceAfterObjectOpen(); if (token == JSONToken::ObjectClose) { if (!handler.finishObject(stack, &value, *properties)) { return false; } break; } goto JSONMember; } case JSONToken::ArrayClose: case JSONToken::ObjectClose: case JSONToken::Colon: case JSONToken::Comma: // Move the current pointer backwards so that the position // reported in the error message is correct. tokenizer.unget(); error("unexpected character"); return handler.errorReturn(); case JSONToken::OOM: return false; case JSONToken::Error: return handler.errorReturn(); } break; } if (stack.empty()) { break; } state = stack.back().state; } if (!tokenizer.consumeTrailingWhitespaces()) { error("unexpected non-whitespace character after JSON data"); return handler.errorReturn(); } MOZ_ASSERT(tokenizer.finished()); MOZ_ASSERT(stack.empty()); setResult(value); return true; } template bool JSONParser::parse(JS::MutableHandle vp) { JS::Rooted tempValue(this->handler.cx); vp.setUndefined(); return this->parseImpl(tempValue, [&](JS::Handle value) { vp.set(value); }); } template class js::JSONParser; template class js::JSONParser; template inline bool JSONSyntaxParseHandler::objectOpen( Vector& stack, PropertyVector** properties) { StackEntry entry{JSONParserState::FinishObjectMember}; if (!stack.append(entry)) { return false; } return true; } template inline bool JSONSyntaxParseHandler::finishObject( Vector& stack, DummyValue* vp, PropertyVector& properties) { stack.popBack(); return true; } template inline bool JSONSyntaxParseHandler::arrayOpen( Vector& stack, ElementVector** elements) { StackEntry entry{JSONParserState::FinishArrayElement}; if (!stack.append(entry)) { return false; } return true; } template inline bool JSONSyntaxParseHandler::finishArray( Vector& stack, DummyValue* vp, ElementVector& elements) { stack.popBack(); return true; } static void ReportJSONSyntaxError(FrontendContext* fc, ErrorMetadata&& metadata, unsigned errorNumber, ...) { va_list args; va_start(args, errorNumber); js::ReportCompileErrorLatin1(fc, std::move(metadata), nullptr, errorNumber, &args); va_end(args); } template void JSONSyntaxParseHandler::reportError(const char* msg, const char* lineString, const char* columnString) { ErrorMetadata metadata; metadata.isMuted = false; metadata.filename = ""; metadata.lineNumber = 0; metadata.columnNumber = 0; ReportJSONSyntaxError(fc, std::move(metadata), JSMSG_JSON_BAD_PARSE, msg, lineString, columnString); } template class js::JSONSyntaxParseHandler; template class js::JSONSyntaxParseHandler; template bool JSONSyntaxParser::parse() { typename HandlerT::DummyValue unused; if (!this->parseImpl(unused, [&](const typename HandlerT::DummyValue& unused) {})) { return false; } return true; } template class js::JSONPerHandlerParser>; template class js::JSONPerHandlerParser>; template class js::JSONSyntaxParser; template class js::JSONSyntaxParser; template static bool IsValidJSONImpl(const CharT* chars, uint32_t len) { FrontendContext fc; // NOTE: We don't set stack quota here because JSON parser doesn't use it. JSONSyntaxParser parser(&fc, mozilla::Range(chars, len)); if (!parser.parse()) { MOZ_ASSERT(fc.hadErrors()); return false; } MOZ_ASSERT(!fc.hadErrors()); return true; } JS_PUBLIC_API bool JS::IsValidJSON(const JS::Latin1Char* chars, uint32_t len) { return IsValidJSONImpl(chars, len); } JS_PUBLIC_API bool JS::IsValidJSON(const char16_t* chars, uint32_t len) { return IsValidJSONImpl(chars, len); }