Avoid malloc in NotFound key status if no message is given.

Summary:
In some places we have NotFound status created with empty message, but it doesn't avoid a malloc. With this patch, the malloc is avoided for that case.

The motivation of it is that I found in db_bench readrandom test when all keys are not existing, about 4% of the total running time is spent on malloc of Status, plus a similar amount of CPU spent on free of them, which is not necessary.

Test Plan: make all check

Reviewers: dhruba, haobo, igor

Reviewed By: haobo

CC: leveldb

Differential Revision: https://reviews.facebook.net/D14691
main
Siying Dong 11 years ago
parent b40c052bfa
commit 18df47b79a
  1. 2
      db/memtable.cc
  2. 4
      db/version_set.cc
  3. 29
      include/rocksdb/status.h
  4. 31
      util/status.cc

@ -225,7 +225,7 @@ bool MemTable::Get(const LookupKey& key, std::string* value, Status* s,
*s = Status::Corruption("Error: Could not perform merge."); *s = Status::Corruption("Error: Could not perform merge.");
} }
} else { } else {
*s = Status::NotFound(Slice()); *s = Status::NotFound();
} }
return true; return true;
} }

@ -545,7 +545,7 @@ void Version::Get(const ReadOptions& options,
case kFound: case kFound:
return; return;
case kDeleted: case kDeleted:
*status = Status::NotFound(Slice()); // Use empty error message for speed *status = Status::NotFound(); // Use empty error message for speed
return; return;
case kCorrupt: case kCorrupt:
*status = Status::Corruption("corrupted key for ", user_key); *status = Status::Corruption("corrupted key for ", user_key);
@ -570,7 +570,7 @@ void Version::Get(const ReadOptions& options,
user_key); user_key);
} }
} else { } else {
*status = Status::NotFound(Slice()); // Use an empty error message for speed *status = Status::NotFound(); // Use an empty error message for speed
} }
} }

