Disable stats about mutex duration by default

Summary: Measuring mutex duration will measure time inside DB mutex, which breaks our best practice. Add a stat level in Statistics class. By default, disable to measure the mutex operations.

Test Plan: Add a unit test to make sure it is off by default.

Reviewers: rven, anthony, IslamAbdelRahman, kradhakrishnan, andrewkr, yhchiang

Reviewed By: yhchiang

Subscribers: MarkCallaghan, leveldb, dhruba

Differential Revision: https://reviews.facebook.net/D53367
main
sdong 9 years ago
parent 0c433cd1eb
commit d20915d52a
  1. 1
      HISTORY.md
  2. 14
      db/db_test.cc
  3. 13
      include/rocksdb/statistics.h
  4. 13
      util/instrumented_mutex.cc

@ -2,6 +2,7 @@
## Unreleased ## Unreleased
### Public API Changes ### Public API Changes
* Add a new perf context level between kEnableCount and kEnableTime. Level 2 now doesn't include timers for mutexes. * Add a new perf context level between kEnableCount and kEnableTime. Level 2 now doesn't include timers for mutexes.
* Statistics of mutex operation durations will not be measured by default. If you want to have them enabled, you need to set Statistics::stats_level_ to kAll.
### New Features ### New Features
* ldb tool now supports operations to non-default column families. * ldb tool now supports operations to non-default column families.

@ -7405,10 +7405,24 @@ TEST_F(DBTest, EncodeDecompressedBlockSizeTest) {
} }
} }
TEST_F(DBTest, MutexWaitStatsDisabledByDefault) {
Options options = CurrentOptions();
options.create_if_missing = true;
options.statistics = rocksdb::CreateDBStatistics();
CreateAndReopenWithCF({"pikachu"}, options);
const uint64_t kMutexWaitDelay = 100;
ThreadStatusUtil::TEST_SetStateDelay(ThreadStatus::STATE_MUTEX_WAIT,
kMutexWaitDelay);
ASSERT_OK(Put("hello", "rocksdb"));
ASSERT_EQ(TestGetTickerCount(options, DB_MUTEX_WAIT_MICROS), 0);
ThreadStatusUtil::TEST_SetStateDelay(ThreadStatus::STATE_MUTEX_WAIT, 0);
}
TEST_F(DBTest, MutexWaitStats) { TEST_F(DBTest, MutexWaitStats) {
Options options = CurrentOptions(); Options options = CurrentOptions();
options.create_if_missing = true; options.create_if_missing = true;
options.statistics = rocksdb::CreateDBStatistics(); options.statistics = rocksdb::CreateDBStatistics();
options.statistics->stats_level_ = StatsLevel::kAll;
CreateAndReopenWithCF({"pikachu"}, options); CreateAndReopenWithCF({"pikachu"}, options);
const uint64_t kMutexWaitDelay = 100; const uint64_t kMutexWaitDelay = 100;
ThreadStatusUtil::TEST_SetStateDelay( ThreadStatusUtil::TEST_SetStateDelay(

@ -109,6 +109,7 @@ enum Tickers : uint32_t {
// Writer has to wait for compaction or flush to finish. // Writer has to wait for compaction or flush to finish.
STALL_MICROS, STALL_MICROS,
// The wait time for db mutex. // The wait time for db mutex.
// Disabled by default. To enable it set stats level to kAll
DB_MUTEX_WAIT_MICROS, DB_MUTEX_WAIT_MICROS,
RATE_LIMIT_DELAY_MILLIS, RATE_LIMIT_DELAY_MILLIS,
NO_ITERATORS, // number of iterators currently open NO_ITERATORS, // number of iterators currently open
@ -316,6 +317,16 @@ struct HistogramData {
double standard_deviation; double standard_deviation;
}; };
enum StatsLevel {
// Collect all stats except the counters requiring to get time inside the
// mutex lock.
kExceptTimeForMutex,
// Collect all stats, including measuring duration of mutex operations.
// If getting time is expensive on the platform to run, it can
// reduce scalability to more threads, especialy for writes.
kAll,
};
// Analyze the performance of a db // Analyze the performance of a db
class Statistics { class Statistics {
public: public:
@ -339,6 +350,8 @@ class Statistics {
virtual bool HistEnabledForType(uint32_t type) const { virtual bool HistEnabledForType(uint32_t type) const {
return type < HISTOGRAM_ENUM_MAX; return type < HISTOGRAM_ENUM_MAX;
} }
StatsLevel stats_level_ = kExceptTimeForMutex;
}; };
// Create a concrete DBStatistics object // Create a concrete DBStatistics object

@ -8,11 +8,18 @@
#include "util/thread_status_util.h" #include "util/thread_status_util.h"
namespace rocksdb { namespace rocksdb {
namespace {
bool ShouldReportToStats(Env* env, Statistics* stats) {
return env != nullptr && stats != nullptr &&
stats->stats_level_ != kExceptTimeForMutex;
}
} // namespace
void InstrumentedMutex::Lock() { void InstrumentedMutex::Lock() {
PERF_CONDITIONAL_TIMER_FOR_MUTEX_GUARD(db_mutex_lock_nanos, PERF_CONDITIONAL_TIMER_FOR_MUTEX_GUARD(db_mutex_lock_nanos,
stats_code_ == DB_MUTEX_WAIT_MICROS); stats_code_ == DB_MUTEX_WAIT_MICROS);
uint64_t wait_time_micros = 0; uint64_t wait_time_micros = 0;
if (env_ != nullptr && stats_ != nullptr) { if (ShouldReportToStats(env_, stats_)) {
{ {
StopWatch sw(env_, nullptr, 0, &wait_time_micros); StopWatch sw(env_, nullptr, 0, &wait_time_micros);
LockInternal(); LockInternal();
@ -34,7 +41,7 @@ void InstrumentedCondVar::Wait() {
PERF_CONDITIONAL_TIMER_FOR_MUTEX_GUARD(db_condition_wait_nanos, PERF_CONDITIONAL_TIMER_FOR_MUTEX_GUARD(db_condition_wait_nanos,
stats_code_ == DB_MUTEX_WAIT_MICROS); stats_code_ == DB_MUTEX_WAIT_MICROS);
uint64_t wait_time_micros = 0; uint64_t wait_time_micros = 0;
if (env_ != nullptr && stats_ != nullptr) { if (ShouldReportToStats(env_, stats_)) {
{ {
StopWatch sw(env_, nullptr, 0, &wait_time_micros); StopWatch sw(env_, nullptr, 0, &wait_time_micros);
WaitInternal(); WaitInternal();
@ -57,7 +64,7 @@ bool InstrumentedCondVar::TimedWait(uint64_t abs_time_us) {
stats_code_ == DB_MUTEX_WAIT_MICROS); stats_code_ == DB_MUTEX_WAIT_MICROS);
uint64_t wait_time_micros = 0; uint64_t wait_time_micros = 0;
bool result = false; bool result = false;
if (env_ != nullptr && stats_ != nullptr) { if (ShouldReportToStats(env_, stats_)) {
{ {
StopWatch sw(env_, nullptr, 0, &wait_time_micros); StopWatch sw(env_, nullptr, 0, &wait_time_micros);
result = TimedWaitInternal(abs_time_us); result = TimedWaitInternal(abs_time_us);

Loading…
Cancel
Save