/* * 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_SQLITE_QUERY_CONSTRAINTS_H_ #define SRC_TRACE_PROCESSOR_SQLITE_QUERY_CONSTRAINTS_H_ #include #include #include "perfetto/ext/base/scoped_file.h" namespace perfetto { namespace trace_processor { // This class stores the constraints (including the order-by information) for // a query on a sqlite3 virtual table and handles their de/serialization into // strings. // This is because the constraint columns and the order-by clauses are passed // to the xBestIndex method but the constraint values are available only in the // xFilter method. Unfortunately sqlite vtable API don't give any hint about // the validity of the constraints (i.e. constraints passed to xBestIndex can // be used by future xFilter calls in the far future). The only mechanism // offered by sqlite is the idxStr string which is returned by the vtable // in the xBestIndex call and passed to each corresponding xFilter call. class QueryConstraints { public: struct Constraint { // Column this constraint refers to. int column; // SQLite op for the constraint. int op; // The original index of this constraint in the aConstraint array. // Used internally by SqliteTable for xBestIndex - this should not be // read or modified by subclasses of SqliteTable. int a_constraint_idx; }; struct OrderBy { int iColumn; unsigned char desc; }; static int FreeSqliteString(char* resource); using SqliteString = base::ScopedResource; explicit QueryConstraints( uint64_t cols_used = std::numeric_limits::max()); ~QueryConstraints(); QueryConstraints(QueryConstraints&&) noexcept; QueryConstraints& operator=(QueryConstraints&&) noexcept; // Two QueryConstraints with the same constraint and orderby vectors // are equal. bool operator==(const QueryConstraints& other) const; void AddConstraint(int column, unsigned char op, int aconstraint_idx); void AddOrderBy(int column, unsigned char desc); void ClearOrderBy() { order_by_.clear(); } // Converts the constraints and order by information to a string for // use by sqlite. SqliteString ToNewSqlite3String() const; // Deserializes the string into QueryConstraints. String given is in the form // C{# of constraints},col1,op1,col2,op2...,O{# of order by},col1,desc1... // For example C1,0,3,O2,1,0,4,1 static QueryConstraints FromString(const char* idxStr); const std::vector& order_by() const { return order_by_; } const std::vector& constraints() const { return constraints_; } std::vector* mutable_order_by() { return &order_by_; } std::vector* mutable_constraints() { return &constraints_; } uint64_t cols_used() const { return cols_used_; } private: QueryConstraints(const QueryConstraints&) = delete; QueryConstraints& operator=(const QueryConstraints&) = delete; std::vector order_by_; std::vector constraints_; // Stores information about which column is used by this query. // If the lowest bit of is set, the first column is used. The second lowest // bit corresponds to the second column etc. If the most significant bit is // set, that means that any column after the first 63 columns could be used. uint64_t cols_used_ = std::numeric_limits::max(); }; } // namespace trace_processor } // namespace perfetto #endif // SRC_TRACE_PROCESSOR_SQLITE_QUERY_CONSTRAINTS_H_