提交 be6c2b45 编写于 作者: L Longda

fixed 22, reimplement b+ tree

Other improvement
1. add code format file
2. add gitignore
3. add memory pool utility
4. remove useless file mpool.h, index_manager.cpp/h
5. rename the interface of disk_buffer_pool basing on the new mem_pool
6. add some logs in db/table/disk_buffer_pool
7. add unit test mem_pool_test/log_test/bplus_tree_test
8. code format all modified files
9. redefine the interface of table_meta
10. change log's  full path as file name
11. add utility function, print_callstack, remove warning of path
上级 f7ea4fc2
---
Language: Cpp
# BasedOnStyle: LLVM
SortIncludes: false
AccessModifierOffset: -2
AlignAfterOpenBracket: DontAlign
AlignConsecutiveAssignments: false
AlignConsecutiveDeclarations: false
DerivePointerAlignment: false
PointerAlignment: Right
ConstructorInitializerIndentWidth: 4
AlignEscapedNewlinesLeft: true
AlignTrailingComments: true
AllowAllParametersOfDeclarationOnNextLine: true
AllowShortBlocksOnASingleLine: true
AllowShortFunctionsOnASingleLine: false
AllowShortIfStatementsOnASingleLine: false
AllowShortCaseLabelsOnASingleLine: false
AllowShortLoopsOnASingleLine: false
AlwaysBreakTemplateDeclarations: true
BinPackArguments: false
AlwaysBreakBeforeMultilineStrings: false
BreakBeforeBinaryOperators: None
BreakBeforeTernaryOperators: true
BreakConstructorInitializersBeforeComma: false
BinPackParameters: true
ColumnLimit: 120
ConstructorInitializerAllOnOneLineOrOnePerLine: true
DerivePointerBinding: true
ExperimentalAutoDetectBinPacking: false
IndentCaseLabels: true
MaxEmptyLinesToKeep: 1
NamespaceIndentation: None
ObjCSpaceBeforeProtocolList: false
PenaltyBreakBeforeFirstCallParameter: 19
PenaltyBreakComment: 200
PenaltyBreakString: 1000
PenaltyBreakFirstLessLess: 120
PenaltyExcessCharacter: 1000000
PenaltyReturnTypeOnItsOwnLine: 200
AlwaysBreakAfterDefinitionReturnType: None
AlwaysBreakAfterReturnType: None
PointerBindsToType: true
Cpp11BracedListStyle: true
Standard: Auto
IndentWidth: 2
TabWidth: 4
UseTab: Never
BreakBeforeBraces: Custom
BraceWrapping:
AfterClass: false
AfterControlStatement: false
AfterEnum: false
AfterFunction: true
AfterNamespace: false
AfterObjCDeclaration: false
AfterStruct: false
AfterUnion: false
AfterExternBlock: false
BeforeCatch: false
BeforeElse: false
IndentBraces: false
SplitEmptyFunction: false
SplitEmptyRecord: false
SplitEmptyNamespace: false
IndentFunctionDeclarationAfterType: true
SpacesInParentheses: false
SpacesInAngles: false
SpaceInEmptyParentheses: false
SpacesInCStyleCastParentheses: false
SpaceAfterControlStatementKeyword: true
SpaceBeforeAssignmentOperators: true
SpacesBeforeTrailingComments: 2
ContinuationIndentWidth: 4
CommentPragmas: '^lint'
MacroBlockBegin: "
END_CATCH_ERROR$"
...
./deps/libevent/*
./deps/googletest/*
./deps/jsoncpp/*
build/*
cmake-build-*/*
.vscode/*
.DS_Store
.idea
compile_commands.json
./.name
./miniob.iml
./vcs.xml
./workspace.xml
./modules.xml
...@@ -226,6 +226,9 @@ char *hex_to_bin(const char *s, char *bin_buff, int *dest_len) { ...@@ -226,6 +226,9 @@ char *hex_to_bin(const char *s, char *bin_buff, int *dest_len) {
} }
bool is_blank(const char *s) { bool is_blank(const char *s) {
if (s == nullptr) {
return true;
}
while (*s != '\0') { while (*s != '\0') {
if (!isspace(*s)) { if (!isspace(*s)) {
return false; return false;
......
...@@ -162,6 +162,11 @@ class LoggerFactory { ...@@ -162,6 +162,11 @@ class LoggerFactory {
extern Log *g_log; extern Log *g_log;
#ifndef __FILE_NAME__
#define __FILE_NAME__ (strrchr(__FILE__,'/')?strrchr(__FILE__,'/')+1:__FILE__)
#endif
#define LOG_HEAD(prefix, level) \ #define LOG_HEAD(prefix, level) \
if (common::g_log) { \ if (common::g_log) { \
time_t now_time; \ time_t now_time; \
...@@ -175,17 +180,17 @@ extern Log *g_log; ...@@ -175,17 +180,17 @@ extern Log *g_log;
common::g_log->rotate(p->tm_year + 1900, p->tm_mon + 1, p->tm_mday); \ common::g_log->rotate(p->tm_year + 1900, p->tm_mon + 1, p->tm_mday); \
} \ } \
snprintf(prefix, sizeof(prefix), "[%s %s %s %s %u]>>", sz_head, \ snprintf(prefix, sizeof(prefix), "[%s %s %s %s %u]>>", sz_head, \
(common::g_log)->prefix_msg(level), __FILE__, \ (common::g_log)->prefix_msg(level), __FILE_NAME__, \
__FUNCTION__, (u32_t)__LINE__); \ __FUNCTION__, (u32_t)__LINE__); \
} }
#define LOG_OUTPUT(level, fmt, ...) \ #define LOG_OUTPUT(level, fmt, ...) \
do { \ do { \
using namespace common; \ using namespace common; \
if (g_log && g_log->check_output(level, __FILE__)) { \ if (g_log && g_log->check_output(level, __FILE_NAME__)) { \
char prefix[ONE_KILO] = {0}; \ char prefix[ONE_KILO] = {0}; \
LOG_HEAD(prefix, level); \ LOG_HEAD(prefix, level); \
g_log->output(level, __FILE__, prefix, fmt, ##__VA_ARGS__); \ g_log->output(level, __FILE_NAME__, prefix, fmt, ##__VA_ARGS__); \
} \ } \
} while (0) } while (0)
......
/* Copyright (c) 2021 Xie Meiyi(xiemeiyi@hust.edu.cn) and OceanBase and/or its affiliates. All rights reserved.
miniob is licensed under Mulan PSL v2.
You can use this software according to the terms and conditions of the Mulan PSL v2.
You may obtain a copy of Mulan PSL v2 at:
http://license.coscl.org.cn/MulanPSL2
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 PSL v2 for more details. */
//
// Created by Longda on 2022/1/28.
//
#include "common/mm/mem_pool.h"
namespace common {
int MemPoolItem::init(int item_size, bool dynamic, int pool_num, int item_num_per_pool)
{
if (pools.empty() == false) {
LOG_WARN("Memory pool has been initialized, but still begin to be initialized, this->name:%s.", this->name.c_str());
return 0;
}
if (item_size <= 0 || pool_num <= 0 || item_num_per_pool <= 0) {
LOG_ERROR("Invalid arguments, item_size:%d, pool_num:%d, item_num_per_pool:%d, this->name:%s.",
item_size,
pool_num,
item_num_per_pool,
this->name.c_str());
return -1;
}
this->item_size = item_size;
this->item_num_per_pool = item_num_per_pool;
// in order to init memory pool, enable dynamic here
this->dynamic = true;
for (int i = 0; i < pool_num; i++) {
if (extend() < 0) {
cleanup();
return -1;
}
}
this->dynamic = dynamic;
LOG_INFO("Extend one pool, this->size:%d, item_size:%d, item_num_per_pool:%d, this->name:%s.",
this->size,
item_size,
item_num_per_pool,
this->name.c_str());
return 0;
}
void MemPoolItem::cleanup()
{
if (pools.empty() == true) {
LOG_WARN("Begin to do cleanup, but there is no memory pool, this->name:%s!", this->name.c_str());
return;
}
MUTEX_LOCK(&this->mutex);
used.clear();
frees.clear();
this->size = 0;
for (std::list<void *>::iterator iter = pools.begin(); iter != pools.end(); iter++) {
void *pool = *iter;
::free(pool);
}
pools.clear();
MUTEX_UNLOCK(&this->mutex);
LOG_INFO("Successfully do cleanup, this->name:%s.", this->name.c_str());
}
int MemPoolItem::extend()
{
if (this->dynamic == false) {
LOG_ERROR("Disable dynamic extend memory pool, but begin to extend, this->name:%s", this->name.c_str());
return -1;
}
MUTEX_LOCK(&this->mutex);
void *pool = malloc(item_num_per_pool * item_size);
if (pool == nullptr) {
MUTEX_UNLOCK(&this->mutex);
LOG_ERROR("Failed to extend memory pool, this->size:%d, item_num_per_pool:%d, this->name:%s.",
this->size,
item_num_per_pool,
this->name.c_str());
return -1;
}
pools.push_back(pool);
this->size += item_num_per_pool;
for (int i = 0; i < item_num_per_pool; i++) {
char *item = (char *)pool + i * item_size;
frees.push_back((void *)item);
}
MUTEX_UNLOCK(&this->mutex);
LOG_INFO("Extend one pool, this->size:%d, item_size:%d, item_num_per_pool:%d, this->name:%s.",
this->size,
item_size,
item_num_per_pool,
this->name.c_str());
return 0;
}
void *MemPoolItem::alloc()
{
MUTEX_LOCK(&this->mutex);
if (frees.empty() == true) {
if (this->dynamic == false) {
MUTEX_UNLOCK(&this->mutex);
return nullptr;
}
if (extend() < 0) {
MUTEX_UNLOCK(&this->mutex);
return nullptr;
}
}
void *buffer = frees.front();
frees.pop_front();
used.insert(buffer);
MUTEX_UNLOCK(&this->mutex);
memset(buffer, 0, sizeof(item_size));
return buffer;
}
void MemPoolItem::free(void *buf)
{
MUTEX_LOCK(&this->mutex);
size_t num = used.erase(buf);
if (num == 0) {
MUTEX_UNLOCK(&this->mutex);
LOG_WARN("No entry of %p in %s.", buf, this->name.c_str());
return;
}
frees.push_back(buf);
MUTEX_UNLOCK(&this->mutex);
return; // TODO for test
}
} // namespace common
\ No newline at end of file
/* Copyright (c) 2021 Xie Meiyi(xiemeiyi@hust.edu.cn) and OceanBase and/or its affiliates. All rights reserved.
miniob is licensed under Mulan PSL v2.
You can use this software according to the terms and conditions of the Mulan PSL v2.
You may obtain a copy of Mulan PSL v2 at:
http://license.coscl.org.cn/MulanPSL2
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 PSL v2 for more details. */
//
// Created by Longda on 2010
//
#ifndef __COMMON_MM_MPOOL_H__
#define __COMMON_MM_MPOOL_H__
#include <queue>
#include <list>
#include <set>
#include <string>
#include <sstream>
#include "common/lang/mutex.h"
#include "common/log/log.h"
#include "common/os/os.h"
namespace common {
#define DEFAULT_ITEM_NUM_PER_POOL 128
#define DEFAULT_POOL_NUM 1
typedef bool (*match)(void *item, void *input_arg);
template <class T>
class MemPool {
public:
MemPool(const char *tag) : name(tag)
{
this->size = 0;
pthread_mutexattr_t mutexatr;
pthread_mutexattr_init(&mutexatr);
pthread_mutexattr_settype(&mutexatr, PTHREAD_MUTEX_RECURSIVE);
MUTEX_INIT(&mutex, &mutexatr);
}
virtual ~MemPool()
{
MUTEX_DESTROY(&mutex);
}
/**
* init memory pool, the major job is to alloc memory for memory pool
* @param pool_num, memory pool's number
* @param item_num_per_pool, how many items per pool.
* @return
*/
virtual int init(
bool dynamic = true, int pool_num = DEFAULT_POOL_NUM, int item_num_per_pool = DEFAULT_ITEM_NUM_PER_POOL) = 0;
/**
* Do cleanup job for memory pool
*/
virtual void cleanup() = 0;
/**
* If dynamic has been set, extend current memory pool,
*/
virtual int extend() = 0;
/**
* Alloc one frame from memory Pool
* @return
*/
virtual T *alloc() = 0;
/**
* Free one item, the resouce will return to memory Pool
* @param item
*/
virtual void free(T *item) = 0;
/**
* Find first used item which match the function of "func"
* @return
*/
virtual T *find(match func, void *arg) = 0;
/**
* Find used item which match the function of "func"
*/
virtual std::list<T *> find_all(match func, void *arg) = 0;
/**
* Mark the item has been changed;
*/
virtual void mark_modified(T *item) = 0;
/**
* Print the MemPool status
* @return
*/
virtual std::string to_string() = 0;
const std::string get_name() const
{
return name;
}
bool is_dynamic() const
{
return dynamic;
}
int get_size() const
{
return size;
}
protected:
pthread_mutex_t mutex;
int size;
bool dynamic;
std::string name;
};
/**
* MemoryPoolSimple is a simple Memory Pool manager,
*/
template <class T>
class MemPoolSimple : public MemPool<T> {
public:
MemPoolSimple(const char *tag) : MemPool<T>(tag)
{}
virtual ~MemPoolSimple()
{
cleanup();
}
/**
* init memory pool, the major job is to alloc memory for memory pool
* @param pool_num, memory pool's number
* @param item_num_per_pool, how many items per pool.
* @return
*/
int init(bool dynamic = true, int pool_num = DEFAULT_POOL_NUM, int item_num_per_pool = DEFAULT_ITEM_NUM_PER_POOL);
/**
* Do cleanup job for memory pool
*/
void cleanup();
/**
* If dynamic has been set, extend current memory pool,
*/
int extend();
/**
* Alloc one frame from memory Pool
* @return
*/
T *alloc();
/**
* Free one item, the resouce will return to memory Pool
* @param item
*/
void free(T *item);
/**
* Find first used item which match the function of "func"
* @return
*/
T *find(match func, void *arg);
/**
* Find used item which match the function of "func"
*/
std::list<T *> find_all(match func, void *arg);
/**
* Mark the item has been changed;
*/
void mark_modified(T *item);
/**
* Print the MemPool status
* @return
*/
std::string to_string();
int get_item_num_per_pool() const
{
return item_num_per_pool;
}
int get_used_num()
{
MUTEX_LOCK(&this->mutex);
auto num = used.size();
MUTEX_UNLOCK(&this->mutex);
return num;
}
protected:
std::list<T *> pools;
std::list<T *> lru_used;
std::set<T *> used;
std::list<T *> frees;
int item_num_per_pool;
};
template <class T>
int MemPoolSimple<T>::init(bool dynamic, int pool_num, int item_num_per_pool)
{
if (pools.empty() == false) {
LOG_WARN("Memory pool has been initialized, but still begin to be initialized, this->name:%s.", this->name.c_str());
return 0;
}
if (pool_num <= 0 || item_num_per_pool <= 0) {
LOG_ERROR("Invalid arguments, pool_num:%d, item_num_per_pool:%d, this->name:%s.",
pool_num,
item_num_per_pool,
this->name.c_str());
return -1;
}
this->item_num_per_pool = item_num_per_pool;
// in order to init memory pool, enable dynamic here
this->dynamic = true;
for (int i = 0; i < pool_num; i++) {
if (extend() < 0) {
cleanup();
return -1;
}
}
this->dynamic = dynamic;
LOG_INFO("Extend one pool, this->size:%d, item_num_per_pool:%d, this->name:%s.",
this->size,
item_num_per_pool,
this->name.c_str());
return 0;
}
template <class T>
void MemPoolSimple<T>::cleanup()
{
if (pools.empty() == true) {
LOG_WARN("Begin to do cleanup, but there is no memory pool, this->name:%s!", this->name.c_str());
return;
}
MUTEX_LOCK(&this->mutex);
lru_used.clear();
used.clear();
frees.clear();
this->size = 0;
for (typename std::list<T *>::iterator iter = pools.begin(); iter != pools.end(); iter++) {
T *pool = *iter;
delete[] pool;
}
pools.clear();
MUTEX_UNLOCK(&this->mutex);
LOG_INFO("Successfully do cleanup, this->name:%s.", this->name.c_str());
}
template <class T>
int MemPoolSimple<T>::extend()
{
if (this->dynamic == false) {
LOG_ERROR("Disable dynamic extend memory pool, but begin to extend, this->name:%s", this->name.c_str());
return -1;
}
MUTEX_LOCK(&this->mutex);
T *pool = new T[item_num_per_pool];
if (pool == nullptr) {
MUTEX_UNLOCK(&this->mutex);
LOG_ERROR("Failed to extend memory pool, this->size:%d, item_num_per_pool:%d, this->name:%s.",
this->size,
item_num_per_pool,
this->name.c_str());
return -1;
}
pools.push_back(pool);
this->size += item_num_per_pool;
for (int i = 0; i < item_num_per_pool; i++) {
frees.push_back(pool + i);
}
MUTEX_UNLOCK(&this->mutex);
LOG_INFO("Extend one pool, this->size:%d, item_num_per_pool:%d, this->name:%s.",
this->size,
item_num_per_pool,
this->name.c_str());
return 0;
}
template <class T>
T *MemPoolSimple<T>::alloc()
{
MUTEX_LOCK(&this->mutex);
if (frees.empty() == true) {
if (this->dynamic == false) {
MUTEX_UNLOCK(&this->mutex);
return nullptr;
}
if (extend() < 0) {
MUTEX_UNLOCK(&this->mutex);
return nullptr;
}
}
T *buffer = frees.front();
frees.pop_front();
used.insert(buffer);
lru_used.push_back(buffer);
MUTEX_UNLOCK(&this->mutex);
return buffer;
}
template <class T>
void MemPoolSimple<T>::free(T *buf)
{
MUTEX_LOCK(&this->mutex);
size_t num = used.erase(buf);
if (num == 0) {
MUTEX_UNLOCK(&this->mutex);
LOG_WARN("No entry of %p in %s.", buf, this->name.c_str());
print_stacktrace();
return;
}
lru_used.remove(buf);
frees.push_back(buf);
MUTEX_UNLOCK(&this->mutex);
return; // TODO for test
}
template <class T>
T *MemPoolSimple<T>::find(match func, void *arg)
{
T *buffer = nullptr;
MUTEX_LOCK(&this->mutex);
for (typename std::list<T *>::iterator it = lru_used.begin(); it != lru_used.end(); ++it) {
T *item = *it;
if ((*func)(item, arg) == false) {
continue;
}
buffer = item;
break;
}
MUTEX_UNLOCK(&this->mutex);
return buffer;
}
template <class T>
std::list<T *> MemPoolSimple<T>::find_all(match func, void *arg)
{
std::list<T *> ret;
MUTEX_LOCK(&this->mutex);
for (typename std::list<T *>::iterator it = lru_used.begin(); it != lru_used.end(); ++it) {
T *item = *it;
if ((*func)(item, arg) == false) {
continue;
}
ret.push_back(item);
}
MUTEX_UNLOCK(&this->mutex);
return ret;
}
template <class T>
void MemPoolSimple<T>::mark_modified(T *buf)
{
MUTEX_LOCK(&this->mutex);
if (used.find(buf) == used.end()) {
MUTEX_UNLOCK(&this->mutex);
LOG_WARN("No entry of %p in %s.", buf, this->name.c_str());
return;
}
lru_used.remove(buf);
lru_used.push_back(buf);
MUTEX_UNLOCK(&this->mutex);
return;
}
template <class T>
std::string MemPoolSimple<T>::to_string()
{
std::stringstream ss;
ss << "name:" << this->name << ","
<< "dyanmic:" << this->dynamic << ","
<< "size:" << this->size << ","
<< "pool_size:" << this->pools.size() << ","
<< "used_size:" << this->used.size() << ","
<< "free_size:" << this->frees.size();
return ss.str();
}
class MemPoolItem {
public:
MemPoolItem(const char *tag) : name(tag)
{
this->size = 0;
pthread_mutexattr_t mutexatr;
pthread_mutexattr_init(&mutexatr);
pthread_mutexattr_settype(&mutexatr, PTHREAD_MUTEX_RECURSIVE);
MUTEX_INIT(&mutex, &mutexatr);
}
virtual ~MemPoolItem()
{
cleanup();
MUTEX_DESTROY(&mutex);
}
/**
* init memory pool, the major job is to alloc memory for memory pool
* @param pool_num, memory pool's number
* @param item_num_per_pool, how many items per pool.
* @return
*/
int init(int item_size, bool dynamic = true, int pool_num = DEFAULT_POOL_NUM,
int item_num_per_pool = DEFAULT_ITEM_NUM_PER_POOL);
/**
* Do cleanup job for memory pool
*/
void cleanup();
/**
* If dynamic has been set, extend current memory pool,
*/
int extend();
/**
* Alloc one frame from memory Pool
* @return
*/
void *alloc();
/**
* Free one item, the resouce will return to memory Pool
* @param item
*/
void free(void *item);
/**
* Check whether this item has been used before.
* @param item
* @return
*/
bool is_used(void *item)
{
MUTEX_LOCK(&mutex);
auto it = used.find(item);
MUTEX_UNLOCK(&mutex);
return it != used.end();
}
std::string to_string()
{
std::stringstream ss;
ss << "name:" << this->name << ","
<< "dyanmic:" << this->dynamic << ","
<< "size:" << this->size << ","
<< "pool_size:" << this->pools.size() << ","
<< "used_size:" << this->used.size() << ","
<< "free_size:" << this->frees.size();
return ss.str();
}
const std::string get_name() const
{
return name;
}
bool is_dynamic() const
{
return dynamic;
}
int get_size() const
{
return size;
}
int get_item_size() const
{
return item_size;
}
int get_item_num_per_pool() const
{
return item_num_per_pool;
}
int get_used_num()
{
MUTEX_LOCK(&mutex);
auto num = used.size();
MUTEX_UNLOCK(&mutex);
return num;
}
protected:
pthread_mutex_t mutex;
std::string name;
bool dynamic;
int size;
int item_size;
int item_num_per_pool;
std::list<void *> pools;
std::set<void *> used;
std::list<void *> frees;
};
} // namespace common
#endif /* __COMMON_MM_MPOOL_H__ */
/* Copyright (c) 2021 Xie Meiyi(xiemeiyi@hust.edu.cn) and OceanBase and/or its affiliates. All rights reserved.
miniob is licensed under Mulan PSL v2.
You can use this software according to the terms and conditions of the Mulan PSL v2.
You may obtain a copy of Mulan PSL v2 at:
http://license.coscl.org.cn/MulanPSL2
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 PSL v2 for more details. */
//
// Created by Longda on 2010
//
#ifndef __COMMON_MM_MPOOL_H__
#define __COMMON_MM_MPOOL_H__
#include <queue>
#include "common/lang/mutex.h"
#include "common/log/log.h"
namespace common {
#define CLMPOOL_DEFAULT_ADDSIZE 16
template<class T>
class MemPool {
public:
MemPool() : mQueue(), mAddSize(CLMPOOL_DEFAULT_ADDSIZE) {
MUTEX_INIT(&mLock, NULL);
}
~MemPool() {
MUTEX_LOCK(&mLock);
while (mQueue.empty() == false) {
T *item = mQueue.front();
mQueue.pop();
delete item;
}
MUTEX_UNLOCK(&mLock);
MUTEX_DESTROY(&mLock);
}
int add(int addSize) {
int ret = 0;
MUTEX_LOCK(&mLock);
for (int i = 0; i < addSize; i++) {
T *item = new T();
if (item == NULL) {
ret = -1;
break;
}
mQueue.push(item);
}
MUTEX_UNLOCK(&mLock);
return ret;
}
int init(int initSize) { return add(initSize); }
T *get() {
T *ret = NULL;
MUTEX_LOCK(&mLock);
if (mQueue.empty() == true) {
add(mAddSize);
}
if (mQueue.empty() == false) {
ret = mQueue.front();
mQueue.pop();
}
MUTEX_UNLOCK(&mLock);
return ret;
}
void put(T *item) {
MUTEX_LOCK(&mLock);
mQueue.push(item);
MUTEX_UNLOCK(&mLock);
}
private:
std::queue<T *> mQueue;
pthread_mutex_t mLock;
int mAddSize;
};
} //namespace common
#endif /* __COMMON_MM_MPOOL_H__ */
...@@ -13,9 +13,11 @@ See the Mulan PSL v2 for more details. */ ...@@ -13,9 +13,11 @@ See the Mulan PSL v2 for more details. */
// //
#include <thread> #include <thread>
#include <execinfo.h>
#include "common/defs.h" #include "common/defs.h"
#include "common/os/os.h" #include "common/os/os.h"
#include "common/log/log.h"
namespace common { namespace common {
// Don't care windows // Don't care windows
...@@ -23,4 +25,19 @@ u32_t getCpuNum() { ...@@ -23,4 +25,19 @@ u32_t getCpuNum() {
return std::thread::hardware_concurrency(); return std::thread::hardware_concurrency();
} }
#define MAX_STACK_SIZE 32
void print_stacktrace()
{
int size = MAX_STACK_SIZE;
void * array[MAX_STACK_SIZE];
int stack_num = backtrace(array, size);
char ** stacktrace = backtrace_symbols(array, stack_num);
for (int i = 0; i < stack_num; ++i)
{
LOG_INFO("%d ----- %s\n", i, stacktrace[i]);
}
free(stacktrace);
}
}//namespace common }//namespace common
\ No newline at end of file
...@@ -18,5 +18,7 @@ namespace common { ...@@ -18,5 +18,7 @@ namespace common {
u32_t getCpuNum(); u32_t getCpuNum();
void print_stacktrace();
} //namespace common } //namespace common
#endif /* __COMMON_OS_OS_H__ */ #endif /* __COMMON_OS_OS_H__ */
...@@ -23,7 +23,8 @@ See the Mulan PSL v2 for more details. */ ...@@ -23,7 +23,8 @@ See the Mulan PSL v2 for more details. */
#include "common/log/log.h" #include "common/log/log.h"
namespace common { namespace common {
std::string getFileName(const std::string &fullPath) { std::string getFileName(const std::string &fullPath)
{
std::string szRt; std::string szRt;
size_t pos; size_t pos;
try { try {
...@@ -36,12 +37,12 @@ std::string getFileName(const std::string &fullPath) { ...@@ -36,12 +37,12 @@ std::string getFileName(const std::string &fullPath) {
szRt = ""; szRt = "";
} }
} catch (...) { } catch (...) {}
}
return szRt; return szRt;
} }
void getFileName(const char *path, std::string &fileName) { void getFileName(const char *path, std::string &fileName)
{
// Don't care the last character as FILE_PATH_SPLIT // Don't care the last character as FILE_PATH_SPLIT
const char *endPos = strrchr(path, FILE_PATH_SPLIT); const char *endPos = strrchr(path, FILE_PATH_SPLIT);
if (endPos == NULL) { if (endPos == NULL) {
...@@ -58,7 +59,8 @@ void getFileName(const char *path, std::string &fileName) { ...@@ -58,7 +59,8 @@ void getFileName(const char *path, std::string &fileName) {
return; return;
} }
std::string getDirName(const std::string &fullPath) { std::string getDirName(const std::string &fullPath)
{
std::string szRt; std::string szRt;
size_t pos; size_t pos;
try { try {
...@@ -72,11 +74,11 @@ std::string getDirName(const std::string &fullPath) { ...@@ -72,11 +74,11 @@ std::string getDirName(const std::string &fullPath) {
szRt = FILE_PATH_SPLIT_STR; szRt = FILE_PATH_SPLIT_STR;
} }
} catch (...) { } catch (...) {}
}
return szRt; return szRt;
} }
void getDirName(const char *path, std::string &parent) { void getDirName(const char *path, std::string &parent)
{
// Don't care the last character as FILE_PATH_SPLIT // Don't care the last character as FILE_PATH_SPLIT
const char *endPos = strrchr(path, FILE_PATH_SPLIT); const char *endPos = strrchr(path, FILE_PATH_SPLIT);
if (endPos == NULL) { if (endPos == NULL) {
...@@ -93,7 +95,8 @@ void getDirName(const char *path, std::string &parent) { ...@@ -93,7 +95,8 @@ void getDirName(const char *path, std::string &parent) {
return; return;
} }
std::string getFilePath(const std::string &fullPath) { std::string getFilePath(const std::string &fullPath)
{
std::string szRt; std::string szRt;
size_t pos; size_t pos;
try { try {
...@@ -106,43 +109,42 @@ std::string getFilePath(const std::string &fullPath) { ...@@ -106,43 +109,42 @@ std::string getFilePath(const std::string &fullPath) {
szRt = ""; szRt = "";
} }
} catch (...) { } catch (...) {}
}
return szRt; return szRt;
} }
std::string getAboslutPath(const char *path) { std::string getAboslutPath(const char *path)
{
std::string aPath(path); std::string aPath(path);
if (path[0] != '/') { if (path[0] != '/') {
const int MAX_SIZE = 256; const int MAX_SIZE = 256;
char current_absolute_path[MAX_SIZE]; char current_absolute_path[MAX_SIZE];
if (NULL == getcwd(current_absolute_path, MAX_SIZE)) { if (NULL == getcwd(current_absolute_path, MAX_SIZE)) {}
}
} }
return aPath; return aPath;
} }
bool is_directory(const char *path) { bool is_directory(const char *path)
{
struct stat st; struct stat st;
return (0 == stat(path, &st)) && (st.st_mode & S_IFDIR); return (0 == stat(path, &st)) && (st.st_mode & S_IFDIR);
} }
bool check_directory(std::string &path) { bool check_directory(std::string &path)
{
while (!path.empty() && path.back() == '/') while (!path.empty() && path.back() == '/')
path.erase(path.size() - 1, 1); path.erase(path.size() - 1, 1);
int len = path.size(); int len = path.size();
if (0 == mkdir(path.c_str(), 0777) || is_directory(path.c_str()) ) if (0 == mkdir(path.c_str(), 0777) || is_directory(path.c_str()))
return true; return true;
bool sep_state = false; bool sep_state = false;
for (int i = 0; i < len; i++) for (int i = 0; i < len; i++) {
{ if (path[i] != '/') {
if (path[i] != '/')
{
if (sep_state) if (sep_state)
sep_state = false; sep_state = false;
continue; continue;
...@@ -164,39 +166,33 @@ bool check_directory(std::string &path) { ...@@ -164,39 +166,33 @@ bool check_directory(std::string &path) {
return true; return true;
} }
int list_file(const char *path, const char *filter_pattern, std::vector<std::string> &files) { int list_file(const char *path, const char *filter_pattern, std::vector<std::string> &files)
{
regex_t reg; regex_t reg;
if (filter_pattern) if (filter_pattern) {
{
const int res = regcomp(&reg, filter_pattern, REG_NOSUB); const int res = regcomp(&reg, filter_pattern, REG_NOSUB);
if (res) if (res) {
{
char errbuf[256]; char errbuf[256];
regerror(res, &reg, errbuf, sizeof(errbuf)); regerror(res, &reg, errbuf, sizeof(errbuf));
LOG_ERROR("regcomp return error. filter pattern %s. errmsg %d:%s", LOG_ERROR("regcomp return error. filter pattern %s. errmsg %d:%s", filter_pattern, res, errbuf);
filter_pattern, res, errbuf);
return -1; return -1;
} }
} }
DIR *pdir = opendir(path); DIR *pdir = opendir(path);
if (!pdir) if (!pdir) {
{
if (filter_pattern) if (filter_pattern)
regfree(&reg); regfree(&reg);
LOG_ERROR("open directory failure. path %s, errmsg %d:%s", LOG_ERROR("open directory failure. path %s, errmsg %d:%s", path, errno, strerror(errno));
path, errno, strerror(errno));
closedir(pdir);
return -1; return -1;
} }
files.clear(); files.clear();
struct dirent entry; struct dirent entry;
struct dirent * pentry = NULL; struct dirent *pentry = NULL;
char tmp_path[PATH_MAX]; char tmp_path[PATH_MAX];
while((0 == readdir_r(pdir, &entry, &pentry)) && (NULL != pentry)) while ((0 == readdir_r(pdir, &entry, &pentry)) && (NULL != pentry)) {
{
if ('.' == entry.d_name[0]) // 跳过./..文件和隐藏文件 if ('.' == entry.d_name[0]) // 跳过./..文件和隐藏文件
continue; continue;
...@@ -215,4 +211,4 @@ int list_file(const char *path, const char *filter_pattern, std::vector<std::str ...@@ -215,4 +211,4 @@ int list_file(const char *path, const char *filter_pattern, std::vector<std::str
return files.size(); return files.size();
} }
} //namespace common } // namespace common
\ No newline at end of file \ No newline at end of file
...@@ -15,64 +15,97 @@ See the Mulan PSL v2 for more details. */ ...@@ -15,64 +15,97 @@ See the Mulan PSL v2 for more details. */
#include "storage/common/bplus_tree_index.h" #include "storage/common/bplus_tree_index.h"
#include "common/log/log.h" #include "common/log/log.h"
BplusTreeIndex::~BplusTreeIndex() noexcept { BplusTreeIndex::~BplusTreeIndex() noexcept
{
close(); close();
} }
RC BplusTreeIndex::create(const char *file_name, const IndexMeta &index_meta, const FieldMeta &field_meta) { RC BplusTreeIndex::create(const char *file_name, const IndexMeta &index_meta, const FieldMeta &field_meta)
{
if (inited_) { if (inited_) {
LOG_WARN("Failed to create index due to the index has been created before. file_name:%s, index:%s, field:%s",
file_name,
index_meta.name(),
index_meta.field());
return RC::RECORD_OPENNED; return RC::RECORD_OPENNED;
} }
RC rc = Index::init(index_meta, field_meta); Index::init(index_meta, field_meta);
if (rc != RC::SUCCESS) {
RC rc = index_handler_.create(file_name, field_meta.type(), field_meta.len());
if (RC::SUCCESS != rc) {
LOG_WARN("Failed to create index_handler, file_name:%s, index:%s, field:%s, rc:%s",
file_name,
index_meta.name(),
index_meta.field(),
strrc(rc));
return rc; return rc;
} }
rc = index_handler_.create(file_name, field_meta.type(), field_meta.len());
if (RC::SUCCESS == rc) {
inited_ = true; inited_ = true;
} LOG_INFO(
return rc; "Successfully create index, file_name:%s, index:%s, field:%s", file_name, index_meta.name(), index_meta.field());
return RC::SUCCESS;
} }
RC BplusTreeIndex::open(const char *file_name, const IndexMeta &index_meta, const FieldMeta &field_meta) { RC BplusTreeIndex::open(const char *file_name, const IndexMeta &index_meta, const FieldMeta &field_meta)
{
if (inited_) { if (inited_) {
LOG_WARN("Failed to open index due to the index has been initedd before. file_name:%s, index:%s, field:%s",
file_name,
index_meta.name(),
index_meta.field());
return RC::RECORD_OPENNED; return RC::RECORD_OPENNED;
} }
RC rc = Index::init(index_meta, field_meta);
if (rc != RC::SUCCESS) { Index::init(index_meta, field_meta);
RC rc = index_handler_.open(file_name);
if (RC::SUCCESS != rc) {
LOG_WARN("Failed to open index_handler, file_name:%s, index:%s, field:%s, rc:%s",
file_name,
index_meta.name(),
index_meta.field(),
strrc(rc));
return rc; return rc;
} }
rc = index_handler_.open(file_name);
if (RC::SUCCESS == rc) {
inited_ = true; inited_ = true;
} LOG_INFO(
return rc; "Successfully open index, file_name:%s, index:%s, field:%s", file_name, index_meta.name(), index_meta.field());
return RC::SUCCESS;
} }
RC BplusTreeIndex::close() { RC BplusTreeIndex::close()
{
if (inited_) { if (inited_) {
LOG_INFO("Begin to close index, file_id:%d, index:%s, field:%s",
index_handler_.get_file_id(),
index_meta_.name(),
index_meta_.field());
index_handler_.close(); index_handler_.close();
inited_ = false; inited_ = false;
} }
LOG_INFO("Successfully close index.");
return RC::SUCCESS; return RC::SUCCESS;
} }
RC BplusTreeIndex::insert_entry(const char *record, const RID *rid) { RC BplusTreeIndex::insert_entry(const char *record, const RID *rid)
{
return index_handler_.insert_entry(record + field_meta_.offset(), rid); return index_handler_.insert_entry(record + field_meta_.offset(), rid);
} }
RC BplusTreeIndex::delete_entry(const char *record, const RID *rid) { RC BplusTreeIndex::delete_entry(const char *record, const RID *rid)
{
return index_handler_.delete_entry(record + field_meta_.offset(), rid); return index_handler_.delete_entry(record + field_meta_.offset(), rid);
} }
IndexScanner *BplusTreeIndex::create_scanner(CompOp comp_op, const char *value) { IndexScanner *BplusTreeIndex::create_scanner(CompOp comp_op, const char *value)
{
BplusTreeScanner *bplus_tree_scanner = new BplusTreeScanner(index_handler_); BplusTreeScanner *bplus_tree_scanner = new BplusTreeScanner(index_handler_);
RC rc = bplus_tree_scanner->open(comp_op, value); RC rc = bplus_tree_scanner->open(comp_op, value);
if (rc != RC::SUCCESS) { if (rc != RC::SUCCESS) {
LOG_ERROR("Failed to open index scanner. rc=%d:%s", rc, strrc(rc)); LOG_WARN("Failed to open index scanner. file_id:%d, rc=%d:%s", index_handler_.get_file_id(), rc, strrc(rc));
delete bplus_tree_scanner; delete bplus_tree_scanner;
return nullptr; return nullptr;
} }
...@@ -81,25 +114,28 @@ IndexScanner *BplusTreeIndex::create_scanner(CompOp comp_op, const char *value) ...@@ -81,25 +114,28 @@ IndexScanner *BplusTreeIndex::create_scanner(CompOp comp_op, const char *value)
return index_scanner; return index_scanner;
} }
RC BplusTreeIndex::sync() { RC BplusTreeIndex::sync()
{
return index_handler_.sync(); return index_handler_.sync();
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
BplusTreeIndexScanner::BplusTreeIndexScanner(BplusTreeScanner *tree_scanner) : BplusTreeIndexScanner::BplusTreeIndexScanner(BplusTreeScanner *tree_scanner) : tree_scanner_(tree_scanner)
tree_scanner_(tree_scanner) { {}
}
BplusTreeIndexScanner::~BplusTreeIndexScanner() noexcept { BplusTreeIndexScanner::~BplusTreeIndexScanner() noexcept
{
tree_scanner_->close(); tree_scanner_->close();
delete tree_scanner_; delete tree_scanner_;
} }
RC BplusTreeIndexScanner::next_entry(RID *rid) { RC BplusTreeIndexScanner::next_entry(RID *rid)
{
return tree_scanner_->next_entry(rid); return tree_scanner_->next_entry(rid);
} }
RC BplusTreeIndexScanner::destroy() { RC BplusTreeIndexScanner::destroy()
{
delete this; delete this;
return RC::SUCCESS; return RC::SUCCESS;
} }
\ No newline at end of file
此差异已折叠。
...@@ -24,6 +24,7 @@ namespace Json { ...@@ -24,6 +24,7 @@ namespace Json {
class Value; class Value;
} // namespace Json } // namespace Json
// Take care of shallow copy
class FieldMeta { class FieldMeta {
public: public:
FieldMeta(); FieldMeta();
...@@ -40,11 +41,12 @@ public: ...@@ -40,11 +41,12 @@ public:
public: public:
void desc(std::ostream &os) const; void desc(std::ostream &os) const;
public: public:
void to_json(Json::Value &json_value) const; void to_json(Json::Value &json_value) const;
static RC from_json(const Json::Value &json_value, FieldMeta &field); static RC from_json(const Json::Value &json_value, FieldMeta &field);
private: protected:
std::string name_; std::string name_;
AttrType attr_type_; AttrType attr_type_;
int attr_offset_; int attr_offset_;
......
此差异已折叠。
此差异已折叠。
...@@ -36,12 +36,13 @@ public: ...@@ -36,12 +36,13 @@ public:
const char *field() const; const char *field() const;
void desc(std::ostream &os) const; void desc(std::ostream &os) const;
public: public:
void to_json(Json::Value &json_value) const; void to_json(Json::Value &json_value) const;
static RC from_json(const TableMeta &table, const Json::Value &json_value, IndexMeta &index); static RC from_json(const TableMeta &table, const Json::Value &json_value, IndexMeta &index);
private: protected:
std::string name_; std::string name_; // index's name
std::string field_; std::string field_; // field's name
}; };
#endif // __OBSERVER_STORAGE_COMMON_INDEX_META_H__ #endif // __OBSERVER_STORAGE_COMMON_INDEX_META_H__
\ No newline at end of file
...@@ -22,6 +22,7 @@ static const char *TABLE_DATA_SUFFIX = ".data"; ...@@ -22,6 +22,7 @@ static const char *TABLE_DATA_SUFFIX = ".data";
static const char *TABLE_INDEX_SUFFIX = ".index"; static const char *TABLE_INDEX_SUFFIX = ".index";
std::string table_meta_file(const char *base_dir, const char *table_name); std::string table_meta_file(const char *base_dir, const char *table_name);
std::string index_data_file(const char *base_dir, const char *table_name, const char *index_name); std::string table_data_file(const char *base_dir, const char *table_name);
std::string table_index_file(const char *base_dir, const char *table_name, const char *index_name);
#endif //__OBSERVER_STORAGE_COMMON_META_UTIL_H_ #endif //__OBSERVER_STORAGE_COMMON_META_UTIL_H_
此差异已折叠。
此差异已折叠。
此差异已折叠。
此差异已折叠。
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册