|  | /* | 
|  | * 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. | 
|  | */ | 
|  |  | 
|  | #ifndef SRC_TRACE_PROCESSOR_TRACE_PROCESSOR_IMPL_H_ | 
|  | #define SRC_TRACE_PROCESSOR_TRACE_PROCESSOR_IMPL_H_ | 
|  |  | 
|  | #include <sqlite3.h> | 
|  |  | 
|  | #include <atomic> | 
|  | #include <functional> | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | #include "perfetto/ext/base/string_view.h" | 
|  | #include "perfetto/trace_processor/basic_types.h" | 
|  | #include "perfetto/trace_processor/status.h" | 
|  | #include "perfetto/trace_processor/trace_processor.h" | 
|  | #include "src/trace_processor/sqlite/db_sqlite_table.h" | 
|  | #include "src/trace_processor/sqlite/query_cache.h" | 
|  | #include "src/trace_processor/sqlite/scoped_db.h" | 
|  | #include "src/trace_processor/trace_processor_storage_impl.h" | 
|  |  | 
|  | #include "src/trace_processor/descriptors.h" | 
|  | #include "src/trace_processor/metrics/metrics.h" | 
|  |  | 
|  | namespace perfetto { | 
|  | namespace trace_processor { | 
|  |  | 
|  | // Coordinates the loading of traces from an arbitrary source and allows | 
|  | // execution of SQL queries on the events in these traces. | 
|  | class TraceProcessorImpl : public TraceProcessor, | 
|  | public TraceProcessorStorageImpl { | 
|  | public: | 
|  | explicit TraceProcessorImpl(const Config&); | 
|  |  | 
|  | ~TraceProcessorImpl() override; | 
|  |  | 
|  | // TraceProcessorStorage implementation: | 
|  | util::Status Parse(std::unique_ptr<uint8_t[]>, size_t) override; | 
|  | void NotifyEndOfFile() override; | 
|  |  | 
|  | // TraceProcessor implementation: | 
|  | Iterator ExecuteQuery(const std::string& sql, | 
|  | int64_t time_queued = 0) override; | 
|  |  | 
|  | util::Status RegisterMetric(const std::string& path, | 
|  | const std::string& sql) override; | 
|  |  | 
|  | util::Status ExtendMetricsProto(const uint8_t* data, size_t size) override; | 
|  |  | 
|  | util::Status ComputeMetric(const std::vector<std::string>& metric_names, | 
|  | std::vector<uint8_t>* metrics) override; | 
|  |  | 
|  | void InterruptQuery() override; | 
|  |  | 
|  | size_t RestoreInitialTables() override; | 
|  |  | 
|  | std::string GetCurrentTraceName() override; | 
|  | void SetCurrentTraceName(const std::string&) override; | 
|  |  | 
|  | private: | 
|  | // Needed for iterators to be able to delete themselves from the vector. | 
|  | friend class IteratorImpl; | 
|  |  | 
|  | template <typename Table> | 
|  | void RegisterDbTable(const Table& table) { | 
|  | DbSqliteTable::RegisterTable(*db_, query_cache_.get(), Table::Schema(), | 
|  | &table, table.table_name()); | 
|  | } | 
|  |  | 
|  | void RegisterDynamicTable( | 
|  | std::unique_ptr<DbSqliteTable::DynamicTableGenerator> generator) { | 
|  | DbSqliteTable::RegisterTable(*db_, query_cache_.get(), | 
|  | std::move(generator)); | 
|  | } | 
|  |  | 
|  | ScopedDb db_; | 
|  | std::unique_ptr<QueryCache> query_cache_; | 
|  |  | 
|  | DescriptorPool pool_; | 
|  | std::vector<metrics::SqlMetricFile> sql_metrics_; | 
|  |  | 
|  | std::vector<IteratorImpl*> iterators_; | 
|  |  | 
|  | // This is atomic because it is set by the CTRL-C signal handler and we need | 
|  | // to prevent single-flow compiler optimizations in ExecuteQuery(). | 
|  | std::atomic<bool> query_interrupted_{false}; | 
|  |  | 
|  | // Keeps track of the tables created by the ingestion process. This is used | 
|  | // by RestoreInitialTables() to delete all the tables/view that have been | 
|  | // created after that point. | 
|  | std::vector<std::string> initial_tables_; | 
|  |  | 
|  | std::string current_trace_name_; | 
|  | uint64_t bytes_parsed_ = 0; | 
|  | }; | 
|  |  | 
|  | // The pointer implementation of TraceProcessor::Iterator. | 
|  | class TraceProcessor::IteratorImpl { | 
|  | public: | 
|  | IteratorImpl(TraceProcessorImpl* impl, | 
|  | sqlite3* db, | 
|  | ScopedStmt, | 
|  | uint32_t column_count, | 
|  | util::Status, | 
|  | uint32_t sql_stats_row); | 
|  | ~IteratorImpl(); | 
|  |  | 
|  | IteratorImpl(IteratorImpl&) noexcept = delete; | 
|  | IteratorImpl& operator=(IteratorImpl&) = delete; | 
|  |  | 
|  | IteratorImpl(IteratorImpl&&) noexcept = default; | 
|  | IteratorImpl& operator=(IteratorImpl&&) = default; | 
|  |  | 
|  | // Methods called by TraceProcessor::Iterator. | 
|  | bool Next() { | 
|  | // Delegate to the cc file to prevent trace_storage.h include in this | 
|  | // file. | 
|  | if (!called_next_) { | 
|  | RecordFirstNextInSqlStats(); | 
|  | called_next_ = true; | 
|  | } | 
|  |  | 
|  | if (!status_.ok()) | 
|  | return false; | 
|  |  | 
|  | int ret = sqlite3_step(*stmt_); | 
|  | if (PERFETTO_UNLIKELY(ret != SQLITE_ROW && ret != SQLITE_DONE)) { | 
|  | status_ = util::ErrStatus("%s", sqlite3_errmsg(db_)); | 
|  | return false; | 
|  | } | 
|  | return ret == SQLITE_ROW; | 
|  | } | 
|  |  | 
|  | SqlValue Get(uint32_t col) { | 
|  | auto column = static_cast<int>(col); | 
|  | auto col_type = sqlite3_column_type(*stmt_, column); | 
|  | SqlValue value; | 
|  | switch (col_type) { | 
|  | case SQLITE_INTEGER: | 
|  | value.type = SqlValue::kLong; | 
|  | value.long_value = sqlite3_column_int64(*stmt_, column); | 
|  | break; | 
|  | case SQLITE_TEXT: | 
|  | value.type = SqlValue::kString; | 
|  | value.string_value = | 
|  | reinterpret_cast<const char*>(sqlite3_column_text(*stmt_, column)); | 
|  | break; | 
|  | case SQLITE_FLOAT: | 
|  | value.type = SqlValue::kDouble; | 
|  | value.double_value = sqlite3_column_double(*stmt_, column); | 
|  | break; | 
|  | case SQLITE_BLOB: | 
|  | value.type = SqlValue::kBytes; | 
|  | value.bytes_value = sqlite3_column_blob(*stmt_, column); | 
|  | value.bytes_count = | 
|  | static_cast<size_t>(sqlite3_column_bytes(*stmt_, column)); | 
|  | break; | 
|  | case SQLITE_NULL: | 
|  | value.type = SqlValue::kNull; | 
|  | break; | 
|  | } | 
|  | return value; | 
|  | } | 
|  |  | 
|  | std::string GetColumnName(uint32_t col) { | 
|  | return sqlite3_column_name(stmt_.get(), static_cast<int>(col)); | 
|  | } | 
|  |  | 
|  | uint32_t ColumnCount() { return column_count_; } | 
|  |  | 
|  | util::Status Status() { return status_; } | 
|  |  | 
|  | // Methods called by TraceProcessorImpl. | 
|  | void Reset(); | 
|  |  | 
|  | private: | 
|  | void RecordFirstNextInSqlStats(); | 
|  |  | 
|  | TraceProcessorImpl* trace_processor_; | 
|  | sqlite3* db_ = nullptr; | 
|  | ScopedStmt stmt_; | 
|  | uint32_t column_count_ = 0; | 
|  | util::Status status_; | 
|  |  | 
|  | uint32_t sql_stats_row_ = 0; | 
|  | bool called_next_ = false; | 
|  | }; | 
|  |  | 
|  | }  // namespace trace_processor | 
|  | }  // namespace perfetto | 
|  |  | 
|  | #endif  // SRC_TRACE_PROCESSOR_TRACE_PROCESSOR_IMPL_H_ |