|  | /* | 
|  | * Copyright (C) 2018 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/trace_storage.h" | 
|  |  | 
|  | #include <string.h> | 
|  | #include <algorithm> | 
|  | #include <limits> | 
|  |  | 
|  | #include "perfetto/ext/base/no_destructor.h" | 
|  |  | 
|  | namespace perfetto { | 
|  | namespace trace_processor { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | template <typename T> | 
|  | void MaybeUpdateMinMax(T begin_it, | 
|  | T end_it, | 
|  | int64_t* min_value, | 
|  | int64_t* max_value) { | 
|  | if (begin_it == end_it) { | 
|  | return; | 
|  | } | 
|  | std::pair<T, T> minmax = std::minmax_element(begin_it, end_it); | 
|  | *min_value = std::min(*min_value, *minmax.first); | 
|  | *max_value = std::max(*max_value, *minmax.second); | 
|  | } | 
|  |  | 
|  | void DbTableMaybeUpdateMinMax(const TypedColumn<int64_t>& column, | 
|  | int64_t* min_value, | 
|  | int64_t* max_value) { | 
|  | if (column.row_map().empty()) | 
|  | return; | 
|  |  | 
|  | SqlValue col_min = *column.Min(); | 
|  | SqlValue col_max = *column.Max(); | 
|  |  | 
|  | PERFETTO_DCHECK(col_min.type == SqlValue::Type::kLong); | 
|  | PERFETTO_DCHECK(col_max.type == SqlValue::Type::kLong); | 
|  |  | 
|  | *min_value = std::min(*min_value, col_min.long_value); | 
|  | *max_value = std::max(*max_value, col_max.long_value); | 
|  | } | 
|  |  | 
|  | std::vector<NullTermStringView> CreateRefTypeStringMap() { | 
|  | std::vector<NullTermStringView> map(static_cast<size_t>(RefType::kRefMax)); | 
|  | map[static_cast<size_t>(RefType::kRefNoRef)] = NullTermStringView(); | 
|  | map[static_cast<size_t>(RefType::kRefUtid)] = "utid"; | 
|  | map[static_cast<size_t>(RefType::kRefCpuId)] = "cpu"; | 
|  | map[static_cast<size_t>(RefType::kRefGpuId)] = "gpu"; | 
|  | map[static_cast<size_t>(RefType::kRefIrq)] = "irq"; | 
|  | map[static_cast<size_t>(RefType::kRefSoftIrq)] = "softirq"; | 
|  | map[static_cast<size_t>(RefType::kRefUpid)] = "upid"; | 
|  | map[static_cast<size_t>(RefType::kRefTrack)] = "track"; | 
|  | return map; | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | const std::vector<NullTermStringView>& GetRefTypeStringMap() { | 
|  | static const base::NoDestructor<std::vector<NullTermStringView>> map( | 
|  | CreateRefTypeStringMap()); | 
|  | return map.ref(); | 
|  | } | 
|  |  | 
|  | TraceStorage::TraceStorage(const Config&) { | 
|  | // Upid/utid 0 is reserved for idle processes/threads. | 
|  | tables::ThreadTable::Row thread_row; | 
|  | thread_row.tid = 0; | 
|  | thread_table_.Insert(thread_row); | 
|  |  | 
|  | tables::ProcessTable::Row process_row; | 
|  | process_row.pid = 0; | 
|  | process_table_.Insert(process_row); | 
|  |  | 
|  | for (uint32_t i = 0; i < variadic_type_ids_.size(); ++i) { | 
|  | variadic_type_ids_[i] = InternString(Variadic::kTypeNames[i]); | 
|  | } | 
|  | } | 
|  |  | 
|  | TraceStorage::~TraceStorage() {} | 
|  |  | 
|  | uint32_t TraceStorage::SqlStats::RecordQueryBegin(const std::string& query, | 
|  | int64_t time_queued, | 
|  | int64_t time_started) { | 
|  | if (queries_.size() >= kMaxLogEntries) { | 
|  | queries_.pop_front(); | 
|  | times_queued_.pop_front(); | 
|  | times_started_.pop_front(); | 
|  | times_first_next_.pop_front(); | 
|  | times_ended_.pop_front(); | 
|  | popped_queries_++; | 
|  | } | 
|  | queries_.push_back(query); | 
|  | times_queued_.push_back(time_queued); | 
|  | times_started_.push_back(time_started); | 
|  | times_first_next_.push_back(0); | 
|  | times_ended_.push_back(0); | 
|  | return static_cast<uint32_t>(popped_queries_ + queries_.size() - 1); | 
|  | } | 
|  |  | 
|  | void TraceStorage::SqlStats::RecordQueryFirstNext(uint32_t row, | 
|  | int64_t time_first_next) { | 
|  | // This means we've popped this query off the queue of queries before it had | 
|  | // a chance to finish. Just silently drop this number. | 
|  | if (popped_queries_ > row) | 
|  | return; | 
|  | uint32_t queue_row = row - popped_queries_; | 
|  | PERFETTO_DCHECK(queue_row < queries_.size()); | 
|  | times_first_next_[queue_row] = time_first_next; | 
|  | } | 
|  |  | 
|  | void TraceStorage::SqlStats::RecordQueryEnd(uint32_t row, int64_t time_ended) { | 
|  | // This means we've popped this query off the queue of queries before it had | 
|  | // a chance to finish. Just silently drop this number. | 
|  | if (popped_queries_ > row) | 
|  | return; | 
|  | uint32_t queue_row = row - popped_queries_; | 
|  | PERFETTO_DCHECK(queue_row < queries_.size()); | 
|  | times_ended_[queue_row] = time_ended; | 
|  | } | 
|  |  | 
|  | std::pair<int64_t, int64_t> TraceStorage::GetTraceTimestampBoundsNs() const { | 
|  | int64_t start_ns = std::numeric_limits<int64_t>::max(); | 
|  | int64_t end_ns = std::numeric_limits<int64_t>::min(); | 
|  | MaybeUpdateMinMax(slices_.start_ns().begin(), slices_.start_ns().end(), | 
|  | &start_ns, &end_ns); | 
|  |  | 
|  | DbTableMaybeUpdateMinMax(raw_table_.ts(), &start_ns, &end_ns); | 
|  | DbTableMaybeUpdateMinMax(counter_table_.ts(), &start_ns, &end_ns); | 
|  | DbTableMaybeUpdateMinMax(slice_table_.ts(), &start_ns, &end_ns); | 
|  | DbTableMaybeUpdateMinMax(heap_profile_allocation_table_.ts(), &start_ns, | 
|  | &end_ns); | 
|  | DbTableMaybeUpdateMinMax(instant_table_.ts(), &start_ns, &end_ns); | 
|  | DbTableMaybeUpdateMinMax(android_log_table_.ts(), &start_ns, &end_ns); | 
|  | DbTableMaybeUpdateMinMax(heap_graph_object_table_.graph_sample_ts(), | 
|  | &start_ns, &end_ns); | 
|  |  | 
|  | if (start_ns == std::numeric_limits<int64_t>::max()) { | 
|  | return std::make_pair(0, 0); | 
|  | } | 
|  | if (start_ns == end_ns) { | 
|  | end_ns += 1; | 
|  | } | 
|  | return std::make_pair(start_ns, end_ns); | 
|  | } | 
|  |  | 
|  | }  // namespace trace_processor | 
|  | }  // namespace perfetto |