| // 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 |