Add enable_thread_tracking to DBOptions

Summary:
Add enable_thread_tracking to DBOptions to allow
tracking thread status related to the DB.  Default is off.

Test Plan:
export ROCKSDB_TESTS=ThreadList
./db_test

Reviewers: ljin, sdong, igor

Reviewed By: igor

Subscribers: dhruba, leveldb

Differential Revision: https://reviews.facebook.net/D29289
This commit is contained in:
Yueh-Hsuan Chiang 2014-11-20 21:13:18 -08:00
parent 9e285d4238
commit 4b63fcbff3
6 changed files with 51 additions and 15 deletions

View File

@ -3725,18 +3725,24 @@ Status DestroyDB(const std::string& dbname, const Options& options) {
#if ROCKSDB_USING_THREAD_STATUS #if ROCKSDB_USING_THREAD_STATUS
void DBImpl::NewThreadStatusCfInfo( void DBImpl::NewThreadStatusCfInfo(
ColumnFamilyData* cfd) const { ColumnFamilyData* cfd) const {
if (db_options_.enable_thread_tracking) {
ThreadStatusImpl::NewColumnFamilyInfo( ThreadStatusImpl::NewColumnFamilyInfo(
this, GetName(), cfd, cfd->GetName()); this, GetName(), cfd, cfd->GetName());
} }
}
void DBImpl::EraseThreadStatusCfInfo( void DBImpl::EraseThreadStatusCfInfo(
ColumnFamilyData* cfd) const { ColumnFamilyData* cfd) const {
if (db_options_.enable_thread_tracking) {
ThreadStatusImpl::EraseColumnFamilyInfo(cfd); ThreadStatusImpl::EraseColumnFamilyInfo(cfd);
} }
}
void DBImpl::EraseThreadStatusDbInfo() const { void DBImpl::EraseThreadStatusDbInfo() const {
if (db_options_.enable_thread_tracking) {
ThreadStatusImpl::EraseDatabaseInfo(this); ThreadStatusImpl::EraseDatabaseInfo(this);
} }
}
Status GetThreadList(std::vector<ThreadStatus>* thread_list) { Status GetThreadList(std::vector<ThreadStatus>* thread_list) {
return thread_local_status.GetThreadList(thread_list); return thread_local_status.GetThreadList(thread_list);

View File

@ -8986,6 +8986,8 @@ TEST(DBTest, DynamicMemtableOptions) {
TEST(DBTest, GetThreadList) { TEST(DBTest, GetThreadList) {
Options options; Options options;
options.env = env_; options.env = env_;
options.enable_thread_tracking = true;
TryReopen(options);
std::vector<ThreadStatus> thread_list; std::vector<ThreadStatus> thread_list;
Status s = GetThreadList(&thread_list); Status s = GetThreadList(&thread_list);
@ -9025,14 +9027,24 @@ TEST(DBTest, GetThreadList) {
if (i == 0) { if (i == 0) {
// repeat the test with multiple column families // repeat the test with multiple column families
CreateAndReopenWithCF({"pikachu", "about-to-remove"}, options); CreateAndReopenWithCF({"pikachu", "about-to-remove"}, options);
ThreadStatusImpl::TEST_VerifyColumnFamilyInfoMap(handles_); ThreadStatusImpl::TEST_VerifyColumnFamilyInfoMap(handles_, true);
} }
} }
db_->DropColumnFamily(handles_[2]); db_->DropColumnFamily(handles_[2]);
handles_.erase(handles_.begin() + 2); handles_.erase(handles_.begin() + 2);
ThreadStatusImpl::TEST_VerifyColumnFamilyInfoMap(handles_); ThreadStatusImpl::TEST_VerifyColumnFamilyInfoMap(handles_, true);
Close(); Close();
ThreadStatusImpl::TEST_VerifyColumnFamilyInfoMap(handles_); ThreadStatusImpl::TEST_VerifyColumnFamilyInfoMap(handles_, true);
}
TEST(DBTest, DisableThreadList) {
Options options;
options.env = env_;
options.enable_thread_tracking = false;
TryReopen(options);
CreateAndReopenWithCF({"pikachu", "about-to-remove"}, options);
// Verify non of the column family info exists
ThreadStatusImpl::TEST_VerifyColumnFamilyInfoMap(handles_, false);
} }
#endif // ROCKSDB_USING_THREAD_STATUS #endif // ROCKSDB_USING_THREAD_STATUS

View File

@ -884,6 +884,12 @@ struct DBOptions {
// When rate limiter is enabled, it automatically enables bytes_per_sync // When rate limiter is enabled, it automatically enables bytes_per_sync
// to 1MB. // to 1MB.
uint64_t bytes_per_sync; uint64_t bytes_per_sync;
// If true, then the status of the threads involved in this DB will
// be tracked and available via GetThreadList() API.
//
// Default: false
bool enable_thread_tracking;
}; };
// Options to control the behavior of a database (passed to DB::Open) // Options to control the behavior of a database (passed to DB::Open)

View File

@ -232,7 +232,8 @@ DBOptions::DBOptions()
advise_random_on_open(true), advise_random_on_open(true),
access_hint_on_compaction_start(NORMAL), access_hint_on_compaction_start(NORMAL),
use_adaptive_mutex(false), use_adaptive_mutex(false),
bytes_per_sync(0) {} bytes_per_sync(0),
enable_thread_tracking(false) {}
DBOptions::DBOptions(const Options& options) DBOptions::DBOptions(const Options& options)
: create_if_missing(options.create_if_missing), : create_if_missing(options.create_if_missing),
@ -274,7 +275,8 @@ DBOptions::DBOptions(const Options& options)
advise_random_on_open(options.advise_random_on_open), advise_random_on_open(options.advise_random_on_open),
access_hint_on_compaction_start(options.access_hint_on_compaction_start), access_hint_on_compaction_start(options.access_hint_on_compaction_start),
use_adaptive_mutex(options.use_adaptive_mutex), use_adaptive_mutex(options.use_adaptive_mutex),
bytes_per_sync(options.bytes_per_sync) {} bytes_per_sync(options.bytes_per_sync),
enable_thread_tracking(options.enable_thread_tracking) {}
static const char* const access_hints[] = { static const char* const access_hints[] = {
"NONE", "NORMAL", "SEQUENTIAL", "WILLNEED" "NONE", "NORMAL", "SEQUENTIAL", "WILLNEED"
@ -342,6 +344,8 @@ void DBOptions::Dump(Logger* log) const {
rate_limiter.get()); rate_limiter.get());
Log(log, " Options.bytes_per_sync: %" PRIu64, Log(log, " Options.bytes_per_sync: %" PRIu64,
bytes_per_sync); bytes_per_sync);
Log(log, " enable_thread_tracking: %d",
enable_thread_tracking);
} // DBOptions::Dump } // DBOptions::Dump
void ColumnFamilyOptions::Dump(Logger* log) const { void ColumnFamilyOptions::Dump(Logger* log) const {

View File

@ -127,7 +127,8 @@ class ThreadStatusImpl {
// Verifies whether the input ColumnFamilyHandles matches // Verifies whether the input ColumnFamilyHandles matches
// the information stored in the current cf_info_map. // the information stored in the current cf_info_map.
static void TEST_VerifyColumnFamilyInfoMap( static void TEST_VerifyColumnFamilyInfoMap(
const std::vector<ColumnFamilyHandle*>& handles); const std::vector<ColumnFamilyHandle*>& handles,
bool check_exist);
protected: protected:
// The thread-local variable for storing thread status. // The thread-local variable for storing thread status.

View File

@ -11,15 +11,22 @@
namespace rocksdb { namespace rocksdb {
void ThreadStatusImpl::TEST_VerifyColumnFamilyInfoMap( void ThreadStatusImpl::TEST_VerifyColumnFamilyInfoMap(
const std::vector<ColumnFamilyHandle*>& handles) { const std::vector<ColumnFamilyHandle*>& handles,
bool check_exist) {
std::unique_lock<std::mutex> lock(thread_list_mutex_); std::unique_lock<std::mutex> lock(thread_list_mutex_);
if (check_exist) {
assert(cf_info_map_.size() == handles.size()); assert(cf_info_map_.size() == handles.size());
}
for (auto* handle : handles) { for (auto* handle : handles) {
auto* cfd = reinterpret_cast<ColumnFamilyHandleImpl*>(handle)->cfd(); auto* cfd = reinterpret_cast<ColumnFamilyHandleImpl*>(handle)->cfd();
auto iter __attribute__((unused)) = cf_info_map_.find(cfd); auto iter __attribute__((unused)) = cf_info_map_.find(cfd);
if (check_exist) {
assert(iter != cf_info_map_.end()); assert(iter != cf_info_map_.end());
assert(iter->second); assert(iter->second);
assert(iter->second->cf_name == cfd->GetName()); assert(iter->second->cf_name == cfd->GetName());
} else {
assert(iter == cf_info_map_.end());
}
} }
} }
} // namespace rocksdb } // namespace rocksdb