summaryrefslogtreecommitdiffstats
path: root/js/src/vm/JSONParser.h
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--js/src/vm/JSONParser.h517
1 files changed, 517 insertions, 0 deletions
diff --git a/js/src/vm/JSONParser.h b/js/src/vm/JSONParser.h
new file mode 100644
index 0000000000..7c86d3e087
--- /dev/null
+++ b/js/src/vm/JSONParser.h
@@ -0,0 +1,517 @@
+/* -*- 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/. */
+
+#ifndef vm_JSONParser_h
+#define 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 <stddef.h> // size_t
+#include <stdint.h> // uint32_t
+#include <utility> // std::move
+
+#include "ds/IdValuePair.h" // IdValuePair
+#include "js/GCVector.h" // JS::GCVector
+#include "js/RootingAPI.h" // JS::Handle, JS::MutableHandle, MutableWrappedPtrOperations
+#include "js/Value.h" // JS::Value, JS::BooleanValue, JS::NullValue
+#include "js/Vector.h" // Vector
+#include "util/StringBuffer.h" // JSStringBuilder
+#include "vm/StringType.h" // JSString, JSAtom
+
+struct JSContext;
+class JSTracer;
+
+namespace js {
+
+class FrontendContext;
+
+enum class JSONToken {
+ String,
+ Number,
+ True,
+ False,
+ Null,
+ ArrayOpen,
+ ArrayClose,
+ ObjectOpen,
+ ObjectClose,
+ Colon,
+ Comma,
+ OOM,
+ Error
+};
+
+enum class JSONStringType { PropertyName, LiteralValue };
+
+template <typename CharT, typename ParserT, typename StringBuilderT>
+class MOZ_STACK_CLASS JSONTokenizer {
+ public:
+ using CharPtr = mozilla::RangedPtr<const CharT>;
+
+ protected:
+ CharPtr current;
+ const CharPtr begin, end;
+
+ ParserT* parser = nullptr;
+
+ public:
+ JSONTokenizer(CharPtr current, const CharPtr begin, const CharPtr end,
+ ParserT* parser)
+ : current(current), begin(begin), end(end), parser(parser) {
+ MOZ_ASSERT(current <= end);
+ MOZ_ASSERT(parser);
+ }
+
+ explicit JSONTokenizer(mozilla::Range<const CharT> data, ParserT* parser)
+ : JSONTokenizer(data.begin(), data.begin(), data.end(), parser) {}
+
+ JSONTokenizer(JSONTokenizer<CharT, ParserT, StringBuilderT>&& other) noexcept
+ : JSONTokenizer(other.current, other.begin, other.end, other.parser) {}
+
+ JSONTokenizer(const JSONTokenizer<CharT, ParserT, StringBuilderT>& other) =
+ delete;
+ void operator=(const JSONTokenizer<CharT, ParserT, StringBuilderT>& other) =
+ delete;
+
+ void fixupParser(ParserT* newParser) { parser = newParser; }
+
+ void getTextPosition(uint32_t* column, uint32_t* line);
+
+ bool consumeTrailingWhitespaces();
+
+ JSONToken advance();
+ JSONToken advancePropertyName();
+ JSONToken advancePropertyColon();
+ JSONToken advanceAfterProperty();
+ JSONToken advanceAfterObjectOpen();
+ JSONToken advanceAfterArrayElement();
+
+ void unget() { --current; }
+
+#ifdef DEBUG
+ bool finished() { return end == current; }
+#endif
+
+ JSONToken token(JSONToken t) {
+ MOZ_ASSERT(t != JSONToken::String);
+ MOZ_ASSERT(t != JSONToken::Number);
+ return t;
+ }
+
+ template <JSONStringType ST>
+ JSONToken stringToken(const CharPtr start, size_t length);
+ template <JSONStringType ST>
+ JSONToken stringToken(StringBuilderT& builder);
+
+ JSONToken numberToken(double d);
+
+ template <JSONStringType ST>
+ JSONToken readString();
+
+ JSONToken readNumber();
+
+ void error(const char* msg);
+};
+
+// Possible states the parser can be in between values.
+enum class JSONParserState {
+ // An array element has just being parsed.
+ FinishArrayElement,
+
+ // An object property has just been parsed.
+ FinishObjectMember,
+
+ // At the start of the parse, before any values have been processed.
+ JSONValue
+};
+
+// Character-type-agnostic base class for JSONFullParseHandler.
+// JSONParser is templatized to work on either Latin1
+// or TwoByte input strings, JSONFullParseHandlerAnyChar holds all state and
+// methods that can be shared between the two encodings.
+class MOZ_STACK_CLASS JSONFullParseHandlerAnyChar {
+ public:
+ // State related to the parser's current position. At all points in the
+ // parse this keeps track of the stack of arrays and objects which have
+ // been started but not finished yet. The actual JS object is not
+ // allocated until the literal is closed, so that the result can be sized
+ // according to its contents and have its type and shape filled in using
+ // caches.
+
+ // State for an array that is currently being parsed. This includes all
+ // elements that have been seen so far.
+ using ElementVector = JS::GCVector<JS::Value, 20>;
+
+ // State for an object that is currently being parsed. This includes all
+ // the key/value pairs that have been seen so far.
+ using PropertyVector = JS::GCVector<IdValuePair, 10>;
+
+ enum class ParseType {
+ // Parsing a string as if by JSON.parse.
+ JSONParse,
+ // Parsing what may or may not be JSON in a string of eval code.
+ // In this case, a failure to parse indicates either syntax that isn't JSON,
+ // or syntax that has different semantics in eval code than in JSON.
+ AttemptForEval,
+ };
+
+ // Stack element for an in progress array or object.
+ struct StackEntry {
+ ElementVector& elements() {
+ MOZ_ASSERT(state == JSONParserState::FinishArrayElement);
+ return *static_cast<ElementVector*>(vector);
+ }
+
+ PropertyVector& properties() {
+ MOZ_ASSERT(state == JSONParserState::FinishObjectMember);
+ return *static_cast<PropertyVector*>(vector);
+ }
+
+ explicit StackEntry(ElementVector* elements)
+ : state(JSONParserState::FinishArrayElement), vector(elements) {}
+
+ explicit StackEntry(PropertyVector* properties)
+ : state(JSONParserState::FinishObjectMember), vector(properties) {}
+
+ JSONParserState state;
+
+ private:
+ void* vector;
+ };
+
+ public:
+ /* Data members */
+
+ JSContext* cx;
+
+ JS::Value v;
+
+ ParseType parseType = ParseType::JSONParse;
+
+ private:
+ // Unused element and property vectors for previous in progress arrays and
+ // objects. These vectors are not freed until the end of the parse to avoid
+ // unnecessary freeing and allocation.
+ Vector<ElementVector*, 5> freeElements;
+ Vector<PropertyVector*, 5> freeProperties;
+
+ public:
+ explicit JSONFullParseHandlerAnyChar(JSContext* cx)
+ : cx(cx), freeElements(cx), freeProperties(cx) {}
+ ~JSONFullParseHandlerAnyChar();
+
+ // Allow move construction for use with Rooted.
+ JSONFullParseHandlerAnyChar(JSONFullParseHandlerAnyChar&& other) noexcept
+ : cx(other.cx),
+ v(other.v),
+ parseType(other.parseType),
+ freeElements(std::move(other.freeElements)),
+ freeProperties(std::move(other.freeProperties)) {}
+
+ JSONFullParseHandlerAnyChar(const JSONFullParseHandlerAnyChar& other) =
+ delete;
+ void operator=(const JSONFullParseHandlerAnyChar& other) = delete;
+
+ JSContext* context() { return cx; }
+
+ JS::Value numberValue() const {
+ MOZ_ASSERT(v.isNumber());
+ return v;
+ }
+
+ inline void setNumberValue(double d);
+
+ JS::Value stringValue() const {
+ MOZ_ASSERT(v.isString());
+ return v;
+ }
+
+ JSAtom* atomValue() const {
+ JS::Value strval = stringValue();
+ return &strval.toString()->asAtom();
+ }
+
+ inline JS::Value booleanValue(bool value) { return JS::BooleanValue(value); }
+ inline JS::Value nullValue() { return JS::NullValue(); }
+
+ inline bool objectOpen(Vector<StackEntry, 10>& stack,
+ PropertyVector** properties);
+ inline bool objectPropertyName(Vector<StackEntry, 10>& stack,
+ bool* isProtoInEval);
+ inline void finishObjectMember(Vector<StackEntry, 10>& stack,
+ JS::Handle<JS::Value> value,
+ PropertyVector** properties);
+ inline bool finishObject(Vector<StackEntry, 10>& stack,
+ JS::MutableHandle<JS::Value> vp,
+ PropertyVector& properties);
+
+ inline bool arrayOpen(Vector<StackEntry, 10>& stack,
+ ElementVector** elements);
+ inline bool arrayElement(Vector<StackEntry, 10>& stack,
+ JS::Handle<JS::Value> value,
+ ElementVector** elements);
+ inline bool finishArray(Vector<StackEntry, 10>& stack,
+ JS::MutableHandle<JS::Value> vp,
+ ElementVector& elements);
+
+ inline bool errorReturn() const {
+ return parseType == ParseType::AttemptForEval;
+ }
+
+ inline bool ignoreError() const {
+ return parseType == ParseType::AttemptForEval;
+ }
+
+ inline void freeStackEntry(StackEntry& entry);
+
+ void trace(JSTracer* trc);
+};
+
+template <typename CharT>
+class MOZ_STACK_CLASS JSONFullParseHandler
+ : public JSONFullParseHandlerAnyChar {
+ using Base = JSONFullParseHandlerAnyChar;
+ using CharPtr = mozilla::RangedPtr<const CharT>;
+
+ public:
+ using ContextT = JSContext;
+
+ class StringBuilder {
+ public:
+ JSStringBuilder buffer;
+
+ explicit StringBuilder(JSContext* cx) : buffer(cx) {}
+
+ bool append(char16_t c);
+ bool append(const CharT* begin, const CharT* end);
+ };
+
+ explicit JSONFullParseHandler(JSContext* cx) : Base(cx) {}
+
+ JSONFullParseHandler(JSONFullParseHandler&& other) noexcept
+ : Base(std::move(other)) {}
+
+ JSONFullParseHandler(const JSONFullParseHandler& other) = delete;
+ void operator=(const JSONFullParseHandler& other) = delete;
+
+ template <JSONStringType ST>
+ inline bool setStringValue(CharPtr start, size_t length);
+ template <JSONStringType ST>
+ inline bool setStringValue(StringBuilder& builder);
+
+ void reportError(const char* msg, const char* lineString,
+ const char* columnString);
+};
+
+template <typename CharT>
+class MOZ_STACK_CLASS JSONSyntaxParseHandler {
+ private:
+ using CharPtr = mozilla::RangedPtr<const CharT>;
+
+ public:
+ /* Types for templatized parser. */
+
+ using ContextT = FrontendContext;
+
+ class DummyValue {};
+
+ struct ElementVector {};
+ struct PropertyVector {};
+
+ class StringBuilder {
+ public:
+ explicit StringBuilder(FrontendContext* fc) {}
+
+ bool append(char16_t c) { return true; }
+ bool append(const CharT* begin, const CharT* end) { return true; }
+ };
+
+ struct StackEntry {
+ JSONParserState state;
+ };
+
+ public:
+ FrontendContext* fc;
+
+ /* Public API */
+
+ /* Create a parser for the provided JSON data. */
+ explicit JSONSyntaxParseHandler(FrontendContext* fc) : fc(fc) {}
+
+ JSONSyntaxParseHandler(JSONSyntaxParseHandler&& other) noexcept
+ : fc(other.fc) {}
+
+ JSONSyntaxParseHandler(const JSONSyntaxParseHandler& other) = delete;
+ void operator=(const JSONSyntaxParseHandler& other) = delete;
+
+ FrontendContext* context() { return fc; }
+
+ template <JSONStringType ST>
+ inline bool setStringValue(CharPtr start, size_t length) {
+ return true;
+ }
+
+ template <JSONStringType ST>
+ inline bool setStringValue(StringBuilder& builder) {
+ return true;
+ }
+
+ inline void setNumberValue(double d) {}
+
+ inline DummyValue numberValue() const { return DummyValue(); }
+
+ inline DummyValue stringValue() const { return DummyValue(); }
+
+ inline DummyValue booleanValue(bool value) { return DummyValue(); }
+ inline DummyValue nullValue() { return DummyValue(); }
+
+ inline bool objectOpen(Vector<StackEntry, 10>& stack,
+ PropertyVector** properties);
+ inline bool objectPropertyName(Vector<StackEntry, 10>& stack,
+ bool* isProtoInEval) {
+ *isProtoInEval = false;
+ return true;
+ }
+ inline void finishObjectMember(Vector<StackEntry, 10>& stack,
+ DummyValue& value,
+ PropertyVector** properties) {}
+ inline bool finishObject(Vector<StackEntry, 10>& stack, DummyValue* vp,
+ PropertyVector& properties);
+
+ inline bool arrayOpen(Vector<StackEntry, 10>& stack,
+ ElementVector** elements);
+ inline bool arrayElement(Vector<StackEntry, 10>& stack, DummyValue& value,
+ ElementVector** elements) {
+ return true;
+ }
+ inline bool finishArray(Vector<StackEntry, 10>& stack, DummyValue* vp,
+ ElementVector& elements);
+
+ inline bool errorReturn() const { return false; }
+
+ inline bool ignoreError() const { return false; }
+
+ inline void freeStackEntry(StackEntry& entry) {}
+
+ void reportError(const char* msg, const char* lineString,
+ const char* columnString);
+};
+
+template <typename CharT, typename HandlerT>
+class MOZ_STACK_CLASS JSONPerHandlerParser {
+ using ContextT = typename HandlerT::ContextT;
+
+ using Tokenizer = JSONTokenizer<CharT, JSONPerHandlerParser<CharT, HandlerT>,
+ typename HandlerT::StringBuilder>;
+
+ public:
+ using StringBuilder = typename HandlerT::StringBuilder;
+
+ public:
+ HandlerT handler;
+ Tokenizer tokenizer;
+
+ // All in progress arrays and objects being parsed, in order from outermost
+ // to innermost.
+ Vector<typename HandlerT::StackEntry, 10> stack;
+
+ public:
+ JSONPerHandlerParser(ContextT* context, mozilla::Range<const CharT> data)
+ : handler(context), tokenizer(data, this), stack(context) {}
+
+ JSONPerHandlerParser(JSONPerHandlerParser&& other) noexcept
+ : handler(std::move(other.handler)),
+ tokenizer(std::move(other.tokenizer)),
+ stack(handler.context()) {
+ tokenizer.fixupParser(this);
+ }
+
+ ~JSONPerHandlerParser();
+
+ JSONPerHandlerParser(const JSONPerHandlerParser<CharT, HandlerT>& other) =
+ delete;
+ void operator=(const JSONPerHandlerParser<CharT, HandlerT>& other) = delete;
+
+ template <typename TempValueT, typename ResultSetter>
+ inline bool parseImpl(TempValueT& value, ResultSetter setResult);
+
+ void outOfMemory();
+
+ void error(const char* msg);
+};
+
+template <typename CharT>
+class MOZ_STACK_CLASS JSONParser
+ : JSONPerHandlerParser<CharT, JSONFullParseHandler<CharT>> {
+ using Base = JSONPerHandlerParser<CharT, JSONFullParseHandler<CharT>>;
+
+ public:
+ using ParseType = JSONFullParseHandlerAnyChar::ParseType;
+
+ /* Public API */
+
+ /* Create a parser for the provided JSON data. */
+ JSONParser(JSContext* cx, mozilla::Range<const CharT> data,
+ ParseType parseType)
+ : Base(cx, data) {
+ this->handler.parseType = parseType;
+ }
+
+ /* Allow move construction for use with Rooted. */
+ JSONParser(JSONParser&& other) noexcept : Base(std::move(other)) {}
+
+ JSONParser(const JSONParser& other) = delete;
+ void operator=(const JSONParser& other) = delete;
+
+ /*
+ * Parse the JSON data specified at construction time. If it parses
+ * successfully, store the prescribed value in *vp and return true. If an
+ * internal error (e.g. OOM) occurs during parsing, return false.
+ * Otherwise, if invalid input was specifed but no internal error occurred,
+ * behavior depends upon the error handling specified at construction: if
+ * error handling is RaiseError then throw a SyntaxError and return false,
+ * otherwise return true and set *vp to |undefined|. (JSON syntax can't
+ * represent |undefined|, so the JSON data couldn't have specified it.)
+ */
+ bool parse(JS::MutableHandle<JS::Value> vp);
+
+ void trace(JSTracer* trc);
+};
+
+template <typename CharT, typename Wrapper>
+class MutableWrappedPtrOperations<JSONParser<CharT>, Wrapper>
+ : public WrappedPtrOperations<JSONParser<CharT>, Wrapper> {
+ public:
+ bool parse(JS::MutableHandle<JS::Value> vp) {
+ return static_cast<Wrapper*>(this)->get().parse(vp);
+ }
+};
+
+template <typename CharT>
+class MOZ_STACK_CLASS JSONSyntaxParser
+ : JSONPerHandlerParser<CharT, JSONSyntaxParseHandler<CharT>> {
+ using HandlerT = JSONSyntaxParseHandler<CharT>;
+ using Base = JSONPerHandlerParser<CharT, HandlerT>;
+
+ public:
+ JSONSyntaxParser(FrontendContext* fc, mozilla::Range<const CharT> data)
+ : Base(fc, data) {}
+
+ JSONSyntaxParser(JSONSyntaxParser<CharT>&& other) noexcept
+ : Base(std::move(other)) {}
+
+ JSONSyntaxParser(const JSONSyntaxParser& other) = delete;
+ void operator=(const JSONSyntaxParser& other) = delete;
+
+ bool parse();
+};
+
+} /* namespace js */
+
+#endif /* vm_JSONParser_h */