blob: bada0f8a5ffd821f1cb8febf6b4627041248f4ab [file] [log] [blame]
// Copyright 2016 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "src/tracing/traced-value.h"
#include "src/base/platform/platform.h"
#include "src/base/vector.h"
#include "src/numbers/conversions.h"
#ifdef V8_USE_PERFETTO
#include "protos/perfetto/trace/track_event/debug_annotation.pbzero.h"
#endif
namespace v8 {
namespace tracing {
namespace {
#define DCHECK_CURRENT_CONTAINER_IS(x) DCHECK_EQ(x, nesting_stack_.back())
#define DCHECK_CONTAINER_STACK_DEPTH_EQ(x) DCHECK_EQ(x, nesting_stack_.size())
#ifdef DEBUG
const bool kStackTypeDict = false;
const bool kStackTypeArray = true;
#define DEBUG_PUSH_CONTAINER(x) nesting_stack_.push_back(x)
#define DEBUG_POP_CONTAINER() nesting_stack_.pop_back()
#else
#define DEBUG_PUSH_CONTAINER(x) ((void)0)
#define DEBUG_POP_CONTAINER() ((void)0)
#endif
void EscapeAndAppendString(const char* value, std::string* result) {
*result += '"';
while (*value) {
unsigned char c = *value++;
switch (c) {
case '\b':
*result += "\\b";
break;
case '\f':
*result += "\\f";
break;
case '\n':
*result += "\\n";
break;
case '\r':
*result += "\\r";
break;
case '\t':
*result += "\\t";
break;
case '\"':
*result += "\\\"";
break;
case '\\':
*result += "\\\\";
break;
default:
if (c < '\x20' || c == '\x7F') {
char number_buffer[8];
base::OS::SNPrintF(number_buffer, arraysize(number_buffer), "\\u%04X",
static_cast<unsigned>(c));
*result += number_buffer;
} else {
*result += c;
}
}
}
*result += '"';
}
} // namespace
// static
std::unique_ptr<TracedValue> TracedValue::Create() {
return std::unique_ptr<TracedValue>(new TracedValue());
}
TracedValue::TracedValue() : first_item_(true) {
DEBUG_PUSH_CONTAINER(kStackTypeDict);
}
TracedValue::~TracedValue() {
DCHECK_CURRENT_CONTAINER_IS(kStackTypeDict);
DEBUG_POP_CONTAINER();
DCHECK_CONTAINER_STACK_DEPTH_EQ(0u);
}
void TracedValue::SetInteger(const char* name, int value) {
DCHECK_CURRENT_CONTAINER_IS(kStackTypeDict);
WriteName(name);
data_ += std::to_string(value);
}
void TracedValue::SetDouble(const char* name, double value) {
DCHECK_CURRENT_CONTAINER_IS(kStackTypeDict);
WriteName(name);
base::EmbeddedVector<char, 100> buffer;
data_ += internal::DoubleToCString(value, buffer);
}
void TracedValue::SetBoolean(const char* name, bool value) {
DCHECK_CURRENT_CONTAINER_IS(kStackTypeDict);
WriteName(name);
data_ += value ? "true" : "false";
}
void TracedValue::SetString(const char* name, const char* value) {
DCHECK_CURRENT_CONTAINER_IS(kStackTypeDict);
WriteName(name);
EscapeAndAppendString(value, &data_);
}
void TracedValue::SetValue(const char* name, TracedValue* value) {
DCHECK_CURRENT_CONTAINER_IS(kStackTypeDict);
WriteName(name);
std::string tmp;
value->AppendAsTraceFormat(&tmp);
data_ += tmp;
}
void TracedValue::BeginDictionary(const char* name) {
DCHECK_CURRENT_CONTAINER_IS(kStackTypeDict);
DEBUG_PUSH_CONTAINER(kStackTypeDict);
WriteName(name);
data_ += '{';
first_item_ = true;
}
void TracedValue::BeginArray(const char* name) {
DCHECK_CURRENT_CONTAINER_IS(kStackTypeDict);
DEBUG_PUSH_CONTAINER(kStackTypeArray);
WriteName(name);
data_ += '[';
first_item_ = true;
}
void TracedValue::AppendInteger(int value) {
DCHECK_CURRENT_CONTAINER_IS(kStackTypeArray);
WriteComma();
data_ += std::to_string(value);
}
void TracedValue::AppendDouble(double value) {
DCHECK_CURRENT_CONTAINER_IS(kStackTypeArray);
WriteComma();
base::EmbeddedVector<char, 100> buffer;
data_ += internal::DoubleToCString(value, buffer);
}
void TracedValue::AppendBoolean(bool value) {
DCHECK_CURRENT_CONTAINER_IS(kStackTypeArray);
WriteComma();
data_ += value ? "true" : "false";
}
void TracedValue::AppendString(const char* value) {
DCHECK_CURRENT_CONTAINER_IS(kStackTypeArray);
WriteComma();
EscapeAndAppendString(value, &data_);
}
void TracedValue::BeginDictionary() {
DCHECK_CURRENT_CONTAINER_IS(kStackTypeArray);
DEBUG_PUSH_CONTAINER(kStackTypeDict);
WriteComma();
data_ += '{';
first_item_ = true;
}
void TracedValue::BeginArray() {
DCHECK_CURRENT_CONTAINER_IS(kStackTypeArray);
DEBUG_PUSH_CONTAINER(kStackTypeArray);
WriteComma();
data_ += '[';
first_item_ = true;
}
void TracedValue::EndDictionary() {
DCHECK_CURRENT_CONTAINER_IS(kStackTypeDict);
DEBUG_POP_CONTAINER();
data_ += '}';
first_item_ = false;
}
void TracedValue::EndArray() {
DCHECK_CURRENT_CONTAINER_IS(kStackTypeArray);
DEBUG_POP_CONTAINER();
data_ += ']';
first_item_ = false;
}
void TracedValue::WriteComma() {
if (first_item_) {
first_item_ = false;
} else {
data_ += ',';
}
}
void TracedValue::WriteName(const char* name) {
WriteComma();
data_ += '"';
data_ += name;
data_ += "\":";
}
void TracedValue::AppendAsTraceFormat(std::string* out) const {
*out += '{';
*out += data_;
*out += '}';
}
#ifdef V8_USE_PERFETTO
void TracedValue::Add(
perfetto::protos::pbzero::DebugAnnotation* annotation) const {
std::string json;
json += "{";
json += data_;
json += "}";
annotation->set_legacy_json_value(json);
}
#endif // V8_USE_PERFETTO
} // namespace tracing
} // namespace v8