diff options
Diffstat (limited to 'js/src/vm/JSONPrinter.cpp')
-rw-r--r-- | js/src/vm/JSONPrinter.cpp | 259 |
1 files changed, 259 insertions, 0 deletions
diff --git a/js/src/vm/JSONPrinter.cpp b/js/src/vm/JSONPrinter.cpp new file mode 100644 index 0000000000..475510a58d --- /dev/null +++ b/js/src/vm/JSONPrinter.cpp @@ -0,0 +1,259 @@ +/* -*- 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/JSONPrinter.h" + +#include "mozilla/Assertions.h" +#include "mozilla/FloatingPoint.h" +#include "mozilla/IntegerPrintfMacros.h" + +#include <stdarg.h> + +#include "util/DoubleToString.h" + +using namespace js; + +JSONPrinter::~JSONPrinter() { + if (dtoaState_) { + DestroyDtoaState(dtoaState_); + } +} + +void JSONPrinter::indent() { + MOZ_ASSERT(indentLevel_ >= 0); + if (indent_) { + out_.putChar('\n'); + for (int i = 0; i < indentLevel_; i++) { + out_.put(" "); + } + } +} + +void JSONPrinter::propertyName(const char* name) { + if (!first_) { + out_.putChar(','); + } + indent(); + out_.printf("\"%s\":", name); + if (indent_) { + out_.put(" "); + } + first_ = false; +} + +void JSONPrinter::beginObject() { + if (!first_) { + out_.putChar(','); + } + indent(); + out_.putChar('{'); + indentLevel_++; + first_ = true; +} + +void JSONPrinter::beginList() { + if (!first_) { + out_.putChar(','); + } + indent(); + out_.putChar('['); + indentLevel_++; + first_ = true; +} + +void JSONPrinter::beginObjectProperty(const char* name) { + propertyName(name); + out_.putChar('{'); + indentLevel_++; + first_ = true; +} + +void JSONPrinter::beginListProperty(const char* name) { + propertyName(name); + out_.putChar('['); + indentLevel_++; + first_ = true; +} + +GenericPrinter& JSONPrinter::beginStringProperty(const char* name) { + propertyName(name); + out_.putChar('"'); + return out_; +} + +void JSONPrinter::endStringProperty() { + endString(); + first_ = false; +} + +GenericPrinter& JSONPrinter::beginString() { + if (!first_) { + out_.putChar(','); + } + indent(); + out_.putChar('"'); + return out_; +} + +void JSONPrinter::endString() { out_.putChar('"'); } + +void JSONPrinter::boolProperty(const char* name, bool value) { + propertyName(name); + out_.put(value ? "true" : "false"); +} + +void JSONPrinter::property(const char* name, const char* value) { + beginStringProperty(name); + out_.put(value); + endStringProperty(); +} + +void JSONPrinter::formatProperty(const char* name, const char* format, ...) { + va_list ap; + va_start(ap, format); + + beginStringProperty(name); + out_.vprintf(format, ap); + endStringProperty(); + + va_end(ap); +} + +void JSONPrinter::formatProperty(const char* name, const char* format, + va_list ap) { + beginStringProperty(name); + out_.vprintf(format, ap); + endStringProperty(); +} + +void JSONPrinter::value(const char* format, ...) { + va_list ap; + va_start(ap, format); + + if (!first_) { + out_.putChar(','); + } + indent(); + out_.putChar('"'); + out_.vprintf(format, ap); + out_.putChar('"'); + + va_end(ap); + first_ = false; +} + +void JSONPrinter::property(const char* name, int32_t value) { + propertyName(name); + out_.printf("%" PRId32, value); +} + +void JSONPrinter::value(int val) { + if (!first_) { + out_.putChar(','); + } + indent(); + out_.printf("%d", val); + first_ = false; +} + +void JSONPrinter::property(const char* name, uint32_t value) { + propertyName(name); + out_.printf("%" PRIu32, value); +} + +void JSONPrinter::property(const char* name, int64_t value) { + propertyName(name); + out_.printf("%" PRId64, value); +} + +void JSONPrinter::property(const char* name, uint64_t value) { + propertyName(name); + out_.printf("%" PRIu64, value); +} + +#if defined(XP_DARWIN) || defined(__OpenBSD__) +void JSONPrinter::property(const char* name, size_t value) { + propertyName(name); + out_.printf("%zu", value); +} +#endif + +void JSONPrinter::floatProperty(const char* name, double value, + size_t precision) { + if (!mozilla::IsFinite(value)) { + propertyName(name); + out_.put("null"); + return; + } + + if (!dtoaState_) { + dtoaState_ = NewDtoaState(); + if (!dtoaState_) { + out_.reportOutOfMemory(); + return; + } + } + + char buffer[DTOSTR_STANDARD_BUFFER_SIZE]; + char* str = js_dtostr(dtoaState_, buffer, sizeof(buffer), DTOSTR_STANDARD, + precision, value); + if (!str) { + out_.reportOutOfMemory(); + return; + } + + property(name, str); +} + +void JSONPrinter::property(const char* name, const mozilla::TimeDuration& dur, + TimePrecision precision) { + if (precision == MICROSECONDS) { + property(name, static_cast<int64_t>(dur.ToMicroseconds())); + return; + } + + propertyName(name); + lldiv_t split; + switch (precision) { + case SECONDS: + split = lldiv(static_cast<int64_t>(dur.ToMilliseconds()), 1000); + break; + case MILLISECONDS: + split = lldiv(static_cast<int64_t>(dur.ToMicroseconds()), 1000); + break; + case MICROSECONDS: + MOZ_ASSERT_UNREACHABLE(""); + }; + out_.printf("%lld.%03lld", split.quot, split.rem); +} + +void JSONPrinter::nullProperty(const char* name) { + propertyName(name); + out_.put("null"); +} + +void JSONPrinter::nullValue() { + if (!first_) { + out_.putChar(','); + } + indent(); + out_.put("null"); + first_ = false; +} + +void JSONPrinter::endObject() { + indentLevel_--; + indent(); + out_.putChar('}'); + first_ = false; +} + +void JSONPrinter::endList() { + indentLevel_--; + indent(); + out_.putChar(']'); + first_ = false; +} |