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

376 lines
10 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.
*/
#include "ob_batch_buffer.h"
#include "lib/allocator/ob_malloc.h"
#include "lib/atomic/atomic128.h"
#include "ob_disk_log_buffer.h"
namespace oceanbase {
using namespace common;
namespace clog {
int ObBatchBuffer::IncPos::try_freeze(IncPos& cur_pos, const int64_t seq)
{
int ret = OB_SUCCESS;
IncPos next_pos;
LOAD128(cur_pos, this);
do {
if (!cur_pos.can_freeze(seq)) {
ret = OB_EAGAIN;
break;
} else {
next_pos.seq_ = cur_pos.seq_ + 1;
next_pos.offset_ = 0;
next_pos.entry_cnt_ = 0;
ret = OB_BLOCK_SWITCHED;
}
} while (!CAS128(this, cur_pos, next_pos));
return ret;
}
int ObBatchBuffer::IncPos::append(IncPos& cur_pos, const int64_t len, const int64_t entry_cnt, const int64_t limit)
{
int ret = OB_SUCCESS;
IncPos next_pos;
LOAD128(cur_pos, this);
do {
const int32_t tmp_entry_cnt = static_cast<int32_t>(entry_cnt);
if (cur_pos.offset_ + len <= limit && cur_pos.entry_cnt_ + tmp_entry_cnt <= MAX_ENTRY_CNT) {
next_pos.seq_ = cur_pos.seq_;
next_pos.offset_ = cur_pos.offset_ + static_cast<offset_t>(len);
next_pos.entry_cnt_ = cur_pos.entry_cnt_ + tmp_entry_cnt;
ret = OB_SUCCESS;
} else {
next_pos.seq_ = cur_pos.seq_ + 1;
next_pos.offset_ = static_cast<offset_t>(len);
next_pos.entry_cnt_ = tmp_entry_cnt;
ret = OB_BLOCK_SWITCHED;
}
} while (!CAS128(this, cur_pos, next_pos));
return ret;
}
ObBatchBuffer::IncPos& ObBatchBuffer::IncPos::next_block()
{
seq_++;
offset_ = 0;
entry_cnt_ = 0;
return *this;
}
class ObBatchBuffer::Block : public ObIBatchBufferTask {
public:
Block(ObBatchBuffer& host, int64_t seq, char* buf, int64_t block_count, bool auto_freeze)
: host_(host),
seq_(seq),
status_seq_(seq),
ref_(0),
buf_(buf),
block_count_(block_count),
auto_freeze_(auto_freeze)
{}
virtual ~Block()
{}
int init(ObLogWriterWrapper* handler);
ObICLogItem* get_flush_task();
int64_t ref(const int64_t delta);
int64_t wait(const int64_t seq);
void fill(const offset_t offset, ObIBufferTask* task);
void freeze(const offset_t offset);
void set_submitted();
void wait_submitted(const int64_t seq);
void after_consume();
int64_t get_seq() const
{
return seq_;
}
void reuse();
private:
ObBatchBuffer& host_;
int64_t seq_;
int64_t status_seq_;
int64_t ref_;
char* buf_;
int64_t block_count_;
ObCLogItem flush_task_;
bool auto_freeze_;
};
int ObBatchBuffer::Block::init(ObLogWriterWrapper* handler)
{
int ret = OB_SUCCESS;
if (OB_FAIL(flush_task_.init(handler, this, &host_))) {
CLOG_LOG(WARN, "flush_task_.init failed", K(ret));
}
return ret;
}
ObICLogItem* ObBatchBuffer::Block::get_flush_task()
{
return &flush_task_;
}
void ObBatchBuffer::Block::reuse()
{
ObIBatchBufferTask::reset();
wait_submitted(seq_);
ATOMIC_STORE(&seq_, seq_ + block_count_);
}
void ObBatchBuffer::Block::after_consume()
{
int tmp_ret = OB_SUCCESS;
reuse();
int64_t next_flush_block_id = seq_ - block_count_ + 1;
host_.update_next_flush_block_id(next_flush_block_id);
if (auto_freeze_ && OB_SUCCESS != (tmp_ret = host_.try_freeze(next_flush_block_id))) {
CLOG_LOG(ERROR, "try_freeze failed", K(tmp_ret));
}
}
int64_t ObBatchBuffer::Block::ref(const int64_t delta)
{
return ATOMIC_AAF(&ref_, delta);
}
int64_t ObBatchBuffer::Block::wait(const int64_t seq)
{
int64_t real_seq = 0;
while ((real_seq = ATOMIC_LOAD(&seq_)) < seq) {
usleep(200);
}
return real_seq;
}
void ObBatchBuffer::Block::fill(const offset_t offset, ObIBufferTask* task)
{
int tmp_ret = OB_SUCCESS;
if (NULL == task || offset < 0) {
tmp_ret = OB_INVALID_ARGUMENT;
CLOG_LOG(WARN, "invalid argument", "ret", tmp_ret, KP(task), K(offset));
} else if (OB_SUCCESS != (tmp_ret = task->fill_buffer(buf_, offset))) {
CLOG_LOG(WARN, "fill_buffer fail", "ret", tmp_ret, KP_(buf), K(offset));
} else {
add_callback_to_list(task);
}
}
void ObBatchBuffer::Block::freeze(const offset_t offset)
{
set_batch_buffer(buf_, offset);
}
void ObBatchBuffer::Block::set_submitted()
{
ATOMIC_STORE(&status_seq_, ATOMIC_LOAD(&seq_) + 1);
}
void ObBatchBuffer::Block::wait_submitted(const int64_t seq)
{
int64_t status_seq = 0;
while ((status_seq = ATOMIC_LOAD(&status_seq_)) < seq + 1) {
usleep(200);
}
return;
}
ObBatchBuffer::ObBatchBuffer()
: is_inited_(false),
handler_(NULL),
block_array_(NULL),
block_count_(OB_INVALID_COUNT),
block_size_(OB_INVALID_SIZE),
next_flush_block_id_(OB_INVALID_ID),
next_pos_(),
auto_freeze_(true)
{}
ObBatchBuffer::~ObBatchBuffer()
{
if (NULL != block_array_) {
for (int64_t i = 0; i < block_count_; i++) {
block_array_[i].~Block();
}
ob_free_align(block_array_);
block_array_ = NULL;
}
handler_ = NULL;
block_count_ = OB_INVALID_COUNT;
block_size_ = OB_INVALID_SIZE;
next_flush_block_id_ = OB_INVALID_ID;
is_inited_ = false;
}
int ObBatchBuffer::init(ObIBufferArena* buffer_pool, ObIBatchBufferConsumer* handler, bool auto_freeze)
{
int ret = OB_SUCCESS;
if (is_inited_) {
ret = OB_INIT_TWICE;
} else if (NULL == buffer_pool || NULL == handler) {
ret = OB_INVALID_ARGUMENT;
} else if (NULL == (block_array_ = (Block*)ob_malloc_align(
16, buffer_pool->get_buffer_count() * sizeof(Block), ObModIds::OB_CLOG_MGR))) {
ret = OB_ALLOCATE_MEMORY_FAILED;
} else {
block_count_ = buffer_pool->get_buffer_count();
block_size_ = buffer_pool->get_buffer_size();
handler_ = handler;
next_flush_block_id_ = 0;
for (int64_t i = 0; i < block_count_; i++) {
new (block_array_ + i) Block(*this, i, buffer_pool->alloc(), block_count_, auto_freeze);
}
auto_freeze_ = auto_freeze;
is_inited_ = true;
}
return ret;
}
int ObBatchBuffer::submit(ObIBufferTask* task)
{
int ret = OB_SUCCESS;
int tmp_ret = OB_SUCCESS;
IncPos cur_pos;
int64_t data_len = 0;
if (!is_inited_) {
ret = OB_NOT_INIT;
} else if (NULL == task || (data_len = task->get_data_len()) <= 0 || data_len >= block_size_) {
ret = OB_INVALID_ARGUMENT;
if (data_len >= block_size_) {
CLOG_LOG(ERROR, "invalid arguments", K(ret), K(data_len), K(block_size_));
} else {
ret = OB_INVALID_BATCH_SIZE;
CLOG_LOG(WARN, "invalid arguments", K(ret), K(data_len), K(block_size_));
}
} else {
int64_t entry_cnt = task->get_entry_cnt();
if (OB_BLOCK_SWITCHED == (ret = next_pos_.append(cur_pos, data_len, entry_cnt, block_size_))) {
ret = fill_buffer(cur_pos, NULL);
cur_pos.next_block();
}
if (OB_SUCC(ret)) {
ret = fill_buffer(cur_pos, task);
}
}
if (OB_SUCC(ret)) {
if (auto_freeze_ && OB_SUCCESS != (tmp_ret = try_freeze(next_flush_block_id_))) {
CLOG_LOG(ERROR, "try_freeze failed", K(tmp_ret), K_(next_flush_block_id));
}
}
return ret;
}
void ObBatchBuffer::update_next_flush_block_id(int64_t block_id)
{
inc_update(next_flush_block_id_, block_id);
}
bool ObBatchBuffer::is_all_consumed() const
{
return next_flush_block_id_ == next_pos_.seq_ && 0 == next_pos_.offset_;
}
int ObBatchBuffer::try_freeze_next_block()
{
int ret = OB_SUCCESS;
if (OB_INVALID_ID != next_flush_block_id_) {
ret = try_freeze(next_flush_block_id_);
}
return ret;
}
int ObBatchBuffer::try_freeze(const int64_t block_id)
{
int ret = OB_SUCCESS;
if (!is_inited_) {
ret = OB_NOT_INIT;
} else if (block_id < 0) {
ret = OB_INVALID_ARGUMENT;
} else {
IncPos cur_pos;
if (OB_BLOCK_SWITCHED == (ret = next_pos_.try_freeze(cur_pos, block_id))) {
if (OB_FAIL(fill_buffer(cur_pos, NULL))) {
CLOG_LOG(WARN, "fill_buffer fail", K(ret));
}
} else if (OB_EAGAIN == ret) {
ret = OB_SUCCESS;
}
}
return ret;
}
int ObBatchBuffer::fill_buffer(const IncPos cur_pos, ObIBufferTask* task)
{
int ret = OB_SUCCESS;
Block* block = NULL;
if (!is_inited_ || NULL == handler_ || NULL == block_array_) {
ret = OB_NOT_INIT;
} else if (NULL == (block = get_block(cur_pos.seq_))) {
ret = OB_ERR_UNEXPECTED;
} else {
int64_t ref_cnt = 0;
ObLogTimerUtility timer;
timer.start_timer();
block->wait(cur_pos.seq_);
timer.finish_timer(__FILE__, __LINE__, CLOG_PERF_WARN_THRESHOLD);
if (NULL != task) {
block->fill(cur_pos.offset_, task);
ref_cnt = block->ref(task->get_data_len());
} else {
block->freeze(cur_pos.offset_);
ref_cnt = block->ref(-cur_pos.offset_);
}
timer.finish_timer(__FILE__, __LINE__, CLOG_PERF_WARN_THRESHOLD);
if (0 == ref_cnt) {
if (OB_FAIL(wait_block(cur_pos.seq_ - 1))) {
CLOG_LOG(ERROR, "wait_block failed", K(ret), "block_id", cur_pos.seq_ - 1);
} else if (OB_FAIL(handler_->submit(block))) {
CLOG_LOG(WARN, "handle submit fail", K(ret));
}
block->set_submitted();
}
timer.finish_timer(__FILE__, __LINE__, CLOG_PERF_WARN_THRESHOLD);
}
return ret;
}
int ObBatchBuffer::wait_block(const int64_t block_id)
{
int ret = OB_SUCCESS;
Block* block = NULL;
if (!is_inited_ || NULL == block_array_) {
ret = OB_NOT_INIT;
} else if (-1 == block_id) {
ret = OB_SUCCESS;
} else if (NULL == (block = get_block(block_id))) {
ret = OB_ERR_UNEXPECTED;
} else {
block->wait_submitted(block_id);
}
return ret;
}
ObBatchBuffer::Block* ObBatchBuffer::get_block(const int64_t block_id)
{
int tmp_ret = OB_SUCCESS;
ObBatchBuffer::Block* ptr_ret = NULL;
if (!is_inited_ || NULL == block_array_ || 0 == block_count_ || OB_INVALID_COUNT == block_count_) {
tmp_ret = OB_NOT_INIT;
CLOG_LOG(WARN, "not init", K(tmp_ret), K_(block_array), K_(block_count));
} else {
ptr_ret = block_array_ + (block_id % block_count_);
}
return ptr_ret;
}
}; // end namespace clog
}; // end namespace oceanbase