378 lines
9.5 KiB
C++
378 lines
9.5 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 <gtest/gtest.h>
|
|
#include "lib/thread/thread_mgr.h"
|
|
|
|
using namespace oceanbase;
|
|
using namespace oceanbase::common;
|
|
using namespace oceanbase::lib;
|
|
|
|
class TestTG : public testing::Test
|
|
{
|
|
protected:
|
|
static void SetUpTestCase()
|
|
{
|
|
ASSERT_EQ(OB_SUCCESS, ObTimerService::get_instance().start());
|
|
}
|
|
|
|
static void TearDownTestCase()
|
|
{
|
|
ObTimerService::get_instance().stop();
|
|
ObTimerService::get_instance().wait();
|
|
ObTimerService::get_instance().destroy();
|
|
}
|
|
};
|
|
|
|
class TestTimerTask : public ObTimerTask
|
|
{
|
|
public:
|
|
TestTimerTask() : running_(false), task_run_count_(0) {}
|
|
|
|
void runTimerTask()
|
|
{
|
|
running_ = true;
|
|
++task_run_count_;
|
|
::usleep(500000);
|
|
running_ = false;
|
|
}
|
|
|
|
bool running_;
|
|
int64_t task_run_count_;
|
|
};
|
|
|
|
TEST_F(TestTG, timer)
|
|
{
|
|
int tg_id = TGDefIDs::TEST1;
|
|
TestTimerTask task;
|
|
// start
|
|
ASSERT_EQ(OB_SUCCESS, TG_START(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_SCHEDULE(tg_id, task, 0, true));
|
|
::usleep(250000);
|
|
ASSERT_TRUE(task.running_);
|
|
::usleep(750000);
|
|
ASSERT_EQ(1, task.task_run_count_);
|
|
ASSERT_EQ(OB_SUCCESS, TG_STOP_R(tg_id));
|
|
TG_WAIT_ONLY(tg_id);
|
|
ASSERT_EQ(OB_SUCCESS, TG_CANCEL_R(tg_id, task));
|
|
ASSERT_EQ(OB_SUCCESS, TG_WAIT_R(tg_id));
|
|
// TG_WAIT = wait + destroy
|
|
ASSERT_EQ(OB_ERR_UNEXPECTED, TG_CANCEL_R(tg_id, task));
|
|
|
|
// restart
|
|
ASSERT_EQ(OB_SUCCESS, TG_START(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_SCHEDULE(tg_id, task, 0, true));
|
|
::usleep(250000);
|
|
ASSERT_TRUE(task.running_);
|
|
::usleep(750000);
|
|
ASSERT_EQ(2, task.task_run_count_);
|
|
ASSERT_EQ(OB_SUCCESS, TG_STOP_R(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_WAIT_R(tg_id));
|
|
|
|
ASSERT_TRUE(TG_EXIST(tg_id));
|
|
TG_DESTROY(tg_id);
|
|
ASSERT_FALSE(TG_EXIST(tg_id));
|
|
}
|
|
|
|
class Handler : public TGTaskHandler
|
|
{
|
|
public:
|
|
void handle(void *task) override
|
|
{
|
|
UNUSED(task);
|
|
++handle_count_;
|
|
::usleep(50000);
|
|
}
|
|
|
|
int64_t handle_count_=0;
|
|
};
|
|
|
|
TEST_F(TestTG, queue_thread)
|
|
{
|
|
int tg_id = TGDefIDs::TEST2;
|
|
Handler handler;
|
|
// start
|
|
ASSERT_EQ(OB_SUCCESS, TG_SET_HANDLER(tg_id, handler));
|
|
ASSERT_EQ(OB_SUCCESS, TG_START(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_PUSH_TASK(tg_id, &tg_id));
|
|
::usleep(50000);
|
|
ASSERT_EQ(OB_SUCCESS, TG_STOP_R(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_WAIT_R(tg_id));
|
|
ASSERT_EQ(1, handler.handle_count_);
|
|
|
|
// restart
|
|
ASSERT_EQ(OB_SUCCESS, TG_SET_HANDLER(tg_id, handler));
|
|
ASSERT_EQ(OB_SUCCESS, TG_START(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_PUSH_TASK(tg_id, &tg_id));
|
|
::usleep(50000);
|
|
ASSERT_EQ(OB_SUCCESS, TG_STOP_R(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_WAIT_R(tg_id));
|
|
ASSERT_EQ(2, handler.handle_count_);
|
|
|
|
ASSERT_TRUE(TG_EXIST(tg_id));
|
|
TG_DESTROY(tg_id);
|
|
ASSERT_FALSE(TG_EXIST(tg_id));
|
|
}
|
|
|
|
class MyDTask: public common::IObDedupTask
|
|
{
|
|
public:
|
|
MyDTask() : common::IObDedupTask(common::T_BLOOMFILTER) {}
|
|
virtual int64_t hash() const { return reinterpret_cast<int64_t>(this); }
|
|
virtual bool operator ==(const IObDedupTask &task) const { return this == &task; }
|
|
virtual int64_t get_deep_copy_size() const
|
|
{
|
|
return sizeof(*this);
|
|
}
|
|
virtual IObDedupTask *deep_copy(char *buffer, const int64_t buf_size) const
|
|
{
|
|
UNUSED(buf_size);
|
|
return new (buffer) MyDTask;
|
|
}
|
|
virtual int64_t get_abs_expired_time() const { return 0; }
|
|
virtual int process()
|
|
{
|
|
handle_count_++;
|
|
return OB_SUCCESS;
|
|
}
|
|
TO_STRING_KV(K(""));
|
|
static int64_t handle_count_;
|
|
};
|
|
int64_t MyDTask::handle_count_ = 0;
|
|
|
|
TEST_F(TestTG, dedup_queue)
|
|
{
|
|
int tg_id = TGDefIDs::TEST3;
|
|
MyDTask task;
|
|
// start
|
|
ASSERT_EQ(OB_SUCCESS, TG_START(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_PUSH_TASK(tg_id, task));
|
|
::usleep(50000);
|
|
ASSERT_EQ(OB_SUCCESS, TG_STOP_R(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_WAIT_R(tg_id));
|
|
ASSERT_EQ(1, task.handle_count_);
|
|
|
|
// restart
|
|
ASSERT_EQ(OB_SUCCESS, TG_START(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_PUSH_TASK(tg_id, task));
|
|
::usleep(50000);
|
|
ASSERT_EQ(OB_SUCCESS, TG_STOP_R(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_WAIT_R(tg_id));
|
|
ASSERT_EQ(2, task.handle_count_);
|
|
|
|
ASSERT_TRUE(TG_EXIST(tg_id));
|
|
TG_DESTROY(tg_id);
|
|
ASSERT_FALSE(TG_EXIST(tg_id));
|
|
}
|
|
|
|
class MyRunnable : public TGRunnable
|
|
{
|
|
public:
|
|
void run1() override
|
|
{
|
|
|
|
run_count_++;
|
|
while (!has_set_stop()) {
|
|
::usleep(50000);
|
|
}
|
|
}
|
|
int64_t run_count_=0;
|
|
};
|
|
|
|
TEST_F(TestTG, thread_pool)
|
|
{
|
|
int tg_id = TGDefIDs::TEST4;
|
|
MyRunnable runnable;
|
|
// start
|
|
ASSERT_EQ(OB_SUCCESS, TG_SET_RUNNABLE(tg_id, runnable));
|
|
ASSERT_EQ(OB_SUCCESS, TG_START(tg_id));
|
|
::usleep(50000);
|
|
ASSERT_EQ(OB_SUCCESS, TG_STOP_R(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_WAIT_R(tg_id));
|
|
ASSERT_EQ(1, runnable.run_count_);
|
|
|
|
// restart
|
|
ASSERT_EQ(OB_SUCCESS, TG_SET_RUNNABLE(tg_id, runnable));
|
|
ASSERT_EQ(OB_SUCCESS, TG_START(tg_id));
|
|
::usleep(50000);
|
|
ASSERT_EQ(OB_SUCCESS, TG_STOP_R(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_WAIT_R(tg_id));
|
|
ASSERT_EQ(2, runnable.run_count_);
|
|
|
|
ASSERT_TRUE(TG_EXIST(tg_id));
|
|
TG_DESTROY(tg_id);
|
|
ASSERT_FALSE(TG_EXIST(tg_id));
|
|
}
|
|
|
|
TEST_F(TestTG, reentrant_thread_pool)
|
|
{
|
|
int tg_id = TGDefIDs::TEST8;
|
|
MyRunnable runnable;
|
|
// start
|
|
ASSERT_EQ(OB_SUCCESS, TG_SET_RUNNABLE(tg_id, runnable));
|
|
ASSERT_EQ(OB_SUCCESS, TG_START(tg_id));
|
|
::usleep(50000);
|
|
ASSERT_EQ(OB_SUCCESS, TG_REENTRANT_LOGICAL_START(tg_id));
|
|
::usleep(50000);
|
|
TG_REENTRANT_LOGICAL_STOP(tg_id);
|
|
::usleep(50000);
|
|
TG_REENTRANT_LOGICAL_WAIT(tg_id);
|
|
ASSERT_EQ(1, runnable.run_count_);
|
|
ASSERT_EQ(OB_SUCCESS, TG_REENTRANT_LOGICAL_START(tg_id));
|
|
::usleep(50000);
|
|
ASSERT_EQ(OB_SUCCESS, TG_STOP_R(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_WAIT_R(tg_id));
|
|
ASSERT_EQ(2, runnable.run_count_);
|
|
// restart
|
|
ASSERT_EQ(OB_SUCCESS, TG_SET_RUNNABLE(tg_id, runnable));
|
|
ASSERT_EQ(OB_SUCCESS, TG_START(tg_id));
|
|
::usleep(50000);
|
|
ASSERT_EQ(OB_SUCCESS, TG_STOP_R(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_WAIT_R(tg_id));
|
|
ASSERT_EQ(2, runnable.run_count_);
|
|
|
|
ASSERT_TRUE(TG_EXIST(tg_id));
|
|
TG_DESTROY(tg_id);
|
|
ASSERT_FALSE(TG_EXIST(tg_id));
|
|
}
|
|
class MyTask : public share::ObAsyncTask
|
|
{
|
|
public:
|
|
virtual int process() override
|
|
{
|
|
handle_count_++;
|
|
::usleep(50000);
|
|
return OB_SUCCESS;
|
|
}
|
|
|
|
virtual int64_t get_deep_copy_size() const override
|
|
{ return sizeof(*this); }
|
|
|
|
virtual ObAsyncTask *deep_copy(char *buf, const int64_t buf_size) const override
|
|
{
|
|
UNUSED(buf_size);
|
|
return new (buf) MyTask();
|
|
}
|
|
static int64_t handle_count_;
|
|
};
|
|
int64_t MyTask::handle_count_ = 0;
|
|
|
|
TEST_F(TestTG, async_task_queue)
|
|
{
|
|
int tg_id = TGDefIDs::TEST5;
|
|
MyTask task;
|
|
// start
|
|
ASSERT_EQ(OB_SUCCESS, TG_START(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_PUSH_TASK(tg_id, task));
|
|
::usleep(50000);
|
|
ASSERT_EQ(OB_SUCCESS, TG_STOP_R(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_WAIT_R(tg_id));
|
|
ASSERT_EQ(1, task.handle_count_);
|
|
|
|
// restart
|
|
ASSERT_EQ(OB_SUCCESS, TG_START(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_PUSH_TASK(tg_id, task));
|
|
::usleep(50000);
|
|
ASSERT_EQ(OB_SUCCESS, TG_STOP_R(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_WAIT_R(tg_id));
|
|
ASSERT_EQ(2, task.handle_count_);
|
|
|
|
ASSERT_TRUE(TG_EXIST(tg_id));
|
|
TG_DESTROY(tg_id);
|
|
ASSERT_FALSE(TG_EXIST(tg_id));
|
|
}
|
|
|
|
class MapQueueThreadHandler : public TGTaskHandler
|
|
{
|
|
public:
|
|
void handle(void *task) override
|
|
{}
|
|
void handle(void *task, volatile bool &is_stopped) override
|
|
{
|
|
UNUSED(task);
|
|
UNUSED(is_stopped);
|
|
++handle_count_;
|
|
::usleep(50000);
|
|
}
|
|
|
|
int64_t handle_count_ = 0;
|
|
};
|
|
|
|
TEST_F(TestTG, map_queue_thread)
|
|
{
|
|
int tg_id = TGDefIDs::TEST6;
|
|
MapQueueThreadHandler handler;
|
|
// start
|
|
ASSERT_EQ(OB_SUCCESS, TG_SET_HANDLER(tg_id, handler));
|
|
ASSERT_EQ(OB_SUCCESS, TG_START(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_PUSH_TASK(tg_id, &tg_id, 0));
|
|
::usleep(500000);
|
|
ASSERT_EQ(OB_SUCCESS, TG_STOP_R(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_WAIT_R(tg_id));
|
|
ASSERT_EQ(1, handler.handle_count_);
|
|
|
|
// restart
|
|
ASSERT_EQ(OB_SUCCESS, TG_SET_HANDLER(tg_id, handler));
|
|
ASSERT_EQ(OB_SUCCESS, TG_START(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_PUSH_TASK(tg_id, &tg_id, 1));
|
|
::usleep(500000);
|
|
ASSERT_EQ(OB_SUCCESS, TG_STOP_R(tg_id));
|
|
ASSERT_EQ(OB_SUCCESS, TG_WAIT_R(tg_id));
|
|
ASSERT_EQ(2, handler.handle_count_);
|
|
|
|
ASSERT_TRUE(TG_EXIST(tg_id));
|
|
TG_DESTROY(tg_id);
|
|
ASSERT_FALSE(TG_EXIST(tg_id));
|
|
}
|
|
|
|
TEST_F(TestTG, alloc_tg_id_inc)
|
|
{
|
|
const int max_tg_count = 122880;
|
|
int test_tg_id = 137;
|
|
// occupy all tg_ids
|
|
for (int i = 0; i < test_tg_id; i++) {
|
|
int tg_id = TG_MGR.alloc_tg_id(0);
|
|
TG_MGR.free_tg_id(tg_id);
|
|
}
|
|
int alloc_tg_id = TG_MGR.alloc_tg_id(0);
|
|
EXPECT_EQ(test_tg_id + TGDefIDs::END, alloc_tg_id);
|
|
TG_MGR.free_tg_id(alloc_tg_id);
|
|
}
|
|
|
|
TEST_F(TestTG, alloc_tg_id_overflow)
|
|
{
|
|
const int max_tg_count = 122880;
|
|
// make tg_seq_ overflow
|
|
for (int i = 0; i < max_tg_count; i++) {
|
|
int tg_id = TG_MGR.alloc_tg_id(0);
|
|
TG_MGR.free_tg_id(tg_id);
|
|
if (tg_id == max_tg_count - 1) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
TG_MGR.free_tg_id(0);
|
|
// start from 0 after tg_seq_ overflow
|
|
int alloc_tg_id = TG_MGR.alloc_tg_id(0);
|
|
EXPECT_EQ(alloc_tg_id, 0);
|
|
TG_MGR.free_tg_id(alloc_tg_id);
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
oceanbase::common::ObLogger::get_logger().set_log_level("INFO");
|
|
OB_LOGGER.set_log_level("INFO");
|
|
OB_LOGGER.set_file_name("test_tg_mgr.log", true);
|
|
::testing::InitGoogleTest(&argc, argv);
|
|
return RUN_ALL_TESTS();
|
|
}
|