blob: 6d6ad13e30f3808d50e4c614e411a34ebcc2b339 [file] [log] [blame]
/*
* Copyright (C) 2023 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_SQLITE_ENGINE_H_
#define SRC_TRACE_PROCESSOR_SQLITE_SQLITE_ENGINE_H_
#include <sqlite3.h>
#include <cstddef>
#include <cstdint>
#include <memory>
#include <optional>
#include <string>
#include <type_traits>
#include <utility>
#include "perfetto/base/logging.h"
#include "perfetto/base/status.h"
#include "perfetto/ext/base/flat_hash_map.h"
#include "perfetto/ext/base/hash.h"
#include "src/trace_processor/sqlite/bindings/sqlite_module.h"
#include "src/trace_processor/sqlite/scoped_db.h"
#include "src/trace_processor/sqlite/sql_source.h"
namespace perfetto::trace_processor {
// Wrapper class around SQLite C API.
//
// The goal of this class is to provide a one-stop-shop mechanism to use SQLite.
// Benefits of this include:
// 1) It allows us to add code which intercepts registration of functions
// and tables and keeps track of this for later lookup.
// 2) Allows easily auditing the SQLite APIs we use making it easy to determine
// what functionality we rely on.
class SqliteEngine {
public:
using Fn = void(sqlite3_context* ctx, int argc, sqlite3_value** argv);
using AggregateFnStep = void(sqlite3_context* ctx,
int argc,
sqlite3_value** argv);
using AggregateFnFinal = void(sqlite3_context* ctx);
using WindowFnStep = void(sqlite3_context* ctx,
int argc,
sqlite3_value** argv);
using WindowFnInverse = void(sqlite3_context* ctx,
int argc,
sqlite3_value** argv);
using WindowFnValue = void(sqlite3_context* ctx);
using WindowFnFinal = void(sqlite3_context* ctx);
using FnCtxDestructor = void(void*);
// Wrapper class for SQLite's |sqlite3_stmt| struct and associated functions.
struct PreparedStatement {
public:
bool Step();
bool IsDone() const;
const char* original_sql() const;
const char* sql() const;
const base::Status& status() const { return status_; }
sqlite3_stmt* sqlite_stmt() const { return stmt_.get(); }
private:
friend class SqliteEngine;
explicit PreparedStatement(ScopedStmt, SqlSource);
ScopedStmt stmt_;
ScopedSqliteString expanded_sql_;
SqlSource sql_source_;
base::Status status_ = base::OkStatus();
};
SqliteEngine();
~SqliteEngine();
SqliteEngine(SqliteEngine&&) noexcept = delete;
SqliteEngine& operator=(SqliteEngine&&) = delete;
// Prepares a SQLite statement for the given SQL.
PreparedStatement PrepareStatement(SqlSource);
// Registers a C++ function to be runnable from SQL.
base::Status RegisterFunction(const char* name,
int argc,
Fn* fn,
void* ctx,
FnCtxDestructor* ctx_destructor,
bool deterministic);
// Registers a C++ aggregate function to be runnable from SQL.
base::Status RegisterAggregateFunction(const char* name,
int argc,
AggregateFnStep* step,
AggregateFnFinal* final,
void* ctx,
FnCtxDestructor* ctx_destructor,
bool deterministic);
// Registers a C++ window function to be runnable from SQL.
base::Status RegisterWindowFunction(const char* name,
int argc,
WindowFnStep* step,
WindowFnInverse* inverse,
WindowFnValue* value,
WindowFnFinal* final,
void* ctx,
FnCtxDestructor* ctx_destructor,
bool deterministic);
// Unregisters a C++ function from SQL.
base::Status UnregisterFunction(const char* name, int argc);
// Registers a SQLite virtual table module with the given name.
template <typename Module>
void RegisterVirtualTableModule(const std::string& module_name,
typename Module::Context* ctx);
// Registers a SQLite virtual table module with the given name.
template <typename Module>
void RegisterVirtualTableModule(const std::string& module_name,
std::unique_ptr<typename Module::Context>);
// Declares a virtual table with SQLite.
base::Status DeclareVirtualTable(const std::string& create_stmt);
// Gets the context for a registered SQL function.
void* GetFunctionContext(const std::string& name, int argc);
sqlite3* db() const { return db_.get(); }
private:
struct FnHasher {
size_t operator()(const std::pair<std::string, int>& x) const {
base::Hasher hasher;
hasher.Update(x.first);
hasher.Update(x.second);
return static_cast<size_t>(hasher.digest());
}
};
std::optional<uint32_t> GetErrorOffset() const;
base::FlatHashMap<std::pair<std::string, int>, void*, FnHasher> fn_ctx_;
ScopedDb db_;
};
} // namespace perfetto::trace_processor
// The rest of this file is just implementation details which we need
// in the header file because it is templated code. We separate it out
// like this to keep the API people actually care about easy to read.
namespace perfetto::trace_processor {
template <typename Module>
void SqliteEngine::RegisterVirtualTableModule(const std::string& module_name,
typename Module::Context* ctx) {
static_assert(std::is_base_of_v<sqlite::Module<Module>, Module>,
"Must subclass sqlite::Module");
int res = sqlite3_create_module_v2(db_.get(), module_name.c_str(),
&Module::kModule, ctx, nullptr);
PERFETTO_CHECK(res == SQLITE_OK);
}
template <typename Module>
void SqliteEngine::RegisterVirtualTableModule(
const std::string& module_name,
std::unique_ptr<typename Module::Context> ctx) {
static_assert(std::is_base_of_v<sqlite::Module<Module>, Module>,
"Must subclass sqlite::Module");
int res = sqlite3_create_module_v2(
db_.get(), module_name.c_str(), &Module::kModule, ctx.release(),
[](void* arg) { delete static_cast<typename Module::Context*>(arg); });
PERFETTO_CHECK(res == SQLITE_OK);
}
} // namespace perfetto::trace_processor
#endif // SRC_TRACE_PROCESSOR_SQLITE_SQLITE_ENGINE_H_