| /* |
| * Copyright (C) 2021 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #include "perfetto/tracing/traced_value.h" |
| |
| #include "perfetto/base/logging.h" |
| #include "perfetto/tracing/debug_annotation.h" |
| #include "perfetto/tracing/internal/track_event_interned_fields.h" |
| #include "protos/perfetto/trace/track_event/debug_annotation.pbzero.h" |
| |
| namespace perfetto { |
| |
| namespace internal { |
| |
| TracedValue CreateTracedValueFromProto( |
| protos::pbzero::DebugAnnotation* annotation, |
| EventContext* event_context) { |
| return TracedValue::CreateFromProto(annotation, event_context); |
| } |
| |
| } // namespace internal |
| |
| // static |
| TracedValue TracedValue::CreateFromProto( |
| protos::pbzero::DebugAnnotation* annotation, |
| EventContext* event_context) { |
| return TracedValue(annotation, event_context, nullptr); |
| } |
| |
| TracedValue::TracedValue(TracedValue&&) = default; |
| TracedValue::~TracedValue() = default; |
| |
| void TracedValue::WriteInt64(int64_t value) && { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| annotation_->set_int_value(value); |
| } |
| |
| void TracedValue::WriteUInt64(uint64_t value) && { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| annotation_->set_uint_value(value); |
| } |
| |
| void TracedValue::WriteDouble(double value) && { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| annotation_->set_double_value(value); |
| } |
| |
| void TracedValue::WriteBoolean(bool value) && { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| annotation_->set_bool_value(value); |
| } |
| |
| void TracedValue::WriteString(const char* value) && { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| annotation_->set_string_value(value); |
| } |
| |
| void TracedValue::WriteString(const char* value, size_t len) && { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| annotation_->set_string_value(value, len); |
| } |
| |
| void TracedValue::WriteString(const std::string& value) && { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| annotation_->set_string_value(value); |
| } |
| |
| void TracedValue::WritePointer(const void* value) && { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| annotation_->set_pointer_value(reinterpret_cast<uint64_t>(value)); |
| } |
| |
| TracedDictionary TracedValue::WriteDictionary() && { |
| // Note: this passes |checked_scope_.is_active_| bit to the parent to be |
| // picked up later by the new TracedDictionary. |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| checked_scope_.Reset(); |
| |
| PERFETTO_DCHECK(!annotation_->is_finalized()); |
| return TracedDictionary(annotation_, |
| protos::pbzero::DebugAnnotation::kDictEntries, |
| event_context_, checked_scope_.parent_scope()); |
| } |
| |
| TracedArray TracedValue::WriteArray() && { |
| // Note: this passes |checked_scope_.is_active_| bit to the parent to be |
| // picked up later by the new TracedDictionary. |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| checked_scope_.Reset(); |
| |
| PERFETTO_DCHECK(!annotation_->is_finalized()); |
| return TracedArray(annotation_, event_context_, |
| checked_scope_.parent_scope()); |
| } |
| |
| protozero::Message* TracedValue::WriteProtoInternal(const char* name) { |
| if (event_context_) { |
| annotation_->set_proto_type_name_iid( |
| internal::InternedDebugAnnotationValueTypeName::Get(event_context_, |
| name)); |
| } else { |
| annotation_->set_proto_type_name(name); |
| } |
| return annotation_->template BeginNestedMessage<protozero::Message>( |
| protos::pbzero::DebugAnnotation::kProtoValueFieldNumber); |
| } |
| |
| TracedArray::TracedArray(TracedValue annotation) |
| : TracedArray(std::move(annotation).WriteArray()) {} |
| |
| TracedValue TracedArray::AppendItem() { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| return TracedValue(annotation_->add_array_values(), event_context_, |
| &checked_scope_); |
| } |
| |
| TracedDictionary TracedArray::AppendDictionary() { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| return AppendItem().WriteDictionary(); |
| } |
| |
| TracedArray TracedArray::AppendArray() { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| return AppendItem().WriteArray(); |
| } |
| |
| TracedDictionary::TracedDictionary(TracedValue annotation) |
| : TracedDictionary(std::move(annotation).WriteDictionary()) {} |
| |
| TracedValue TracedDictionary::AddItem(StaticString key) { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| protos::pbzero::DebugAnnotation* item = |
| message_->BeginNestedMessage<protos::pbzero::DebugAnnotation>(field_id_); |
| item->set_name(key.value); |
| return TracedValue(item, event_context_, &checked_scope_); |
| } |
| |
| TracedValue TracedDictionary::AddItem(DynamicString key) { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| protos::pbzero::DebugAnnotation* item = |
| message_->BeginNestedMessage<protos::pbzero::DebugAnnotation>(field_id_); |
| item->set_name(key.value); |
| return TracedValue(item, event_context_, &checked_scope_); |
| } |
| |
| TracedDictionary TracedDictionary::AddDictionary(StaticString key) { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| return AddItem(key).WriteDictionary(); |
| } |
| |
| TracedDictionary TracedDictionary::AddDictionary(DynamicString key) { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| return AddItem(key).WriteDictionary(); |
| } |
| |
| TracedArray TracedDictionary::AddArray(StaticString key) { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| return AddItem(key).WriteArray(); |
| } |
| |
| TracedArray TracedDictionary::AddArray(DynamicString key) { |
| PERFETTO_DCHECK(checked_scope_.is_active()); |
| return AddItem(key).WriteArray(); |
| } |
| |
| } // namespace perfetto |