|  | /* | 
|  | * Copyright (C) 2020 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 "src/trace_processor/tp_metatrace.h" | 
|  |  | 
|  | namespace perfetto { | 
|  | namespace trace_processor { | 
|  | namespace metatrace { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | using ProtoEnum = protos::pbzero::MetatraceCategories; | 
|  | ProtoEnum MetatraceCategoriesToProtoEnum(MetatraceCategories categories) { | 
|  | // Note: these are intentionally chained ifs and not else-ifs as it's possible | 
|  | // for multiple of these if statements to be true. | 
|  | ProtoEnum result = ProtoEnum::NONE; | 
|  | if (categories & MetatraceCategories::QUERY_TIMELINE) | 
|  | result = static_cast<ProtoEnum>(result | ProtoEnum::QUERY_TIMELINE); | 
|  | if (categories & MetatraceCategories::FUNCTION_CALL) | 
|  | result = static_cast<ProtoEnum>(result | ProtoEnum::FUNCTION_CALL); | 
|  | if (categories & MetatraceCategories::QUERY_DETAILED) | 
|  | result = static_cast<ProtoEnum>(result | ProtoEnum::QUERY_DETAILED); | 
|  | if (categories & MetatraceCategories::DB) | 
|  | result = static_cast<ProtoEnum>(result | ProtoEnum::DB); | 
|  | if (categories & MetatraceCategories::API_TIMELINE) | 
|  | result = static_cast<ProtoEnum>(result | ProtoEnum::API_TIMELINE); | 
|  | return result; | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | Category g_enabled_categories = Category::NONE; | 
|  |  | 
|  | void Enable(MetatraceConfig config) { | 
|  | g_enabled_categories = MetatraceCategoriesToProtoEnum(config.categories); | 
|  | if (config.override_buffer_size) { | 
|  | RingBuffer::GetInstance()->Resize(config.override_buffer_size); | 
|  | } | 
|  | } | 
|  |  | 
|  | void DisableAndReadBuffer(std::function<void(Record*)> fn) { | 
|  | g_enabled_categories = Category::NONE; | 
|  | if (!fn) | 
|  | return; | 
|  | RingBuffer::GetInstance()->ReadAll(fn); | 
|  | } | 
|  |  | 
|  | RingBuffer::RingBuffer() : data_(kDefaultCapacity) { | 
|  | static_assert((kDefaultCapacity & (kDefaultCapacity - 1)) == 0, | 
|  | "Capacity should be a power of 2"); | 
|  | } | 
|  |  | 
|  | void RingBuffer::Resize(size_t requested_capacity) { | 
|  | size_t actual_capacity = 1; | 
|  | while (actual_capacity < requested_capacity) | 
|  | actual_capacity <<= 1; | 
|  | data_.resize(actual_capacity); | 
|  | start_idx_ = 0; | 
|  | write_idx_ = 0; | 
|  | } | 
|  |  | 
|  | void RingBuffer::ReadAll(std::function<void(Record*)> fn) { | 
|  | // Mark as reading so we don't get reentrancy in obtaining new | 
|  | // trace events. | 
|  | is_reading_ = true; | 
|  |  | 
|  | uint64_t start = (write_idx_ - start_idx_) < data_.size() | 
|  | ? start_idx_ | 
|  | : write_idx_ - data_.size(); | 
|  | uint64_t end = write_idx_; | 
|  |  | 
|  | // Increment the write index by kCapacity + 1. This ensures that if | 
|  | // ScopedEntry is destoryed in |fn| below, we won't get overwrites | 
|  | // while reading the buffer. | 
|  | // This works because of the logic in ~ScopedEntry and | 
|  | // RingBuffer::HasOverwritten which ensures that we don't overwrite entries | 
|  | // more than kCapcity elements in the past. | 
|  | write_idx_ += data_.size() + 1; | 
|  |  | 
|  | for (uint64_t i = start; i < end; ++i) { | 
|  | Record* record = At(i); | 
|  |  | 
|  | // If the slice was unfinished for some reason, don't emit it. | 
|  | if (record->duration_ns != 0) { | 
|  | fn(record); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Ensure that the start pointer is updated to the write pointer. | 
|  | start_idx_ = write_idx_; | 
|  |  | 
|  | // Remove the reading marker. | 
|  | is_reading_ = false; | 
|  | } | 
|  |  | 
|  | }  // namespace metatrace | 
|  | }  // namespace trace_processor | 
|  | }  // namespace perfetto |