@ -25,7 +25,7 @@ namespace rocksdb {
class Status { class Status {
public: public:
// Create a success status. // Create a success status.
Status() : state_(nullptr) { } Status() : code_(kOk), state_(nullptr) { }
~Status() { delete[] state_; } ~Status() { delete[] state_; }
// Copy the specified status. // Copy the specified status.
@ -39,6 +39,10 @@ class Status {
static Status NotFound(const Slice& msg, const Slice& msg2 = Slice()) { static Status NotFound(const Slice& msg, const Slice& msg2 = Slice()) {
return Status(kNotFound, msg, msg2); return Status(kNotFound, msg, msg2);
} }
// Fast path for not found without malloc;
static Status NotFound() {
return Status(kNotFound);
}
static Status Corruption(const Slice& msg, const Slice& msg2 = Slice()) { static Status Corruption(const Slice& msg, const Slice& msg2 = Slice()) {
return Status(kCorruption, msg, msg2); return Status(kCorruption, msg, msg2);
} }
@ -59,7 +63,7 @@ class Status {
} }
// Returns true iff the status indicates success. // Returns true iff the status indicates success.
bool ok() const { return (state_ == nullptr); } bool ok() const { return code() == kOk; }
// Returns true iff the status indicates a NotFound error. // Returns true iff the status indicates a NotFound error.
bool IsNotFound() const { return code() == kNotFound; } bool IsNotFound() const { return code() == kNotFound; }
@ -87,13 +91,6 @@ class Status {
std::string ToString() const; std::string ToString() const;
private: private:
// OK status has a nullptr state_. Otherwise, state_ is a new[] array
// of the following form:
// state_[0..3] == length of message
// state_[4] == code
// state_[5..] == message
const char* state_;
enum Code { enum Code {
kOk = 0, kOk = 0,
kNotFound = 1, kNotFound = 1,
@ -105,20 +102,30 @@ class Status {
kIncomplete = 7 kIncomplete = 7
}; };
// A nullptr state_ (which is always the case for OK) means the message
// is empty.
// of the following form:
// state_[0..3] == length of message
// state_[4..] == message
Code code_;
const char* state_;
Code code() const { Code code() const {
return (state_ == nullptr) ? kOk : static_cast<Code>(state_[4]); return code_;
} }
explicit Status(Code code) : code_(code), state_(nullptr) { }
Status(Code code, const Slice& msg, const Slice& msg2); Status(Code code, const Slice& msg, const Slice& msg2);
static const char* CopyState(const char* s); static const char* CopyState(const char* s);
}; };
inline Status::Status(const Status& s) { inline Status::Status(const Status& s) {
code_ = s.code_;
state_ = (s.state_ == nullptr) ? nullptr : CopyState(s.state_); state_ = (s.state_ == nullptr) ? nullptr : CopyState(s.state_);
} }
inline void Status::operator=(const Status& s) { inline void Status::operator=(const Status& s) {
// The following condition catches both aliasing (when this == &s), // The following condition catches both aliasing (when this == &s),
// and the common case where both s and *this are ok. // and the common case where both s and *this are ok.
code_ = s.code_;
if (state_ != s.state_) { if (state_ != s.state_) {
delete[] state_; delete[] state_;
state_ = (s.state_ == nullptr) ? nullptr : CopyState(s.state_); state_ = (s.state_ == nullptr) ? nullptr : CopyState(s.state_);

@ -16,38 +16,34 @@ namespace rocksdb {
const char* Status::CopyState(const char* state) { const char* Status::CopyState(const char* state) {
uint32_t size; uint32_t size;
memcpy(&size, state, sizeof(size)); memcpy(&size, state, sizeof(size));
char* result = new char[size + 5]; char* result = new char[size + 4];
memcpy(result, state, size + 5); memcpy(result, state, size + 4);
return result; return result;
} }
Status::Status(Code code, const Slice& msg, const Slice& msg2) { Status::Status(Code code, const Slice& msg, const Slice& msg2) :
code_(code) {
assert(code != kOk); assert(code != kOk);
const uint32_t len1 = msg.size(); const uint32_t len1 = msg.size();
const uint32_t len2 = msg2.size(); const uint32_t len2 = msg2.size();
const uint32_t size = len1 + (len2 ? (2 + len2) : 0); const uint32_t size = len1 + (len2 ? (2 + len2) : 0);
char* result = new char[size + 5]; char* result = new char[size + 4];
memcpy(result, &size, sizeof(size)); memcpy(result, &size, sizeof(size));
result[4] = static_cast<char>(code); memcpy(result + 4, msg.data(), len1);
memcpy(result + 5, msg.data(), len1);
if (len2) { if (len2) {
result[5 + len1] = ':'; result[4 + len1] = ':';
result[6 + len1] = ' '; result[5 + len1] = ' ';
memcpy(result + 7 + len1, msg2.data(), len2); memcpy(result + 6 + len1, msg2.data(), len2);
} }
state_ = result; state_ = result;
} }
std::string Status::ToString() const { std::string Status::ToString() const {
if (state_ == nullptr) {
return "OK";
} else {
char tmp[30]; char tmp[30];
const char* type; const char* type;
switch (code()) { switch (code_) {
case kOk: case kOk:
type = "OK"; return "OK";
break;
case kNotFound: case kNotFound:
type = "NotFound: "; type = "NotFound: ";
break; break;
@ -73,11 +69,12 @@ std::string Status::ToString() const {
break; break;
} }
std::string result(type); std::string result(type);
if (state_ != nullptr) {
uint32_t length; uint32_t length;
memcpy(&length, state_, sizeof(length)); memcpy(&length, state_, sizeof(length));
result.append(state_ + 5, length); result.append(state_ + 4, length);
return result;
} }
return result;
} }
} // namespace rocksdb } // namespace rocksdb

Loading…
Cancel
Save