blob: 4214bb7ea6ae4a4a5b0911ea4b1db8dc222e5c0d [file] [log] [blame]
/*
* Copyright (C) 2019 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_SQLITE_DB_SQLITE_TABLE_H_
#define SRC_TRACE_PROCESSOR_SQLITE_DB_SQLITE_TABLE_H_
#include "src/trace_processor/containers/bit_vector.h"
#include "src/trace_processor/db/table.h"
#include "src/trace_processor/dynamic/dynamic_table_generator.h"
#include "src/trace_processor/sqlite/query_cache.h"
#include "src/trace_processor/sqlite/sqlite_table.h"
namespace perfetto {
namespace trace_processor {
// Implements the SQLite table interface for db tables.
class DbSqliteTable : public SqliteTable {
public:
enum class TableComputation {
// Mode when the table is static (i.e. passed in at construction
// time).
kStatic,
// Mode when table is dynamically computed at filter time.
kDynamic,
};
class Cursor : public SqliteTable::Cursor {
public:
Cursor(DbSqliteTable*, QueryCache*);
Cursor(Cursor&&) noexcept = default;
Cursor& operator=(Cursor&&) = default;
// Implementation of SqliteTable::Cursor.
int Filter(const QueryConstraints& qc,
sqlite3_value** argv,
FilterHistory) override;
int Next() override;
int Eof() override;
int Column(sqlite3_context*, int N) override;
private:
enum class Mode {
kSingleRow,
kTable,
};
// Tries to create a sorted table to cache in |sorted_cache_table_| if the
// constraint set matches the requirements.
void TryCacheCreateSortedTable(const QueryConstraints&, FilterHistory);
const Table* SourceTable() const {
// Try and use the sorted cache table (if it exists) to speed up the
// sorting. Otherwise, just use the original table.
return sorted_cache_table_ ? &*sorted_cache_table_ : upstream_table_;
}
Cursor(const Cursor&) = delete;
Cursor& operator=(const Cursor&) = delete;
DbSqliteTable* db_sqlite_table_ = nullptr;
QueryCache* cache_ = nullptr;
const Table* upstream_table_ = nullptr;
// Only valid for |db_sqlite_table_->computation_| ==
// TableComputation::kDynamic.
std::unique_ptr<Table> dynamic_table_;
// Only valid for Mode::kSingleRow.
base::Optional<uint32_t> single_row_;
// Only valid for Mode::kTable.
base::Optional<Table> db_table_;
base::Optional<Table::Iterator> iterator_;
bool eof_ = true;
// Stores a sorted version of |db_table_| sorted on a repeated equals
// constraint. This allows speeding up repeated subqueries in joins
// significantly.
std::shared_ptr<Table> sorted_cache_table_;
// Stores the count of repeated equality queries to decide whether it is
// wortwhile to sort |db_table_| to create |sorted_cache_table_|.
uint32_t repeated_cache_count_ = 0;
Mode mode_ = Mode::kSingleRow;
std::vector<Constraint> constraints_;
std::vector<Order> orders_;
};
struct QueryCost {
double cost;
uint32_t rows;
};
struct Context {
QueryCache* cache;
Table::Schema schema;
TableComputation computation;
// Only valid when computation == TableComputation::kStatic.
const Table* static_table;
// Only valid when computation == TableComputation::kDynamic.
std::unique_ptr<DynamicTableGenerator> generator;
};
static void RegisterTable(sqlite3* db,
QueryCache* cache,
Table::Schema schema,
const Table* table,
const std::string& name);
static void RegisterTable(sqlite3* db,
QueryCache* cache,
std::unique_ptr<DynamicTableGenerator> generator);
DbSqliteTable(sqlite3*, Context context);
virtual ~DbSqliteTable() override;
// Table implementation.
base::Status Init(int,
const char* const*,
SqliteTable::Schema*) override final;
std::unique_ptr<SqliteTable::Cursor> CreateCursor() override;
int ModifyConstraints(QueryConstraints*) override final;
int BestIndex(const QueryConstraints&, BestIndexInfo*) override final;
// These static functions are useful to allow other callers to make use
// of them.
static SqliteTable::Schema ComputeSchema(const Table::Schema&,
const char* table_name);
static void ModifyConstraints(const Table::Schema&, QueryConstraints*);
static void BestIndex(const Table::Schema&,
uint32_t row_count,
const QueryConstraints&,
BestIndexInfo*);
// static for testing.
static QueryCost EstimateCost(const Table::Schema&,
uint32_t row_count,
const QueryConstraints& qc);
private:
QueryCache* cache_ = nullptr;
Table::Schema schema_;
TableComputation computation_ = TableComputation::kStatic;
// Only valid when computation_ == TableComputation::kStatic.
const Table* static_table_ = nullptr;
// Only valid when computation_ == TableComputation::kDynamic.
std::unique_ptr<DynamicTableGenerator> generator_;
};
} // namespace trace_processor
} // namespace perfetto
#endif // SRC_TRACE_PROCESSOR_SQLITE_DB_SQLITE_TABLE_H_