summaryrefslogtreecommitdiffstats
path: root/js/src/vm/JSONParser.h
diff options
context:
space:
mode:
Diffstat (limited to 'js/src/vm/JSONParser.h')
-rw-r--r--js/src/vm/JSONParser.h284
1 files changed, 284 insertions, 0 deletions
diff --git a/js/src/vm/JSONParser.h b/js/src/vm/JSONParser.h
new file mode 100644
index 0000000000..a88980b382
--- /dev/null
+++ b/js/src/vm/JSONParser.h
@@ -0,0 +1,284 @@
+/* -*- 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/Attributes.h"
+#include "mozilla/Range.h"
+
+#include "jspubtd.h"
+
+#include "ds/IdValuePair.h"
+#include "vm/StringType.h"
+
+namespace js {
+
+// JSONParser base class. JSONParser is templatized to work on either Latin1
+// or TwoByte input strings, JSONParserBase holds all state and methods that
+// can be shared between the two encodings.
+class MOZ_STACK_CLASS JSONParserBase {
+ public:
+ 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,
+ };
+
+ private:
+ /* Data members */
+ Value v;
+
+ protected:
+ JSContext* const cx;
+
+ const ParseType parseType;
+
+ enum Token {
+ String,
+ Number,
+ True,
+ False,
+ Null,
+ ArrayOpen,
+ ArrayClose,
+ ObjectOpen,
+ ObjectClose,
+ Colon,
+ Comma,
+ OOM,
+ Error
+ };
+
+ // 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.
+ typedef GCVector<Value, 20> ElementVector;
+
+ // State for an object that is currently being parsed. This includes all
+ // the key/value pairs that have been seen so far.
+ typedef GCVector<IdValuePair, 10> PropertyVector;
+
+ // Possible states the parser can be in between values.
+ enum ParserState {
+ // 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
+ };
+
+ // Stack element for an in progress array or object.
+ struct StackEntry {
+ ElementVector& elements() {
+ MOZ_ASSERT(state == FinishArrayElement);
+ return *static_cast<ElementVector*>(vector);
+ }
+
+ PropertyVector& properties() {
+ MOZ_ASSERT(state == FinishObjectMember);
+ return *static_cast<PropertyVector*>(vector);
+ }
+
+ explicit StackEntry(ElementVector* elements)
+ : state(FinishArrayElement), vector(elements) {}
+
+ explicit StackEntry(PropertyVector* properties)
+ : state(FinishObjectMember), vector(properties) {}
+
+ ParserState state;
+
+ private:
+ void* vector;
+ };
+
+ // All in progress arrays and objects being parsed, in order from outermost
+ // to innermost.
+ Vector<StackEntry, 10> stack;
+
+ // 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;
+
+#ifdef DEBUG
+ Token lastToken;
+#endif
+
+ JSONParserBase(JSContext* cx, ParseType parseType)
+ : cx(cx),
+ parseType(parseType),
+ stack(cx),
+ freeElements(cx),
+ freeProperties(cx)
+#ifdef DEBUG
+ ,
+ lastToken(Error)
+#endif
+ {
+ }
+ ~JSONParserBase();
+
+ // Allow move construction for use with Rooted.
+ JSONParserBase(JSONParserBase&& other)
+ : v(other.v),
+ cx(other.cx),
+ parseType(other.parseType),
+ stack(std::move(other.stack)),
+ freeElements(std::move(other.freeElements)),
+ freeProperties(std::move(other.freeProperties))
+#ifdef DEBUG
+ ,
+ lastToken(std::move(other.lastToken))
+#endif
+ {
+ }
+
+ Value numberValue() const {
+ MOZ_ASSERT(lastToken == Number);
+ MOZ_ASSERT(v.isNumber());
+ return v;
+ }
+
+ Value stringValue() const {
+ MOZ_ASSERT(lastToken == String);
+ MOZ_ASSERT(v.isString());
+ return v;
+ }
+
+ JSAtom* atomValue() const {
+ Value strval = stringValue();
+ return &strval.toString()->asAtom();
+ }
+
+ Token token(Token t) {
+ MOZ_ASSERT(t != String);
+ MOZ_ASSERT(t != Number);
+#ifdef DEBUG
+ lastToken = t;
+#endif
+ return t;
+ }
+
+ Token stringToken(JSString* str) {
+ this->v = StringValue(str);
+#ifdef DEBUG
+ lastToken = String;
+#endif
+ return String;
+ }
+
+ Token numberToken(double d) {
+ this->v = NumberValue(d);
+#ifdef DEBUG
+ lastToken = Number;
+#endif
+ return Number;
+ }
+
+ enum StringType { PropertyName, LiteralValue };
+
+ bool errorReturn();
+
+ bool finishObject(MutableHandleValue vp, PropertyVector& properties);
+ bool finishArray(MutableHandleValue vp, ElementVector& elements);
+
+ void trace(JSTracer* trc);
+
+ private:
+ JSONParserBase(const JSONParserBase& other) = delete;
+ void operator=(const JSONParserBase& other) = delete;
+};
+
+template <typename CharT>
+class MOZ_STACK_CLASS JSONParser : public JSONParserBase {
+ private:
+ using CharPtr = mozilla::RangedPtr<const CharT>;
+
+ CharPtr current;
+ const CharPtr begin, end;
+
+ public:
+ /* Public API */
+
+ /* Create a parser for the provided JSON data. */
+ JSONParser(JSContext* cx, mozilla::Range<const CharT> data,
+ ParseType parseType)
+ : JSONParserBase(cx, parseType),
+ current(data.begin()),
+ begin(current),
+ end(data.end()) {
+ MOZ_ASSERT(current <= end);
+ }
+
+ /* Allow move construction for use with Rooted. */
+ JSONParser(JSONParser&& other)
+ : JSONParserBase(std::move(other)),
+ current(other.current),
+ begin(other.begin),
+ end(other.end) {}
+
+ /*
+ * 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(MutableHandleValue vp);
+
+ void trace(JSTracer* trc) { JSONParserBase::trace(trc); }
+
+ private:
+ template <StringType ST>
+ Token readString();
+
+ Token readNumber();
+
+ Token advance();
+ Token advancePropertyName();
+ Token advancePropertyColon();
+ Token advanceAfterProperty();
+ Token advanceAfterObjectOpen();
+ Token advanceAfterArrayElement();
+
+ void error(const char* msg);
+
+ void getTextPosition(uint32_t* column, uint32_t* line);
+
+ private:
+ JSONParser(const JSONParser& other) = delete;
+ void operator=(const JSONParser& other) = delete;
+};
+
+template <typename CharT, typename Wrapper>
+class MutableWrappedPtrOperations<JSONParser<CharT>, Wrapper>
+ : public WrappedPtrOperations<JSONParser<CharT>, Wrapper> {
+ public:
+ bool parse(MutableHandleValue vp) {
+ return static_cast<Wrapper*>(this)->get().parse(vp);
+ }
+};
+
+} /* namespace js */
+
+#endif /* vm_JSONParser_h */