init push

This commit is contained in:
oceanbase-admin
2021-05-31 22:56:52 +08:00
commit cea7de1475
7020 changed files with 5689869 additions and 0 deletions

View File

@ -0,0 +1,394 @@
/**
* 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.
*/
#define USING_LOG_PREFIX SHARE
#include "ob_sequence_cache.h"
#include "share/schema/ob_schema_struct.h"
#include "share/ob_worker.h"
using namespace oceanbase::common;
using namespace oceanbase::common::number;
using namespace oceanbase::share;
using namespace oceanbase::share::schema;
ObSequenceCache::ObSequenceCache() : inited_(false)
{}
int ObSequenceCache::init(share::schema::ObMultiVersionSchemaService& schema_service, common::ObMySQLProxy& sql_proxy)
{
int ret = OB_SUCCESS;
// const int64_t SEQUENCE_CACHE_BUCKET_SIZE = 1024;
dml_proxy_.init(schema_service, sql_proxy);
inited_ = true;
// ret = sequence_cache_.create(SEQUENCE_CACHE_BUCKET_SIZE, ObModIds::OB_SCHEMA_SEQUENCE);
ret = sequence_cache_.init();
return ret;
}
ObSequenceCache& ObSequenceCache::get_instance()
{
static ObSequenceCache instance_;
return instance_;
}
int ObSequenceCache::move_next(
const ObSequenceSchema& schema, ObSequenceCacheItem& cache, ObIAllocator& allocator, ObSequenceValue& nextval)
{
int ret = OB_SUCCESS;
bool need_refill = false;
if (OB_FAIL(need_refill_cache(schema, cache, allocator, need_refill))) {
LOG_WARN("fail check if need refill cache", K(schema), K(ret));
} else if (need_refill) {
ret = OB_SIZE_OVERFLOW;
} else {
if (OB_UNLIKELY(!cache.base_on_last_number_)) {
// nextval = cache.curr_node_.start();
if (OB_SUCC(nextval.set(cache.curr_node_.start()))) {
cache.base_on_last_number_ = true;
}
} else if (OB_UNLIKELY(schema.get_cycle_flag() && // cycle case
((schema.get_increment_by() > static_cast<int64_t>(0) &&
cache.curr_node_.start() < cache.last_number()) ||
(schema.get_increment_by() < static_cast<int64_t>(0) &&
cache.curr_node_.start() > cache.last_number())))) {
// cycle shows up when start < last_number
// nextval = cache.curr_node_.start();
if (OB_FAIL(nextval.set(cache.curr_node_.start()))) {
LOG_WARN("fail deep copy node value", K(ret));
}
} else {
ObNumber new_start;
if (OB_FAIL(cache.last_number().add(schema.get_increment_by(), new_start, allocator))) {
LOG_WARN("fail calc new_start", K(ret));
} else if (schema.get_increment_by() > static_cast<int64_t>(0)) {
//
// last start
// |____o________|_ _ _ _ o'_ _ _ _ _|_____o''___________
//
//
if (new_start > cache.curr_node_.start()) {
if (OB_FAIL(cache.curr_node_.set_start(new_start))) {
LOG_WARN("fail update new_start value to cache.curr_node_", K(ret));
}
} else {
// start unchanged, use start as nextval
}
} else {
if (new_start < cache.curr_node_.start()) {
if (OB_FAIL(cache.curr_node_.set_start(new_start))) {
LOG_WARN("fail update new_start value to cache.curr_node_", K(ret));
}
} else {
// start unchanged, use start as nextval
}
}
if (OB_SUCC(ret)) {
if (OB_FAIL(nextval.set(cache.curr_node_.start()))) {
LOG_WARN("fail deep copy node value", K(ret));
}
}
}
if (OB_SUCC(ret)) {
if (OB_FAIL(cache.set_last_number(nextval.val()))) {
LOG_WARN("fail to cache last_number", K(nextval), K(ret));
}
}
}
return ret;
}
int ObSequenceCache::need_refill_cache(
const ObSequenceSchema& schema, ObSequenceCacheItem& cache, common::ObIAllocator& allocator, bool& refill)
{
int ret = OB_SUCCESS;
refill = false;
if (OB_UNLIKELY(cache.curr_node_.start() == cache.curr_node_.end())) {
refill = true; // cache not init
} else if (OB_UNLIKELY(!cache.base_on_last_number_)) {
refill = false;
} else if (schema.get_increment_by() > static_cast<int64_t>(0)) {
if (OB_UNLIKELY(cache.curr_node_.start() < cache.last_number())) {
refill = false;
} else {
// refill = (cache.curr_node_.end() - cache.last_number() <= schema.get_increment_by());
ObNumber diff;
if (OB_FAIL(cache.curr_node_.end().sub(cache.last_number(), diff, allocator))) {
LOG_ERROR("fail sub number, unexpected", K(ret));
} else {
refill = (diff <= schema.get_increment_by());
}
}
if (refill && cache.with_prefetch_node_ && OB_SUCC(ret)) {
if (OB_FAIL(cache.combine_prefetch_node())) {
LOG_WARN("fail combine prefetch node", K(ret));
} else if (cache.curr_node_.start() < cache.last_number()) {
refill = false;
} else {
// refill = (cache.curr_node_.end() - cache.last_number() <= schema.get_increment_by());
ObNumber diff;
if (OB_FAIL(cache.curr_node_.end().sub(cache.last_number(), diff, allocator))) {
LOG_ERROR("fail sub number, unexpected", K(ret));
} else {
refill = (diff <= schema.get_increment_by());
}
}
LOG_INFO("after combine prefetch node",
"id",
schema.get_sequence_id(),
"increment_by",
schema.get_increment_by().format(),
"inclusive_start",
cache.curr_node_.start().format(),
"exclusive_end",
cache.curr_node_.end().format(),
"last_number",
cache.last_number().format(),
K(refill),
K(ret));
}
} else {
if (OB_UNLIKELY(cache.curr_node_.start() > cache.last_number())) {
refill = false;
} else {
// refill = (cache.curr_node_.end() - cache.last_number() >= schema.get_increment_by());
ObNumber diff;
if (OB_FAIL(cache.curr_node_.end().sub(cache.last_number(), diff, allocator))) {
LOG_ERROR("fail sub number, unexpected", K(ret));
} else {
refill = (diff >= schema.get_increment_by());
}
}
if (refill && cache.with_prefetch_node_ && OB_SUCC(ret)) {
if (OB_FAIL(cache.combine_prefetch_node())) {
LOG_WARN("fail combine prefetch node", K(ret));
} else if (cache.curr_node_.start() > cache.last_number()) {
refill = false;
} else {
// refill = (cache.curr_node_.end() - cache.last_number() >= schema.get_increment_by());
ObNumber diff;
if (OB_FAIL(cache.curr_node_.end().sub(cache.last_number(), diff, allocator))) {
LOG_ERROR("fail sub number, unexpected", K(ret));
} else {
refill = (diff >= schema.get_increment_by());
}
}
LOG_INFO("after combine prefetch node",
"id",
schema.get_sequence_id(),
"increment_by",
schema.get_increment_by().format(),
"inclusive_start",
cache.curr_node_.start().format(),
"exclusive_end",
cache.curr_node_.end().format(),
"last_number",
cache.last_number().format(),
K(refill));
}
}
return ret;
}
int ObSequenceCache::refill_sequence_cache(
const ObSequenceSchema& schema, common::ObIAllocator& allocator, ObSequenceCacheItem& cache)
{
int ret = OB_SUCCESS;
SequenceCacheNode next_range;
bool need_refetch = false;
int times = 0;
ObNumber next_number;
do {
times++;
need_refetch = false;
if (OB_FAIL(dml_proxy_.next_batch(
schema.get_tenant_id(), schema.get_sequence_id(), schema.get_sequence_option(), next_range))) {
LOG_WARN("fail get next sequence batch", K(schema), K(ret));
} else {
if (schema.get_cycle_flag() && cache.base_on_last_number_) {
if (schema.get_increment_by() > static_cast<int64_t>(0)) {
if (cache.curr_node_.start() > next_range.start()) {
cache.base_on_last_number_ = false;
LOG_INFO("got next batch in a new cycle", K(cache));
} else if (OB_FAIL(cache.last_number().add(schema.get_increment_by(), next_number, allocator))) {
LOG_WARN("fail add numbers", K(ret));
} else if (next_number >= next_range.end()) {
need_refetch = true;
cache.base_on_last_number_ = false;
LOG_INFO("next batch not enough, need refetch", K(need_refetch), K(cache), K(schema));
}
} else {
if (cache.curr_node_.start() < next_range.start()) {
cache.base_on_last_number_ = false;
LOG_INFO("got next batch in a new cycle", K(cache));
} else if (OB_FAIL(cache.last_number().add(schema.get_increment_by(), next_number, allocator))) {
LOG_WARN("fail add numbers", K(ret));
} else if (next_number <= next_range.end()) {
need_refetch = true;
cache.base_on_last_number_ = false;
LOG_INFO("next batch not enough, need refetch", K(need_refetch), K(cache), K(schema));
}
}
}
if (!need_refetch) {
if (OB_FAIL(cache.curr_node_.set_start(next_range.start()))) {
LOG_WARN("fail set start", K(next_range), K(ret));
} else if (OB_FAIL(cache.curr_node_.set_end(next_range.end()))) {
LOG_WARN("fail set end", K(next_range), K(ret));
} else {
LOG_INFO("update sequence curr_node cache success", K(cache));
}
cache.last_refresh_ts_ = ObTimeUtility::current_time();
}
}
if (times == 2) {
LOG_INFO("refetch next batch result", K(cache), K(ret));
}
} while (OB_SUCC(ret) && need_refetch);
return ret;
}
int ObSequenceCache::prefetch_sequence_cache(const ObSequenceSchema& schema, ObSequenceCacheItem& cache)
{
int ret = OB_SUCCESS;
if (OB_FAIL(dml_proxy_.prefetch_next_batch(
schema.get_tenant_id(), schema.get_sequence_id(), schema.get_sequence_option(), cache.prefetch_node_))) {
LOG_WARN("fail get next sequence batch", K(schema), K(ret));
} else {
cache.last_refresh_ts_ = ObTimeUtility::current_time();
LOG_INFO("prefetch sequence", "id", schema.get_sequence_id(), K(cache));
}
return ret;
}
int ObSequenceCache::get_item(CacheItemKey& key, ObSequenceCacheItem*& item)
{
int ret = OB_SUCCESS;
if (OB_ENTRY_NOT_EXIST == (ret = sequence_cache_.get(key, item))) {
lib::ObMutexGuard guard(cache_mutex_);
if (OB_ENTRY_NOT_EXIST == (ret = sequence_cache_.get(key, item))) {
if (NULL == (item = op_alloc(ObSequenceCacheItem))) {
ret = OB_ALLOCATE_MEMORY_FAILED;
LOG_ERROR("failed to alloc cache node", K(key), K(ret));
} else if (OB_FAIL(sequence_cache_.insert_and_get(key, item))) {
LOG_WARN("fail set cache item", K(key), K(ret));
}
if (OB_FAIL(ret) && nullptr != item) {
op_free(item);
item = nullptr;
}
}
}
return ret;
}
int ObSequenceCache::nextval(const ObSequenceSchema& schema, ObIAllocator& allocator, ObSequenceValue& nextval)
{
int ret = OB_SUCCESS;
/* 1. if cache item not exist, create a obsolete cache item
* 2. read and lock cache item
* 3. if cache item obsolete
* - update cache item value via sql
* - read cache item
* - if cache item obsolete, fail
* 4. update cache item
* 5. cleanup: unlock cache item
*/
bool need_prefetch = false;
CacheItemKey key(schema.get_sequence_id());
ObSequenceCacheItem* item = nullptr;
if (OB_FAIL(get_item(key, item))) {
LOG_WARN("fail get item", K(key), K(ret));
} else if (OB_ISNULL(item)) {
ret = OB_ERR_UNEXPECTED;
} else {
lib::ObMutexGuard guard(item->alloc_mutex_);
share::DisableSchedInterGuard sched_guard;
{
LOG_DEBUG("nextval", K(schema));
// step 1. get next val in cache
ret = move_next(schema, *item, allocator, nextval);
// setp 2. cache resources used up, refill cache
if (OB_SIZE_OVERFLOW == ret) {
LOG_INFO("no more avaliable value in current cache, try refill cache", K(*item), K(ret));
if (OB_FAIL(refill_sequence_cache(schema, allocator, *item))) {
LOG_WARN("fail refill sequence cache", K(*item), K(ret));
} else if (OB_FAIL(move_next(schema, *item, allocator, nextval))) {
LOG_WARN("fail move next", K(*item), K(ret));
}
}
// step 3. try prefetch
if (OB_SUCC(ret) && !item->prefetching_ &&
schema.get_cache_size() > static_cast<int64_t>(1) && /* cache size = 1, disable prefetch */
schema.get_order_flag() == false /* order prefetch */) {
if (OB_UNLIKELY(!item->with_prefetch_node_)) {
// const int64_t rest = std::abs(item->curr_node_.end() - item->curr_node_.start());
// const int64_t full = std::abs(schema.get_increment_by() * schema.get_cache_size());
ObNumber rest;
ObNumber full;
ObNumberCalc calc(item->curr_node_.end(), allocator);
// magic value
static const int64_t PREFETCH_OP_THRESHOLD = 4;
//
// const int64_t rest = std::abs(item->curr_node_.end_ - item->curr_node_.start_) * PREFETCH_OP_THRESHOLD;
// const int64_t full = std::abs(schema.get_increment_by() * schema.get_cache_size());
// if (rest > full) {
// enable prefetch
// }
//
if (OB_FAIL(calc.sub(item->curr_node_.start()).mul(PREFETCH_OP_THRESHOLD).get_result(rest))) {
LOG_WARN("fail do number sub", K(ret));
} else if (OB_FAIL(schema.get_increment_by().mul(schema.get_cache_size(), full, allocator))) {
LOG_WARN("fail do number multiply", K(ret));
} else if (rest.abs() > full.abs()) {
item->prefetching_ = true;
need_prefetch = true;
}
}
}
}
if (OB_SUCC(ret) && need_prefetch) {
ObSequenceCacheItem mock_item;
if (OB_FAIL(prefetch_sequence_cache(schema, mock_item))) {
int prefetch_err = ret;
ret = OB_SUCCESS;
LOG_WARN("fail refill sequence cache. ignore prefrech error", K(prefetch_err), K(ret));
} else {
LOG_INFO("dump item", K(mock_item), K(*item));
if (item->with_prefetch_node_) {
LOG_INFO("new item has been fetched by other, ignore");
} else {
item->last_refresh_ts_ = mock_item.last_refresh_ts_;
item->with_prefetch_node_ = true;
if (OB_FAIL(item->prefetch_node_.set_start(mock_item.prefetch_node_.start()))) {
LOG_WARN("fail set start for pretch node", K(ret));
} else if (OB_FAIL(item->prefetch_node_.set_end(mock_item.prefetch_node_.end()))) {
LOG_WARN("fail set end for pretch node", K(ret));
}
}
}
item->prefetching_ = false;
}
}
if (nullptr != item) {
sequence_cache_.revert(item);
}
return ret;
}