Files
oceanbase/src/sql/plan_cache/ob_plan_cache_value.h
gm 4a92b6d7df reformat source code
according to code styles, 'AccessModifierOffset' should be -2.
2021-06-17 10:40:36 +08:00

365 lines
12 KiB
C++

/**
* Copyright (c) 2021 OceanBase
* OceanBase CE is licensed under Mulan PubL v2.
* You can use this software according to the terms and conditions of the Mulan PubL v2.
* You may obtain a copy of Mulan PubL v2 at:
* http://license.coscl.org.cn/MulanPubL-2.0
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
* EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
* MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
* See the Mulan PubL v2 for more details.
*/
#ifndef OCEANBASE_SQL_PLAN_CACHE_OB_PLAN_CACHE_VALUE_
#define OCEANBASE_SQL_PLAN_CACHE_OB_PLAN_CACHE_VALUE_
#include "lib/lock/ob_spin_rwlock.h"
#include "common/object/ob_object.h"
#include "sql/plan_cache/ob_plan_set.h"
#include "sql/engine/ob_physical_plan.h"
namespace test {
class TestPlanSet_basic_Test;
class TestPlanCacheValue_basic_Test;
} // namespace test
namespace oceanbase {
namespace common {
class ObIAllocator;
}
namespace share {
class ObIPartitionLocationCache;
namespace schema {
class ObSchemaGetterGuard;
}
} // namespace share
namespace sql {
class ObExecContext;
class ObPlanCache;
class ObPCVSet;
class ObPhysicalPlan;
class ObTablePartitionInfo;
enum ObPlanCacheItemStatus { PLAN_INVALID, PLAN_EXPIRED, PLAN_VALID };
struct PCVSchemaObj {
uint64_t tenant_id_;
uint64_t database_id_;
int64_t schema_id_;
int64_t schema_version_;
share::schema::ObSchemaType schema_type_;
share::schema::ObTableType table_type_;
common::ObString table_name_;
bool is_tmp_table_;
bool is_explicit_db_name_;
common::ObIAllocator* inner_alloc_;
PCVSchemaObj()
: tenant_id_(common::OB_INVALID_ID),
database_id_(common::OB_INVALID_ID),
schema_id_(common::OB_INVALID_ID),
schema_version_(0),
schema_type_(share::schema::OB_MAX_SCHEMA),
table_type_(share::schema::MAX_TABLE_TYPE),
is_tmp_table_(false),
is_explicit_db_name_(false),
inner_alloc_(nullptr)
{}
explicit PCVSchemaObj(ObIAllocator* alloc)
: tenant_id_(common::OB_INVALID_ID),
database_id_(common::OB_INVALID_ID),
schema_id_(common::OB_INVALID_ID),
schema_version_(0),
schema_type_(share::schema::OB_MAX_SCHEMA),
table_type_(share::schema::MAX_TABLE_TYPE),
is_tmp_table_(false),
is_explicit_db_name_(false),
inner_alloc_(alloc)
{}
int init(const share::schema::ObTableSchema* schema);
int init_with_version_obj(const share::schema::ObSchemaObjVersion& schema_obj_version);
int init_without_copy_name(const share::schema::ObSimpleTableSchemaV2* schema);
void set_allocator(common::ObIAllocator* alloc)
{
inner_alloc_ = alloc;
}
bool compare_schema(const share::schema::ObTableSchema& schema) const
{
bool ret = false;
ret = tenant_id_ == schema.get_tenant_id() && database_id_ == schema.get_database_id() &&
schema_id_ == schema.get_table_id() && schema_version_ == schema.get_schema_version() &&
table_type_ == schema.get_table_type();
return ret;
}
bool match_compare(const PCVSchemaObj& other) const
{
bool ret = true;
ret = tenant_id_ == other.tenant_id_ && database_id_ == other.database_id_ && table_type_ == other.table_type_;
return ret;
}
bool operator==(const PCVSchemaObj& other) const;
bool operator!=(const PCVSchemaObj& other) const
{
return !operator==(other);
}
void reset();
~PCVSchemaObj();
TO_STRING_KV(K_(tenant_id), K_(database_id), K_(schema_id), K_(schema_version), K_(schema_type), K_(table_type),
K_(table_name), K_(is_tmp_table), K_(is_explicit_db_name));
};
class ObPlanCacheValue : public common::ObDLinkBase<ObPlanCacheValue> {
public:
static const int64_t MAX_SQL_LENGTH = 2048;
static const int32_t MAX_PLAN_PER_SQL = 8;
const static int64_t OB_INFLUENCE_PLAN_SYS_VAR_LEN = 512;
public:
ObPlanCacheValue();
~ObPlanCacheValue()
{
reset();
}
int init(ObPCVSet* pcv_set, const ObCacheObject* cache_obj, ObPlanCacheCtx& pc_ctx);
// alloc and free
void reset();
int32_t get_type()
{
return 0;
}
int match(ObPlanCacheCtx& pc_ctx, const common::ObIArray<PCVSchemaObj>& schema_array, bool& is_same);
static int resolver_params(ObPlanCacheCtx& pc_ctx, stmt::StmtType stmt_type,
const ObIArray<ObCharsetType>& param_charset_type, const ObBitSet<>& neg_param_index,
const ObBitSet<>& not_param_index, const ObBitSet<>& must_be_positive_idx, ObIArray<ObPCParam*>& raw_params,
ParamStore* obj_params);
int resolve_multi_stmt_params(ObPlanCacheCtx& pc_ctx);
static int create_multi_stmt_param_store(
common::ObIAllocator& allocator, int64_t query_num, int64_t param_num, ParamStore& param_store);
static int check_multi_stmt_param_type(ObPlanCacheCtx& pc_ctx, const stmt::StmtType stmt_type,
const ObIArray<ObCharsetType>& param_charset_type, const ObBitSet<>& neg_param_index_,
const ObBitSet<>& not_param_index, const ObBitSet<>& must_be_positive_idx, ParamStore& param_store,
bool& is_valid);
static int check_multi_stmt_not_param_value(const ObIArray<ObFastParserResult>& multi_stmt_fp_results,
const ObIArray<NotParamInfo>& not_param_info, bool& is_same);
static int check_not_param_value(
const ObFastParserResult& fp_result, const ObIArray<NotParamInfo>& not_param_info, bool& is_same);
// choose an appropriate physical plan
int choose_plan(ObPlanCacheCtx& pc_ctx, const common::ObIArray<PCVSchemaObj>& schema_array, ObCacheObject*& plan);
// add a physical plan
int add_plan(ObCacheObject& cache_obj, const common::ObIArray<PCVSchemaObj>& schema_array, ObPlanCacheCtx& pc_ctx);
// this sql contain can't be parameterized value
const common::ObBitSet<>& get_not_param_index() const
{
return not_param_index_;
}
const common::ObBitSet<>& get_neg_param_index() const
{
return neg_param_index_;
}
// remove physical plan(pointer)
// void remove_plan(ObExecContext &exec_context, ObPhysicalPlan &plan);
// remove all reference count to phy plan
int clear_plan_set();
// get phy plan total mem size
int64_t get_mem_size();
int set_pcv_set(ObPCVSet* pcv_set);
ObPCVSet* get_pcv_set() const
{
return pcv_set_;
}
// cut when length of stmt is more than MAX_SQL_LENGTH;
void set_use_global_location_cache(bool use_global_location_cache)
{
use_global_location_cache_ = use_global_location_cache;
}
bool get_use_global_location_cache()
{
return use_global_location_cache_;
}
// StmtStat *get_stmt_stat() { return &stmt_stat_;}
int remove_plan_stat();
void set_tenant_schema_version(int64_t version)
{
tenant_schema_version_ = version;
}
void set_sys_schema_version(int64_t version)
{
sys_schema_version_ = version;
}
void set_merged_version(int64_t version)
{
merged_version_ = version;
}
int64_t get_merged_version() const
{
return merged_version_;
}
bool is_plan_fixed()
{
return outline_state_.is_plan_fixed_;
}
void set_outline_state(const ObOutlineState& outline_state)
{
outline_state_ = outline_state;
}
int set_outline_params_wrapper(const share::schema::ObOutlineParamsWrapper& params)
{
return outline_params_wrapper_.assign(params);
}
char* get_sql_id()
{
return sql_id_;
}
ObIAllocator* get_pc_alloc() const
{
return pc_alloc_;
}
const share::schema::ObMaxConcurrentParam* get_outline_param(int64_t index) const;
/**
* @brief get sessid
*
* @param void
* @retval session id cached in this pcv
*/
uint64_t get_sessid() const
{
return sessid_;
}
// get all dependency schemas, used for get plan
int get_all_dep_schema(ObPlanCacheCtx& pc_ctx, const uint64_t database_id, int64_t& new_schema_version,
bool& need_check_schema, common::ObIArray<PCVSchemaObj>& schema_array);
// get all dependency schemas, used for add plan
static int get_all_dep_schema(share::schema::ObSchemaGetterGuard& schema_guard,
const DependenyTableStore& dep_schema_objs, common::ObIArray<PCVSchemaObj>& schema_array);
int lift_tenant_schema_version(int64_t new_schema_version);
private:
// used for add plan
// check table version, view table version, merged version
int check_value_version_for_add(
const ObCacheObject& cache_obj, const common::ObIArray<PCVSchemaObj>& schema_array, bool& result);
// used for get plan
// check table version, view table version, merged version
int check_value_version_for_get(share::schema::ObSchemaGetterGuard* schema_guard, bool need_check_schema,
const common::ObIArray<PCVSchemaObj>& schema_array, const uint64_t tenant_id, const int64_t merged_version,
bool& result);
bool is_old_table_stat(const common::ObIArray<common::ObOptTableStatVersion>& table_stat_versions);
int get_outline_version(share::schema::ObSchemaGetterGuard& schema_guard, const uint64_t tenant_id,
share::schema::ObSchemaObjVersion& local_outline_version);
int get_partition_location_cache(ObPlanCacheCtx& pc_ctx, share::ObIPartitionLocationCache*& location_cache_used);
int get_outline_param_index(ObExecContext& exec_ctx, int64_t& param_idx) const;
static int handle_varchar_charset(ObCharsetType charset_type, ObIAllocator& allocator, ParseNode*& node);
/**
* @brief if there is a temporary table in dependency tables
* @retval is_contain: true for containing temporary table
*/
bool is_contain_tmp_tbl() const;
/**
* @brief if there is a sys package/type in dependency tables
* @retval is_contain: true for containing sys package/type
*/
bool is_contain_sys_pl_object() const;
/**
* @brief get temp table names in dependency tables
*
* @retval tmp_tbl_names Temp table names
*/
int get_tmp_depend_tbl_names(TmpTableNameArray& tmp_tbl_names);
int create_new_plan_set(const ObPlanSetType plan_set_type, ObPlanSet*& new_plan_set);
void free_plan_set(ObPlanSet* plan_set);
int set_stored_schema_objs(
const DependenyTableStore& dep_table_store, share::schema::ObSchemaGetterGuard* schema_guard);
int check_dep_schema_version(const common::ObIArray<PCVSchemaObj>& schema_array,
const common::ObIArray<PCVSchemaObj*>& pcv_schema_objs, bool& is_old_version);
int match_dep_schema(const ObPlanCacheCtx& pc_ctx, const common::ObIArray<PCVSchemaObj>& schema_array, bool& is_same);
int check_value_version(bool need_check_schema, const share::schema::ObSchemaObjVersion& outline_version,
const ObIArray<PCVSchemaObj>& schema_array, const int64_t merged_version, const TableStatVersions& table_stats,
bool& is_old_version);
int need_check_schema_version(ObPlanCacheCtx& pc_ctx, int64_t& new_schema_version, bool& need_check);
static int rm_space_for_neg_num(ParseNode* param_node, ObIAllocator& allocator);
friend class ::test::TestPlanSet_basic_Test;
friend class ::test::TestPlanCacheValue_basic_Test;
private:
//*********** for match **************
common::ObSEArray<NotParamInfo, 4> not_param_info_;
common::ObBitSet<> not_param_index_;
// two param index are recorded in neg_param_index_;
// 1. negative const, like select -1 from dual, index of -1 is recorded here
// 2. const param that transformed from minus op, like select 1 - 2 from dual, index of '- 2' is recorded here
common::ObBitSet<> neg_param_index_;
// ps mode match will use variables below
common::ObSEArray<PsNotParamInfo, 4> not_param_var_;
common::ObSEArray<ObCharsetType, 4> param_charset_type_;
ObSqlTraits sql_traits_;
//*************************
common::ObString outline_signature_;
common::ObString constructed_sql_;
ObPCVSet* pcv_set_;
common::ObIAllocator* pc_alloc_;
common::SpinRWLock rwlock_;
// plan id
int64_t last_plan_id_;
// a list of plan sets with different param types combination
common::ObDList<ObPlanSet> plan_sets_;
// if there is no virtual table in ObPhysicalPlan, set true(default), or set false
bool use_global_location_cache_;
TableStatVersions table_stat_versions_;
int64_t tenant_schema_version_;
int64_t sys_schema_version_;
int64_t merged_version_;
ObOutlineState outline_state_;
share::schema::ObOutlineParamsWrapper outline_params_wrapper_;
char sql_id_[OB_MAX_SQL_ID_LENGTH + 1];
// session id for temporary table
uint64_t sessid_;
// sess_create_time_ for temporary table
uint64_t sess_create_time_;
// wether this pcv's plans contains sys table (oracle mode)
bool contain_sys_name_table_;
bool use_static_engine_;
bool use_static_engine_conf_;
common::ObFixedArray<PCVSchemaObj*, common::ObIAllocator> stored_schema_objs_;
common::ObBitSet<> must_be_positive_idx_;
stmt::StmtType stmt_type_;
DISALLOW_COPY_AND_ASSIGN(ObPlanCacheValue);
};
} // namespace sql
} // namespace oceanbase
#endif