248c063ba1
Summary: Report elapsed time of a thread operation in micros in ThreadStatus instead of start time of a thread operation in seconds since the Epoch, 1970-01-01 00:00:00 (UTC). Test Plan: ./db_bench --benchmarks=fillrandom --num=100000 --threads=40 \ --max_background_compactions=10 --max_background_flushes=3 \ --thread_status_per_interval=1000 --key_size=16 --value_size=1000 \ --num_column_families=10 Sample Output: ThreadID ThreadType cfName Operation ElapsedTime Stage State 140667724562496 High Pri column_family_name_000002 Flush 772.419 ms FlushJob::WriteLevel0Table 140667728756800 High Pri default Flush 617.845 ms FlushJob::WriteLevel0Table 140667732951104 High Pri column_family_name_000005 Flush 772.078 ms FlushJob::WriteLevel0Table 140667875557440 Low Pri column_family_name_000008 Compaction 1409.216 ms CompactionJob::Install 140667737145408 Low Pri 140667749728320 Low Pri 140667816837184 Low Pri column_family_name_000007 Compaction 1071.815 ms CompactionJob::ProcessKeyValueCompaction 140667787477056 Low Pri column_family_name_000009 Compaction 772.516 ms CompactionJob::ProcessKeyValueCompaction 140667741339712 Low Pri 140667758116928 Low Pri column_family_name_000004 Compaction 620.739 ms CompactionJob::ProcessKeyValueCompaction 140667753922624 Low Pri 140667842003008 Low Pri column_family_name_000006 Compaction 1260.079 ms CompactionJob::ProcessKeyValueCompaction 140667745534016 Low Pri Reviewers: sdong, igor, rven Subscribers: dhruba, leveldb Differential Revision: https://reviews.facebook.net/D35769
207 lines
7.4 KiB
C++
207 lines
7.4 KiB
C++
// Copyright (c) 2013, Facebook, Inc. All rights reserved.
|
|
// This source code is licensed under the BSD-style license found in the
|
|
// LICENSE file in the root directory of this source tree. An additional grant
|
|
// of patent rights can be found in the PATENTS file in the same directory.
|
|
//
|
|
// The implementation of ThreadStatus.
|
|
//
|
|
// Note that we make get and set access to ThreadStatusData lockless.
|
|
// As a result, ThreadStatusData as a whole is not atomic. However,
|
|
// we guarantee consistent ThreadStatusData all the time whenever
|
|
// user call GetThreadList(). This consistency guarantee is done
|
|
// by having the following constraint in the internal implementation
|
|
// of set and get order:
|
|
//
|
|
// 1. When reset any information in ThreadStatusData, always start from
|
|
// clearing up the lower-level information first.
|
|
// 2. When setting any information in ThreadStatusData, always start from
|
|
// setting the higher-level information.
|
|
// 3. When returning ThreadStatusData to the user, fields are fetched from
|
|
// higher-level to lower-level. In addition, where there's a nullptr
|
|
// in one field, then all fields that has lower-level than that field
|
|
// should be ignored.
|
|
//
|
|
// The high to low level information would be:
|
|
// thread_id > thread_type > db > cf > operation > state
|
|
//
|
|
// This means user might not always get full information, but whenever
|
|
// returned by the GetThreadList() is guaranteed to be consistent.
|
|
#pragma once
|
|
#include <atomic>
|
|
#include <list>
|
|
#include <memory>
|
|
#include <mutex>
|
|
#include <string>
|
|
#include <unordered_map>
|
|
#include <unordered_set>
|
|
#include <vector>
|
|
|
|
#include "rocksdb/status.h"
|
|
#include "rocksdb/thread_status.h"
|
|
#include "port/port_posix.h"
|
|
#include "util/thread_operation.h"
|
|
|
|
namespace rocksdb {
|
|
|
|
class ColumnFamilyHandle;
|
|
|
|
// The structure that keeps constant information about a column family.
|
|
struct ConstantColumnFamilyInfo {
|
|
#if ROCKSDB_USING_THREAD_STATUS
|
|
public:
|
|
ConstantColumnFamilyInfo(
|
|
const void* _db_key,
|
|
const std::string& _db_name,
|
|
const std::string& _cf_name) :
|
|
db_key(_db_key), db_name(_db_name), cf_name(_cf_name) {}
|
|
const void* db_key;
|
|
const std::string db_name;
|
|
const std::string cf_name;
|
|
#endif // ROCKSDB_USING_THREAD_STATUS
|
|
};
|
|
|
|
// the internal data-structure that is used to reflect the current
|
|
// status of a thread using a set of atomic pointers.
|
|
struct ThreadStatusData {
|
|
#if ROCKSDB_USING_THREAD_STATUS
|
|
explicit ThreadStatusData() : thread_id(0), enable_tracking(false) {
|
|
thread_type.store(ThreadStatus::USER);
|
|
cf_key.store(nullptr);
|
|
operation_type.store(ThreadStatus::OP_UNKNOWN);
|
|
op_start_time.store(0);
|
|
state_type.store(ThreadStatus::STATE_UNKNOWN);
|
|
}
|
|
|
|
uint64_t thread_id;
|
|
|
|
// A flag to indicate whether the thread tracking is enabled
|
|
// in the current thread. This value will be updated based on whether
|
|
// the associated Options::enable_thread_tracking is set to true
|
|
// in ThreadStatusUtil::SetColumnFamily().
|
|
//
|
|
// If set to false, then SetThreadOperation and SetThreadState
|
|
// will be no-op.
|
|
bool enable_tracking;
|
|
|
|
std::atomic<ThreadStatus::ThreadType> thread_type;
|
|
std::atomic<const void*> cf_key;
|
|
std::atomic<ThreadStatus::OperationType> operation_type;
|
|
std::atomic<uint64_t> op_start_time;
|
|
std::atomic<ThreadStatus::OperationStage> operation_stage;
|
|
std::atomic<ThreadStatus::StateType> state_type;
|
|
#endif // ROCKSDB_USING_THREAD_STATUS
|
|
};
|
|
|
|
// The class that stores and updates the status of the current thread
|
|
// using a thread-local ThreadStatusData.
|
|
//
|
|
// In most of the case, you should use ThreadStatusUtil to update
|
|
// the status of the current thread instead of using ThreadSatusUpdater
|
|
// directly.
|
|
//
|
|
// @see ThreadStatusUtil
|
|
class ThreadStatusUpdater {
|
|
public:
|
|
ThreadStatusUpdater() {}
|
|
|
|
// Releases all ThreadStatusData of all active threads.
|
|
virtual ~ThreadStatusUpdater() {}
|
|
|
|
// Unregister the current thread.
|
|
void UnregisterThread();
|
|
|
|
// Reset the status of the current thread. This includes resetting
|
|
// ColumnFamilyInfoKey, ThreadOperation, and ThreadState.
|
|
void ResetThreadStatus();
|
|
|
|
// Set the thread type of the current thread.
|
|
void SetThreadType(ThreadStatus::ThreadType ttype);
|
|
|
|
// Update the column-family info of the current thread by setting
|
|
// its thread-local pointer of ThreadStateInfo to the correct entry.
|
|
void SetColumnFamilyInfoKey(const void* cf_key);
|
|
|
|
// returns the column family info key.
|
|
const void* GetColumnFamilyInfoKey();
|
|
|
|
// Update the thread operation of the current thread.
|
|
void SetThreadOperation(const ThreadStatus::OperationType type);
|
|
|
|
// The start time of the current thread operation. It is in the format
|
|
// of micro-seconds since some fixed point in time.
|
|
void SetOperationStartTime(const uint64_t start_time);
|
|
|
|
// Update the thread operation stage of the current thread.
|
|
ThreadStatus::OperationStage SetThreadOperationStage(
|
|
const ThreadStatus::OperationStage stage);
|
|
|
|
// Clear thread operation of the current thread.
|
|
void ClearThreadOperation();
|
|
|
|
// Update the thread state of the current thread.
|
|
void SetThreadState(const ThreadStatus::StateType type);
|
|
|
|
// Clear the thread state of the current thread.
|
|
void ClearThreadState();
|
|
|
|
// Obtain the status of all active registered threads.
|
|
Status GetThreadList(
|
|
std::vector<ThreadStatus>* thread_list);
|
|
|
|
// Create an entry in the global ColumnFamilyInfo table for the
|
|
// specified column family. This function should be called only
|
|
// when the current thread does not hold db_mutex.
|
|
void NewColumnFamilyInfo(
|
|
const void* db_key, const std::string& db_name,
|
|
const void* cf_key, const std::string& cf_name);
|
|
|
|
// Erase all ConstantColumnFamilyInfo that is associated with the
|
|
// specified db instance. This function should be called only when
|
|
// the current thread does not hold db_mutex.
|
|
void EraseDatabaseInfo(const void* db_key);
|
|
|
|
// Erase the ConstantColumnFamilyInfo that is associated with the
|
|
// specified ColumnFamilyData. This function should be called only
|
|
// when the current thread does not hold db_mutex.
|
|
void EraseColumnFamilyInfo(const void* cf_key);
|
|
|
|
// Verifies whether the input ColumnFamilyHandles matches
|
|
// the information stored in the current cf_info_map.
|
|
void TEST_VerifyColumnFamilyInfoMap(
|
|
const std::vector<ColumnFamilyHandle*>& handles,
|
|
bool check_exist);
|
|
|
|
protected:
|
|
#if ROCKSDB_USING_THREAD_STATUS
|
|
// The thread-local variable for storing thread status.
|
|
static __thread ThreadStatusData* thread_status_data_;
|
|
|
|
// Obtain the pointer to the thread status data. It also performs
|
|
// initialization when necessary.
|
|
ThreadStatusData* InitAndGet();
|
|
|
|
// The mutex that protects cf_info_map and db_key_map.
|
|
std::mutex thread_list_mutex_;
|
|
|
|
// The current status data of all active threads.
|
|
std::unordered_set<ThreadStatusData*> thread_data_set_;
|
|
|
|
// A global map that keeps the column family information. It is stored
|
|
// globally instead of inside DB is to avoid the situation where DB is
|
|
// closing while GetThreadList function already get the pointer to its
|
|
// CopnstantColumnFamilyInfo.
|
|
std::unordered_map<
|
|
const void*, std::unique_ptr<ConstantColumnFamilyInfo>> cf_info_map_;
|
|
|
|
// A db_key to cf_key map that allows erasing elements in cf_info_map
|
|
// associated to the same db_key faster.
|
|
std::unordered_map<
|
|
const void*, std::unordered_set<const void*>> db_key_map_;
|
|
|
|
#else
|
|
static ThreadStatusData* thread_status_data_;
|
|
#endif // ROCKSDB_USING_THREAD_STATUS
|
|
};
|
|
|
|
} // namespace rocksdb
|