| /* |
| * 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/sqlite/sqlite_table.h" |
| |
| #include <string.h> |
| #include <algorithm> |
| #include <cinttypes> |
| #include <map> |
| #include <memory> |
| |
| #include "perfetto/base/logging.h" |
| #include "perfetto/base/status.h" |
| #include "perfetto/ext/base/status_or.h" |
| #include "perfetto/ext/base/string_view.h" |
| #include "sqlite3.h" |
| #include "src/trace_processor/sqlite/sqlite_engine.h" |
| #include "src/trace_processor/tp_metatrace.h" |
| #include "src/trace_processor/util/status_macros.h" |
| |
| namespace perfetto { |
| namespace trace_processor { |
| |
| namespace { |
| |
| std::string TypeToSqlString(SqlValue::Type type) { |
| switch (type) { |
| case SqlValue::Type::kString: |
| return "TEXT"; |
| case SqlValue::Type::kLong: |
| return "BIGINT"; |
| case SqlValue::Type::kDouble: |
| return "DOUBLE"; |
| case SqlValue::Type::kBytes: |
| return "BLOB"; |
| case SqlValue::Type::kNull: |
| PERFETTO_FATAL("Cannot map unknown column type"); |
| } |
| PERFETTO_FATAL("Not reached"); // For gcc |
| } |
| |
| std::string OpToDebugString(int op) { |
| switch (op) { |
| case SQLITE_INDEX_CONSTRAINT_EQ: |
| return "="; |
| case SQLITE_INDEX_CONSTRAINT_NE: |
| return "!="; |
| case SQLITE_INDEX_CONSTRAINT_GE: |
| return ">="; |
| case SQLITE_INDEX_CONSTRAINT_GT: |
| return ">"; |
| case SQLITE_INDEX_CONSTRAINT_LE: |
| return "<="; |
| case SQLITE_INDEX_CONSTRAINT_LT: |
| return "<"; |
| case SQLITE_INDEX_CONSTRAINT_LIKE: |
| return "like"; |
| case SQLITE_INDEX_CONSTRAINT_ISNULL: |
| return "is null"; |
| case SQLITE_INDEX_CONSTRAINT_ISNOTNULL: |
| return "is not null"; |
| case SQLITE_INDEX_CONSTRAINT_IS: |
| return "is"; |
| case SQLITE_INDEX_CONSTRAINT_ISNOT: |
| return "is not"; |
| case SQLITE_INDEX_CONSTRAINT_GLOB: |
| return "glob"; |
| case SQLITE_INDEX_CONSTRAINT_LIMIT: |
| return "limit"; |
| case SQLITE_INDEX_CONSTRAINT_OFFSET: |
| return "offset"; |
| case SqliteTable::CustomFilterOpcode::kSourceGeqOpCode: |
| return "source_geq"; |
| default: |
| PERFETTO_FATAL("Operator to string conversion not impemented for %d", op); |
| } |
| } |
| |
| void ConstraintsToString(const QueryConstraints& qc, |
| const SqliteTable::Schema& schema, |
| std::string& out) { |
| bool is_first = true; |
| for (const auto& cs : qc.constraints()) { |
| if (!is_first) { |
| out.append(","); |
| } |
| out.append(schema.columns()[static_cast<size_t>(cs.column)].name()); |
| out.append(" "); |
| out.append(OpToDebugString(cs.op)); |
| is_first = false; |
| } |
| } |
| |
| void OrderByToString(const QueryConstraints& qc, |
| const SqliteTable::Schema& schema, |
| std::string& out) { |
| bool is_first = true; |
| for (const auto& ob : qc.order_by()) { |
| if (!is_first) { |
| out.append(","); |
| } |
| out.append(schema.columns()[static_cast<size_t>(ob.iColumn)].name()); |
| out.append(" "); |
| out.append(std::to_string(ob.desc)); |
| is_first = false; |
| } |
| } |
| |
| std::string QcDebugStr(const QueryConstraints& qc, |
| const SqliteTable::Schema& schema) { |
| std::string str_result; |
| str_result.reserve(512); |
| |
| str_result.append("C"); |
| str_result.append(std::to_string(qc.constraints().size())); |
| str_result.append(","); |
| ConstraintsToString(qc, schema, str_result); |
| str_result.append(";"); |
| |
| str_result.append("O"); |
| str_result.append(std::to_string(qc.order_by().size())); |
| str_result.append(","); |
| OrderByToString(qc, schema, str_result); |
| str_result.append(";"); |
| |
| str_result.append("U"); |
| str_result.append(std::to_string(qc.cols_used())); |
| |
| return str_result; |
| } |
| |
| void WriteQueryConstraintsToMetatrace(metatrace::Record* r, |
| const QueryConstraints& qc, |
| const SqliteTable::Schema& schema) { |
| r->AddArg("constraint_count", std::to_string(qc.constraints().size())); |
| std::string constraints; |
| ConstraintsToString(qc, schema, constraints); |
| r->AddArg("constraints", constraints); |
| r->AddArg("order_by_count", std::to_string(qc.order_by().size())); |
| std::string order_by; |
| OrderByToString(qc, schema, order_by); |
| r->AddArg("order_by", order_by); |
| r->AddArg("columns_used", std::to_string(qc.cols_used())); |
| } |
| |
| } // namespace |
| |
| // static |
| bool SqliteTable::debug = false; |
| |
| SqliteTable::SqliteTable() = default; |
| SqliteTable::~SqliteTable() = default; |
| |
| base::Status SqliteTable::ModifyConstraints(QueryConstraints*) { |
| return base::OkStatus(); |
| } |
| |
| int SqliteTable::FindFunction(const char*, FindFunctionFn*, void**) { |
| return 0; |
| } |
| |
| base::Status SqliteTable::Update(int, sqlite3_value**, sqlite3_int64*) { |
| return base::ErrStatus("Updating not supported"); |
| } |
| |
| bool SqliteTable::ReadConstraints(int idxNum, const char* idxStr, int argc) { |
| bool cache_hit = true; |
| if (idxNum != qc_hash_) { |
| qc_cache_ = QueryConstraints::FromString(idxStr); |
| qc_hash_ = idxNum; |
| cache_hit = false; |
| } |
| |
| PERFETTO_TP_TRACE(metatrace::Category::QUERY, "SQLITE_TABLE_READ_CONSTRAINTS", |
| [&](metatrace::Record* r) { |
| r->AddArg("cache_hit", std::to_string(cache_hit)); |
| r->AddArg("name", name_); |
| WriteQueryConstraintsToMetatrace(r, qc_cache_, schema_); |
| r->AddArg("raw_constraints", idxStr); |
| r->AddArg("argc", std::to_string(argc)); |
| }); |
| |
| // Logging this every ReadConstraints just leads to log spam on joins making |
| // it unusable. Instead, only print this out when we miss the cache (which |
| // happens precisely when the constraint set from SQLite changes.) |
| if (SqliteTable::debug && !cache_hit) { |
| PERFETTO_LOG("[%s::ParseConstraints] constraints=%s argc=%d", name_.c_str(), |
| QcDebugStr(qc_cache_, schema_).c_str(), argc); |
| } |
| return cache_hit; |
| } |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| // SqliteTable::BaseCursor implementation |
| //////////////////////////////////////////////////////////////////////////////// |
| |
| SqliteTable::BaseCursor::BaseCursor(SqliteTable* table) : table_(table) { |
| // This is required to prevent us from leaving this field uninitialised if |
| // we ever move construct the Cursor. |
| pVtab = table; |
| } |
| SqliteTable::BaseCursor::~BaseCursor() = default; |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| // SqliteTable::Column implementation |
| //////////////////////////////////////////////////////////////////////////////// |
| |
| SqliteTable::Column::Column(size_t index, |
| std::string name, |
| SqlValue::Type type, |
| bool hidden) |
| : index_(index), name_(name), type_(type), hidden_(hidden) {} |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| // SqliteTable::Schema implementation |
| //////////////////////////////////////////////////////////////////////////////// |
| |
| SqliteTable::Schema::Schema() = default; |
| |
| SqliteTable::Schema::Schema(std::vector<Column> columns, |
| std::vector<size_t> primary_keys) |
| : columns_(std::move(columns)), primary_keys_(std::move(primary_keys)) { |
| for (size_t i = 0; i < columns_.size(); i++) { |
| PERFETTO_CHECK(columns_[i].index() == i); |
| } |
| for (auto key : primary_keys_) { |
| PERFETTO_CHECK(key < columns_.size()); |
| } |
| } |
| |
| SqliteTable::Schema::Schema(const Schema&) = default; |
| SqliteTable::Schema& SqliteTable::Schema::operator=(const Schema&) = default; |
| |
| std::string SqliteTable::Schema::ToCreateTableStmt() const { |
| std::string stmt = "CREATE TABLE x("; |
| for (size_t i = 0; i < columns_.size(); ++i) { |
| const Column& col = columns_[i]; |
| stmt += " " + col.name(); |
| |
| if (col.type() != SqlValue::Type::kNull) { |
| stmt += " " + TypeToSqlString(col.type()); |
| } else if (std::find(primary_keys_.begin(), primary_keys_.end(), i) != |
| primary_keys_.end()) { |
| PERFETTO_FATAL("Unknown type for primary key column %s", |
| col.name().c_str()); |
| } |
| if (col.hidden()) { |
| stmt += " HIDDEN"; |
| } |
| stmt += ","; |
| } |
| stmt += " PRIMARY KEY("; |
| for (size_t i = 0; i < primary_keys_.size(); i++) { |
| if (i != 0) |
| stmt += ", "; |
| stmt += columns_[primary_keys_[i]].name(); |
| } |
| stmt += ")) WITHOUT ROWID;"; |
| return stmt; |
| } |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| // TypedSqliteTableBase implementation |
| //////////////////////////////////////////////////////////////////////////////// |
| |
| TypedSqliteTableBase::~TypedSqliteTableBase() = default; |
| |
| base::Status TypedSqliteTableBase::DeclareAndAssignVtab( |
| std::unique_ptr<SqliteTable> table, |
| sqlite3_vtab** tab) { |
| auto create_stmt = table->schema().ToCreateTableStmt(); |
| PERFETTO_DLOG("Create table statement: %s", create_stmt.c_str()); |
| RETURN_IF_ERROR(table->engine_->DeclareVirtualTable(create_stmt)); |
| *tab = table.release(); |
| return base::OkStatus(); |
| } |
| |
| int TypedSqliteTableBase::xDestroy(sqlite3_vtab* t) { |
| auto* table = static_cast<SqliteTable*>(t); |
| table->engine_->OnSqliteTableDestroyed(table->name_); |
| delete table; |
| return SQLITE_OK; |
| } |
| |
| int TypedSqliteTableBase::xDestroyFatal(sqlite3_vtab*) { |
| PERFETTO_FATAL("xDestroy should not be called"); |
| } |
| |
| int TypedSqliteTableBase::xConnectRestoreTable(sqlite3*, |
| void* arg, |
| int, |
| const char* const* argv, |
| sqlite3_vtab** tab, |
| char** pzErr) { |
| auto* xArg = static_cast<BaseModuleArg*>(arg); |
| |
| // SQLite guarantees that argv[2] contains the name of the table. |
| std::string table_name = argv[2]; |
| base::StatusOr<std::unique_ptr<SqliteTable>> table = |
| xArg->engine->RestoreSqliteTable(table_name); |
| if (!table.status().ok()) { |
| *pzErr = sqlite3_mprintf("%s", table.status().c_message()); |
| return SQLITE_ERROR; |
| } |
| base::Status status = DeclareAndAssignVtab(std::move(table.value()), tab); |
| if (!status.ok()) { |
| *pzErr = sqlite3_mprintf("%s", status.c_message()); |
| return SQLITE_ERROR; |
| } |
| return SQLITE_OK; |
| } |
| |
| int TypedSqliteTableBase::xDisconnectSaveTable(sqlite3_vtab* t) { |
| auto* table = static_cast<TypedSqliteTableBase*>(t); |
| base::Status status = table->engine_->SaveSqliteTable( |
| table->name(), std::unique_ptr<SqliteTable>(table)); |
| return table->SetStatusAndReturn(status); |
| } |
| |
| base::Status TypedSqliteTableBase::InitInternal(SqliteEngine* engine, |
| int argc, |
| const char* const* argv) { |
| // Set the engine to allow saving into it later. |
| engine_ = engine; |
| |
| // SQLite guarantees that argv[0] will be the "module" name: this is the |
| // same as |table_name| passed to the Register function. |
| module_name_ = argv[0]; |
| |
| // SQLite guarantees that argv[2] contains the name of the table: for |
| // non-arg taking tables, this will be the same as |table_name| but for |
| // arg-taking tables, this will be the table name as defined by the |
| // user in the CREATE VIRTUAL TABLE call. |
| name_ = argv[2]; |
| |
| Schema schema; |
| RETURN_IF_ERROR(Init(argc, argv, &schema)); |
| schema_ = std::move(schema); |
| return base::OkStatus(); |
| } |
| |
| int TypedSqliteTableBase::xOpen(sqlite3_vtab* t, |
| sqlite3_vtab_cursor** ppCursor) { |
| auto* table = static_cast<TypedSqliteTableBase*>(t); |
| *ppCursor = |
| static_cast<sqlite3_vtab_cursor*>(table->CreateCursor().release()); |
| return SQLITE_OK; |
| } |
| |
| int TypedSqliteTableBase::xBestIndex(sqlite3_vtab* t, sqlite3_index_info* idx) { |
| auto* table = static_cast<TypedSqliteTableBase*>(t); |
| |
| QueryConstraints qc(idx->colUsed); |
| |
| for (int i = 0; i < idx->nConstraint; i++) { |
| const auto& cs = idx->aConstraint[i]; |
| if (!cs.usable) |
| continue; |
| qc.AddConstraint(cs.iColumn, cs.op, i); |
| } |
| |
| for (int i = 0; i < idx->nOrderBy; i++) { |
| int column = idx->aOrderBy[i].iColumn; |
| bool desc = idx->aOrderBy[i].desc; |
| qc.AddOrderBy(column, desc); |
| } |
| |
| int ret = table->SetStatusAndReturn(table->ModifyConstraints(&qc)); |
| if (ret != SQLITE_OK) |
| return ret; |
| |
| BestIndexInfo info; |
| info.estimated_cost = idx->estimatedCost; |
| info.estimated_rows = idx->estimatedRows; |
| info.sqlite_omit_constraint.resize(qc.constraints().size()); |
| |
| ret = table->BestIndex(qc, &info); |
| |
| if (ret != SQLITE_OK) |
| return ret; |
| |
| idx->orderByConsumed = qc.order_by().empty() || info.sqlite_omit_order_by; |
| idx->estimatedCost = info.estimated_cost; |
| idx->estimatedRows = info.estimated_rows; |
| |
| // First pass: mark all constraints as omitted to ensure that any pruned |
| // constraints are not checked for by SQLite. |
| for (int i = 0; i < idx->nConstraint; ++i) { |
| auto& u = idx->aConstraintUsage[i]; |
| u.omit = true; |
| } |
| |
| // Second pass: actually set the correct omit and index values for all |
| // retained constraints. |
| for (uint32_t i = 0; i < qc.constraints().size(); ++i) { |
| auto& u = idx->aConstraintUsage[qc.constraints()[i].a_constraint_idx]; |
| u.omit = info.sqlite_omit_constraint[i]; |
| u.argvIndex = static_cast<int>(i) + 1; |
| } |
| |
| PERFETTO_TP_TRACE( |
| metatrace::Category::QUERY, "SQLITE_TABLE_BEST_INDEX", |
| [&](metatrace::Record* r) { |
| r->AddArg("name", table->name()); |
| WriteQueryConstraintsToMetatrace(r, qc, table->schema()); |
| r->AddArg("order_by_consumed", std::to_string(idx->orderByConsumed)); |
| r->AddArg("estimated_cost", std::to_string(idx->estimatedCost)); |
| r->AddArg("estimated_rows", |
| std::to_string(static_cast<int64_t>(idx->estimatedRows))); |
| }); |
| |
| auto out_qc_str = qc.ToNewSqlite3String(); |
| if (SqliteTable::debug) { |
| PERFETTO_LOG( |
| "[%s::BestIndex] constraints=%s orderByConsumed=%d estimatedCost=%f " |
| "estimatedRows=%" PRId64, |
| table->name().c_str(), QcDebugStr(qc, table->schema()).c_str(), |
| idx->orderByConsumed, idx->estimatedCost, |
| static_cast<int64_t>(idx->estimatedRows)); |
| } |
| |
| idx->idxStr = out_qc_str.release(); |
| idx->needToFreeIdxStr = true; |
| idx->idxNum = ++table->best_index_num_; |
| |
| return SQLITE_OK; |
| } |
| |
| } // namespace trace_processor |
| } // namespace perfetto |