Some minor refactoring on the code

Summary: I made some cleanup while reading the source code in `db`. Most changes are about style, naming or C++ 11 new features.

Test Plan: ran `make check`

Reviewers: haobo, dhruba, sdong

CC: leveldb

Differential Revision: https://reviews.facebook.net/D15009
main
kailiu 11 years ago
parent f1cec73a76
commit 476416c27c
  1. 144
      db/db_impl.cc
  2. 7
      db/db_impl.h
  3. 1
      db/log_format.h
  4. 8
      db/version_edit.cc
  5. 4
      db/version_edit.h
  6. 98
      db/version_set.cc
  7. 4
      db/version_set.h

@ -20,8 +20,8 @@
#include <vector> #include <vector>
#include "db/builder.h" #include "db/builder.h"
#include "db/dbformat.h"
#include "db/db_iter.h" #include "db/db_iter.h"
#include "db/dbformat.h"
#include "db/filename.h" #include "db/filename.h"
#include "db/log_reader.h" #include "db/log_reader.h"
#include "db/log_writer.h" #include "db/log_writer.h"
@ -43,7 +43,6 @@
#include "rocksdb/statistics.h" #include "rocksdb/statistics.h"
#include "rocksdb/status.h" #include "rocksdb/status.h"
#include "rocksdb/table.h" #include "rocksdb/table.h"
#include "port/port.h"
#include "table/block.h" #include "table/block.h"
#include "table/block_based_table_factory.h" #include "table/block_based_table_factory.h"
#include "table/merger.h" #include "table/merger.h"
@ -59,7 +58,7 @@
namespace rocksdb { namespace rocksdb {
void dumpLeveldbBuildVersion(Logger * log); void DumpLeveldbBuildVersion(Logger * log);
// Information kept for every waiting writer // Information kept for every waiting writer
struct DBImpl::Writer { struct DBImpl::Writer {
@ -266,9 +265,7 @@ DBImpl::DBImpl(const Options& options, const std::string& dbname)
storage_options_(options), storage_options_(options),
bg_work_gate_closed_(false), bg_work_gate_closed_(false),
refitting_level_(false) { refitting_level_(false) {
mem_->Ref(); mem_->Ref();
env_->GetAbsolutePath(dbname, &db_absolute_path_); env_->GetAbsolutePath(dbname, &db_absolute_path_);
stall_leveln_slowdown_.resize(options.num_levels); stall_leveln_slowdown_.resize(options.num_levels);
@ -282,16 +279,15 @@ DBImpl::DBImpl(const Options& options, const std::string& dbname)
const int table_cache_size = options_.max_open_files - 10; const int table_cache_size = options_.max_open_files - 10;
table_cache_.reset(new TableCache(dbname_, &options_, table_cache_.reset(new TableCache(dbname_, &options_,
storage_options_, table_cache_size)); storage_options_, table_cache_size));
versions_.reset(new VersionSet(dbname_, &options_, storage_options_, versions_.reset(new VersionSet(dbname_, &options_, storage_options_,
table_cache_.get(), &internal_comparator_)); table_cache_.get(), &internal_comparator_));
dumpLeveldbBuildVersion(options_.info_log.get()); DumpLeveldbBuildVersion(options_.info_log.get());
options_.Dump(options_.info_log.get()); options_.Dump(options_.info_log.get());
char name[100]; char name[100];
Status st = env_->GetHostName(name, 100L); Status s = env_->GetHostName(name, 100L);
if (st.ok()) { if (s.ok()) {
host_name_ = name; host_name_ = name;
} else { } else {
Log(options_.info_log, "Can't get hostname, use localhost as host name."); Log(options_.info_log, "Can't get hostname, use localhost as host name.");
@ -502,7 +498,7 @@ void DBImpl::SuperVersion::Init(MemTable* new_mem, const MemTableList& new_imm,
} }
// Returns the list of live files in 'sst_live' and the list // Returns the list of live files in 'sst_live' and the list
// of all files in the filesystem in 'all_files'. // of all files in the filesystem in 'candidate_files'.
// no_full_scan = true -- never do the full scan using GetChildren() // no_full_scan = true -- never do the full scan using GetChildren()
// force = false -- don't force the full scan, except every // force = false -- don't force the full scan, except every
// options_.delete_obsolete_files_period_micros // options_.delete_obsolete_files_period_micros
@ -554,15 +550,18 @@ void DBImpl::FindObsoleteFiles(DeletionState& deletion_state,
versions_->AddLiveFiles(&deletion_state.sst_live); versions_->AddLiveFiles(&deletion_state.sst_live);
if (doing_the_full_scan) { if (doing_the_full_scan) {
// set of all files in the directory // set of all files in the directory. We'll exclude files that are still
env_->GetChildren(dbname_, &deletion_state.all_files); // Ignore errors // alive in the subsequent processings.
env_->GetChildren(
dbname_, &deletion_state.candidate_files
); // Ignore errors
//Add log files in wal_dir //Add log files in wal_dir
if (options_.wal_dir != dbname_) { if (options_.wal_dir != dbname_) {
std::vector<std::string> log_files; std::vector<std::string> log_files;
env_->GetChildren(options_.wal_dir, &log_files); // Ignore errors env_->GetChildren(options_.wal_dir, &log_files); // Ignore errors
deletion_state.all_files.insert( deletion_state.candidate_files.insert(
deletion_state.all_files.end(), deletion_state.candidate_files.end(),
log_files.begin(), log_files.begin(),
log_files.end() log_files.end()
); );
@ -575,11 +574,10 @@ void DBImpl::FindObsoleteFiles(DeletionState& deletion_state,
// files in sst_delete_files and log_delete_files. // files in sst_delete_files and log_delete_files.
// It is not necessary to hold the mutex when invoking this method. // It is not necessary to hold the mutex when invoking this method.
void DBImpl::PurgeObsoleteFiles(DeletionState& state) { void DBImpl::PurgeObsoleteFiles(DeletionState& state) {
// check if there is anything to do // check if there is anything to do
if (!state.all_files.size() && if (state.candidate_files.empty() &&
!state.sst_delete_files.size() && state.sst_delete_files.empty() &&
!state.log_delete_files.size()) { state.log_delete_files.empty()) {
return; return;
} }
@ -589,38 +587,52 @@ void DBImpl::PurgeObsoleteFiles(DeletionState& state) {
if (state.manifest_file_number == 0) { if (state.manifest_file_number == 0) {
return; return;
} }
uint64_t number;
FileType type;
std::vector<std::string> old_log_files; std::vector<std::string> old_log_files;
// Now, convert live list to an unordered set, WITHOUT mutex held; // Now, convert live list to an unordered set, WITHOUT mutex held;
// set is slow. // set is slow.
std::unordered_set<uint64_t> live_set(state.sst_live.begin(), std::unordered_set<uint64_t> sst_live(
state.sst_live.end()); state.sst_live.begin(), state.sst_live.end()
);
state.all_files.reserve(state.all_files.size() + auto& candidate_files = state.candidate_files;
state.sst_delete_files.size()); candidate_files.reserve(
candidate_files.size() +
state.sst_delete_files.size() +
state.log_delete_files.size());
// We may ignore the dbname when generating the file names.
const char* kDumbDbName = "";
for (auto file : state.sst_delete_files) { for (auto file : state.sst_delete_files) {
state.all_files.push_back(TableFileName("", file->number).substr(1)); candidate_files.push_back(
TableFileName(kDumbDbName, file->number).substr(1)
);
delete file; delete file;
} }
state.all_files.reserve(state.all_files.size() + for (auto file_num : state.log_delete_files) {
state.log_delete_files.size()); if (file_num > 0) {
for (auto filenum : state.log_delete_files) { candidate_files.push_back(
if (filenum > 0) { LogFileName(kDumbDbName, file_num).substr(1)
state.all_files.push_back(LogFileName("", filenum).substr(1)); );
} }
} }
// dedup state.all_files so we don't try to delete the same // dedup state.candidate_files so we don't try to delete the same
// file twice // file twice
sort(state.all_files.begin(), state.all_files.end()); sort(candidate_files.begin(), candidate_files.end());
auto unique_end = unique(state.all_files.begin(), state.all_files.end()); candidate_files.erase(
unique(candidate_files.begin(), candidate_files.end()),
candidate_files.end()
);
for (const auto& to_delete : candidate_files) {
uint64_t number;
FileType type;
// Ignore file if we cannot recognize it.
if (!ParseFileName(to_delete, &number, &type)) {
continue;
}
for (size_t i = 0; state.all_files.begin() + i < unique_end; i++) {
if (ParseFileName(state.all_files[i], &number, &type)) {
bool keep = true; bool keep = true;
switch (type) { switch (type) {
case kLogFile: case kLogFile:
@ -633,17 +645,17 @@ void DBImpl::PurgeObsoleteFiles(DeletionState& state) {
keep = (number >= state.manifest_file_number); keep = (number >= state.manifest_file_number);
break; break;
case kTableFile: case kTableFile:
keep = (live_set.find(number) != live_set.end()); keep = (sst_live.find(number) != sst_live.end());
break; break;
case kTempFile: case kTempFile:
// Any temp files that are currently being written to must // Any temp files that are currently being written to must
// be recorded in pending_outputs_, which is inserted into "live" // be recorded in pending_outputs_, which is inserted into "live"
keep = (live_set.find(number) != live_set.end()); keep = (sst_live.find(number) != sst_live.end());
break; break;
case kInfoLogFile: case kInfoLogFile:
keep = true; keep = true;
if (number != 0) { if (number != 0) {
old_log_files.push_back(state.all_files[i]); old_log_files.push_back(to_delete);
} }
break; break;
case kCurrentFile: case kCurrentFile:
@ -654,35 +666,35 @@ void DBImpl::PurgeObsoleteFiles(DeletionState& state) {
break; break;
} }
if (!keep) { if (keep) {
continue;
}
if (type == kTableFile) { if (type == kTableFile) {
// evict from cache // evict from cache
table_cache_->Evict(number); table_cache_->Evict(number);
} }
std::string fname = ((type == kLogFile) ? options_.wal_dir : dbname_) + std::string fname = ((type == kLogFile) ? options_.wal_dir : dbname_) +
"/" + state.all_files[i]; "/" + to_delete;
Log(options_.info_log, Log(options_.info_log,
"Delete type=%d #%lu", "Delete type=%d #%lu",
int(type), int(type),
(unsigned long)number); (unsigned long)number);
Status st; if (type == kLogFile &&
if (type == kLogFile && (options_.WAL_ttl_seconds > 0 || (options_.WAL_ttl_seconds > 0 || options_.WAL_size_limit_MB > 0)) {
options_.WAL_size_limit_MB > 0)) { Status s = env_->RenameFile(fname,
st = env_->RenameFile(fname,
ArchivedLogFileName(options_.wal_dir, number)); ArchivedLogFileName(options_.wal_dir, number));
if (!st.ok()) { if (!s.ok()) {
Log(options_.info_log, Log(options_.info_log,
"RenameFile logfile #%lu FAILED -- %s\n", "RenameFile logfile #%lu FAILED -- %s\n",
(unsigned long)number, st.ToString().c_str()); (unsigned long)number, s.ToString().c_str());
} }
} else { } else {
st = env_->DeleteFile(fname); Status s = env_->DeleteFile(fname);
if (!st.ok()) { if (!s.ok()) {
Log(options_.info_log, "Delete type=%d #%lu FAILED -- %s\n", Log(options_.info_log, "Delete type=%d #%lu FAILED -- %s\n",
int(type), (unsigned long)number, st.ToString().c_str()); int(type), (unsigned long)number, s.ToString().c_str());
}
}
} }
} }
} }
@ -839,7 +851,9 @@ void DBImpl::PurgeObsoleteWALFiles() {
// If externalTable is set, then apply recovered transactions // If externalTable is set, then apply recovered transactions
// to that table. This is used for readonly mode. // to that table. This is used for readonly mode.
Status DBImpl::Recover(VersionEdit* edit, MemTable* external_table, Status DBImpl::Recover(
VersionEdit* edit,
MemTable* external_table,
bool error_if_log_file_exist) { bool error_if_log_file_exist) {
mutex_.AssertHeld(); mutex_.AssertHeld();
@ -906,10 +920,11 @@ Status DBImpl::Recover(VersionEdit* edit, MemTable* external_table,
if (!s.ok()) { if (!s.ok()) {
return s; return s;
} }
uint64_t number;
FileType type;
std::vector<uint64_t> logs; std::vector<uint64_t> logs;
for (size_t i = 0; i < filenames.size(); i++) { for (size_t i = 0; i < filenames.size(); i++) {
uint64_t number;
FileType type;
if (ParseFileName(filenames[i], &number, &type) if (ParseFileName(filenames[i], &number, &type)
&& type == kLogFile && type == kLogFile
&& ((number >= min_log) || (number == prev_log))) { && ((number >= min_log) || (number == prev_log))) {
@ -925,12 +940,12 @@ Status DBImpl::Recover(VersionEdit* edit, MemTable* external_table,
// Recover in the order in which the logs were generated // Recover in the order in which the logs were generated
std::sort(logs.begin(), logs.end()); std::sort(logs.begin(), logs.end());
for (size_t i = 0; i < logs.size(); i++) { for (const auto& log : logs) {
s = RecoverLogFile(logs[i], edit, &max_sequence, external_table); s = RecoverLogFile(log, edit, &max_sequence, external_table);
// The previous incarnation may not have written any MANIFEST // The previous incarnation may not have written any MANIFEST
// records after allocating this log number. So we manually // records after allocating this log number. So we manually
// update the file number allocation counter in VersionSet. // update the file number allocation counter in VersionSet.
versions_->MarkFileNumberUsed(logs[i]); versions_->MarkFileNumberUsed(log);
} }
if (s.ok()) { if (s.ok()) {
@ -1147,7 +1162,6 @@ Status DBImpl::WriteLevel0Table(std::vector<MemTable*> &mems, VersionEdit* edit,
} }
base->Unref(); base->Unref();
// re-acquire the most current version // re-acquire the most current version
base = versions_->current(); base = versions_->current();
@ -3285,7 +3299,7 @@ Status DBImpl::MakeRoomForWrite(bool force,
} else { } else {
unique_ptr<WritableFile> lfile; unique_ptr<WritableFile> lfile;
MemTable* memtmp = nullptr; MemTable* new_mem = nullptr;
// Attempt to switch to a new memtable and trigger compaction of old. // Attempt to switch to a new memtable and trigger compaction of old.
// Do this without holding the dbmutex lock. // Do this without holding the dbmutex lock.
@ -3306,7 +3320,7 @@ Status DBImpl::MakeRoomForWrite(bool force,
// Our final size should be less than write_buffer_size // Our final size should be less than write_buffer_size
// (compression, etc) but err on the side of caution. // (compression, etc) but err on the side of caution.
lfile->SetPreallocationBlockSize(1.1 * options_.write_buffer_size); lfile->SetPreallocationBlockSize(1.1 * options_.write_buffer_size);
memtmp = new MemTable( new_mem = new MemTable(
internal_comparator_, mem_rep_factory_, NumberLevels(), options_); internal_comparator_, mem_rep_factory_, NumberLevels(), options_);
new_superversion = new SuperVersion(options_.max_write_buffer_number); new_superversion = new SuperVersion(options_.max_write_buffer_number);
} }
@ -3315,7 +3329,7 @@ Status DBImpl::MakeRoomForWrite(bool force,
if (!s.ok()) { if (!s.ok()) {
// Avoid chewing through file number space in a tight loop. // Avoid chewing through file number space in a tight loop.
versions_->ReuseFileNumber(new_log_number); versions_->ReuseFileNumber(new_log_number);
assert (!memtmp); assert (!new_mem);
break; break;
} }
logfile_number_ = new_log_number; logfile_number_ = new_log_number;
@ -3325,7 +3339,7 @@ Status DBImpl::MakeRoomForWrite(bool force,
if (force) { if (force) {
imm_.FlushRequested(); imm_.FlushRequested();
} }
mem_ = memtmp; mem_ = new_mem;
mem_->Ref(); mem_->Ref();
Log(options_.info_log, Log(options_.info_log,
"New memtable created with log file: #%lu\n", "New memtable created with log file: #%lu\n",
@ -3806,7 +3820,7 @@ Status DB::Open(const Options& options, const std::string& dbname, DB** dbptr) {
delete impl; delete impl;
return s; return s;
} }
impl->mutex_.Lock(); impl->mutex_.Lock(); // DBImpl::Recover() requires lock being held
VersionEdit edit(impl->NumberLevels()); VersionEdit edit(impl->NumberLevels());
s = impl->Recover(&edit); // Handles create_if_missing, error_if_exists s = impl->Recover(&edit); // Handles create_if_missing, error_if_exists
if (s.ok()) { if (s.ok()) {
@ -3929,7 +3943,7 @@ Status DestroyDB(const std::string& dbname, const Options& options) {
// //
// A global method that can dump out the build version // A global method that can dump out the build version
void dumpLeveldbBuildVersion(Logger * log) { void DumpLeveldbBuildVersion(Logger * log) {
Log(log, "Git sha %s", rocksdb_build_git_sha); Log(log, "Git sha %s", rocksdb_build_git_sha);
Log(log, "Compile time %s %s", Log(log, "Compile time %s %s",
rocksdb_build_compile_time, rocksdb_build_compile_date); rocksdb_build_compile_time, rocksdb_build_compile_date);

@ -11,6 +11,7 @@
#include <deque> #include <deque>
#include <set> #include <set>
#include <vector> #include <vector>
#include "db/dbformat.h" #include "db/dbformat.h"
#include "db/log_writer.h" #include "db/log_writer.h"
#include "db/snapshot.h" #include "db/snapshot.h"
@ -159,7 +160,7 @@ class DBImpl : public DB {
// needed for CleanupIteratorState // needed for CleanupIteratorState
struct DeletionState { struct DeletionState {
inline bool HaveSomethingToDelete() const { inline bool HaveSomethingToDelete() const {
return all_files.size() || return candidate_files.size() ||
sst_delete_files.size() || sst_delete_files.size() ||
log_delete_files.size(); log_delete_files.size();
} }
@ -167,7 +168,7 @@ class DBImpl : public DB {
// a list of all files that we'll consider deleting // a list of all files that we'll consider deleting
// (every once in a while this is filled up with all files // (every once in a while this is filled up with all files
// in the DB directory) // in the DB directory)
std::vector<std::string> all_files; std::vector<std::string> candidate_files;
// the list of all live sst files that cannot be deleted // the list of all live sst files that cannot be deleted
std::vector<uint64_t> sst_live; std::vector<uint64_t> sst_live;
@ -214,7 +215,7 @@ class DBImpl : public DB {
}; };
// Returns the list of live files in 'live' and the list // Returns the list of live files in 'live' and the list
// of all files in the filesystem in 'all_files'. // of all files in the filesystem in 'candidate_files'.
// If force == false and the last call was less than // If force == false and the last call was less than
// options_.delete_obsolete_files_period_micros microseconds ago, // options_.delete_obsolete_files_period_micros microseconds ago,
// it will not fill up the deletion_state // it will not fill up the deletion_state

@ -17,7 +17,6 @@ namespace log {
enum RecordType { enum RecordType {
// Zero is reserved for preallocated files // Zero is reserved for preallocated files
kZeroType = 0, kZeroType = 0,
kFullType = 1, kFullType = 1,
// For fragments // For fragments

@ -74,12 +74,10 @@ void VersionEdit::EncodeTo(std::string* dst) const {
PutLengthPrefixedSlice(dst, compact_pointers_[i].second.Encode()); PutLengthPrefixedSlice(dst, compact_pointers_[i].second.Encode());
} }
for (DeletedFileSet::const_iterator iter = deleted_files_.begin(); for (const auto& deleted : deleted_files_) {
iter != deleted_files_.end();
++iter) {
PutVarint32(dst, kDeletedFile); PutVarint32(dst, kDeletedFile);
PutVarint32(dst, iter->first); // level PutVarint32(dst, deleted.first /* level */);
PutVarint64(dst, iter->second); // file number PutVarint64(dst, deleted.second /* file number */);
} }
for (size_t i = 0; i < new_files_.size(); i++) { for (size_t i = 0; i < new_files_.size(); i++) {

@ -75,6 +75,7 @@ class VersionEdit {
const InternalKey& largest, const InternalKey& largest,
const SequenceNumber& smallest_seqno, const SequenceNumber& smallest_seqno,
const SequenceNumber& largest_seqno) { const SequenceNumber& largest_seqno) {
assert(smallest_seqno <= largest_seqno);
FileMetaData f; FileMetaData f;
f.number = file; f.number = file;
f.file_size = file_size; f.file_size = file_size;
@ -82,13 +83,12 @@ class VersionEdit {
f.largest = largest; f.largest = largest;
f.smallest_seqno = smallest_seqno; f.smallest_seqno = smallest_seqno;
f.largest_seqno = largest_seqno; f.largest_seqno = largest_seqno;
assert(smallest_seqno <= largest_seqno);
new_files_.push_back(std::make_pair(level, f)); new_files_.push_back(std::make_pair(level, f));
} }
// Delete the specified "file" from the specified "level". // Delete the specified "file" from the specified "level".
void DeleteFile(int level, uint64_t file) { void DeleteFile(int level, uint64_t file) {
deleted_files_.insert(std::make_pair(level, file)); deleted_files_.insert({level, file});
} }
// Number of edits // Number of edits

@ -958,6 +958,7 @@ class VersionSet::Builder {
} }
} }
} }
delete[] levels_; delete[] levels_;
base_->Unref(); base_->Unref();
} }
@ -1043,19 +1044,17 @@ class VersionSet::Builder {
// Delete files // Delete files
const VersionEdit::DeletedFileSet& del = edit->deleted_files_; const VersionEdit::DeletedFileSet& del = edit->deleted_files_;
for (VersionEdit::DeletedFileSet::const_iterator iter = del.begin(); for (const auto& del_file : del) {
iter != del.end(); const auto level = del_file.first;
++iter) { const auto number = del_file.second;
const int level = iter->first;
const uint64_t number = iter->second;
levels_[level].deleted_files.insert(number); levels_[level].deleted_files.insert(number);
CheckConsistencyForDeletes(edit, number, level); CheckConsistencyForDeletes(edit, number, level);
} }
// Add new files // Add new files
for (size_t i = 0; i < edit->new_files_.size(); i++) { for (const auto& new_file : edit->new_files_) {
const int level = edit->new_files_[i].first; const int level = new_file.first;
FileMetaData* f = new FileMetaData(edit->new_files_[i].second); FileMetaData* f = new FileMetaData(new_file.second);
f->refs = 1; f->refs = 1;
// We arrange to automatically compact this file after // We arrange to automatically compact this file after
@ -1088,23 +1087,21 @@ class VersionSet::Builder {
for (int level = 0; level < vset_->NumberLevels(); level++) { for (int level = 0; level < vset_->NumberLevels(); level++) {
// Merge the set of added files with the set of pre-existing files. // Merge the set of added files with the set of pre-existing files.
// Drop any deleted files. Store the result in *v. // Drop any deleted files. Store the result in *v.
const std::vector<FileMetaData*>& base_files = base_->files_[level]; const auto& base_files = base_->files_[level];
std::vector<FileMetaData*>::const_iterator base_iter = base_files.begin(); auto base_iter = base_files.begin();
std::vector<FileMetaData*>::const_iterator base_end = base_files.end(); auto base_end = base_files.end();
const FileSet* added = levels_[level].added_files; const auto& added_files = *levels_[level].added_files;
v->files_[level].reserve(base_files.size() + added->size()); v->files_[level].reserve(base_files.size() + added_files.size());
for (FileSet::const_iterator added_iter = added->begin();
added_iter != added->end(); for (const auto& added : added_files) {
++added_iter) {
// Add all smaller files listed in base_ // Add all smaller files listed in base_
for (std::vector<FileMetaData*>::const_iterator bpos for (auto bpos = std::upper_bound(base_iter, base_end, added, cmp);
= std::upper_bound(base_iter, base_end, *added_iter, cmp);
base_iter != bpos; base_iter != bpos;
++base_iter) { ++base_iter) {
MaybeAddFile(v, level, *base_iter); MaybeAddFile(v, level, *base_iter);
} }
MaybeAddFile(v, level, *added_iter); MaybeAddFile(v, level, added);
} }
// Add remaining base files // Add remaining base files
@ -1120,7 +1117,7 @@ class VersionSet::Builder {
if (levels_[level].deleted_files.count(f->number) > 0) { if (levels_[level].deleted_files.count(f->number) > 0) {
// File is deleted: do nothing // File is deleted: do nothing
} else { } else {
std::vector<FileMetaData*>* files = &v->files_[level]; auto* files = &v->files_[level];
if (level > 0 && !files->empty()) { if (level > 0 && !files->empty()) {
// Must not overlap // Must not overlap
assert(vset_->icmp_.Compare((*files)[files->size()-1]->largest, assert(vset_->icmp_.Compare((*files)[files->size()-1]->largest,
@ -1210,7 +1207,9 @@ void VersionSet::AppendVersion(Version* v) {
v->next_->prev_ = v; v->next_->prev_ = v;
} }
Status VersionSet::LogAndApply(VersionEdit* edit, port::Mutex* mu, Status VersionSet::LogAndApply(
VersionEdit* edit,
port::Mutex* mu,
bool new_descriptor_log) { bool new_descriptor_log) {
mu->AssertHeld(); mu->AssertHeld();
@ -1232,17 +1231,16 @@ Status VersionSet::LogAndApply(VersionEdit* edit, port::Mutex* mu,
ManifestWriter* last_writer = &w; ManifestWriter* last_writer = &w;
assert(!manifest_writers_.empty()); assert(!manifest_writers_.empty());
assert(manifest_writers_.front() == &w); assert(manifest_writers_.front() == &w);
std::deque<ManifestWriter*>::iterator iter = manifest_writers_.begin(); for (const auto& writer : manifest_writers_) {
for (; iter != manifest_writers_.end(); ++iter) { last_writer = writer;
last_writer = *iter; LogAndApplyHelper(&builder, v, writer->edit, mu);
LogAndApplyHelper(&builder, v, last_writer->edit, mu); batch_edits.push_back(writer->edit);
batch_edits.push_back(last_writer->edit);
} }
builder.SaveTo(v); builder.SaveTo(v);
// Initialize new descriptor log file if necessary by creating // Initialize new descriptor log file if necessary by creating
// a temporary file that contains a snapshot of the current version. // a temporary file that contains a snapshot of the current version.
std::string new_manifest_file; std::string new_manifest_filename;
uint64_t new_manifest_file_size = 0; uint64_t new_manifest_file_size = 0;
Status s; Status s;
// we will need this if we are creating new manifest // we will need this if we are creating new manifest
@ -1256,7 +1254,7 @@ Status VersionSet::LogAndApply(VersionEdit* edit, port::Mutex* mu,
} }
if (new_descriptor_log) { if (new_descriptor_log) {
new_manifest_file = DescriptorFileName(dbname_, manifest_file_number_); new_manifest_filename = DescriptorFileName(dbname_, manifest_file_number_);
edit->SetNextFile(next_file_number_); edit->SetNextFile(next_file_number_);
} }
@ -1271,9 +1269,10 @@ Status VersionSet::LogAndApply(VersionEdit* edit, port::Mutex* mu,
// This is fine because everything inside of this block is serialized -- // This is fine because everything inside of this block is serialized --
// only one thread can be here at the same time // only one thread can be here at the same time
if (!new_manifest_file.empty()) { if (!new_manifest_filename.empty()) {
unique_ptr<WritableFile> descriptor_file; unique_ptr<WritableFile> descriptor_file;
s = env_->NewWritableFile(new_manifest_file, &descriptor_file, s = env_->NewWritableFile(new_manifest_filename,
&descriptor_file,
storage_options_); storage_options_);
if (s.ok()) { if (s.ok()) {
descriptor_log_.reset(new log::Writer(std::move(descriptor_file))); descriptor_log_.reset(new log::Writer(std::move(descriptor_file)));
@ -1321,7 +1320,7 @@ Status VersionSet::LogAndApply(VersionEdit* edit, port::Mutex* mu,
// If we just created a new descriptor file, install it by writing a // If we just created a new descriptor file, install it by writing a
// new CURRENT file that points to it. // new CURRENT file that points to it.
if (s.ok() && !new_manifest_file.empty()) { if (s.ok() && !new_manifest_filename.empty()) {
s = SetCurrentFile(env_, dbname_, manifest_file_number_); s = SetCurrentFile(env_, dbname_, manifest_file_number_);
if (s.ok() && old_manifest_file_number < manifest_file_number_) { if (s.ok() && old_manifest_file_number < manifest_file_number_) {
// delete old manifest file // delete old manifest file
@ -1356,9 +1355,9 @@ Status VersionSet::LogAndApply(VersionEdit* edit, port::Mutex* mu,
Log(options_->info_log, "Error in committing version %lu", Log(options_->info_log, "Error in committing version %lu",
(unsigned long)v->GetVersionNumber()); (unsigned long)v->GetVersionNumber());
delete v; delete v;
if (!new_manifest_file.empty()) { if (!new_manifest_filename.empty()) {
descriptor_log_.reset(); descriptor_log_.reset();
env_->DeleteFile(new_manifest_file); env_->DeleteFile(new_manifest_filename);
} }
} }
@ -1410,27 +1409,33 @@ Status VersionSet::Recover() {
}; };
// Read "CURRENT" file, which contains a pointer to the current manifest file // Read "CURRENT" file, which contains a pointer to the current manifest file
std::string current; std::string manifest_filename;
Status s = ReadFileToString(env_, CurrentFileName(dbname_), &current); Status s = ReadFileToString(
env_, CurrentFileName(dbname_), &manifest_filename
);
if (!s.ok()) { if (!s.ok()) {
return s; return s;
} }
if (current.empty() || current[current.size()-1] != '\n') { if (manifest_filename.empty() ||
manifest_filename.back() != '\n') {
return Status::Corruption("CURRENT file does not end with newline"); return Status::Corruption("CURRENT file does not end with newline");
} }
current.resize(current.size() - 1); // remove the trailing '\n'
manifest_filename.resize(manifest_filename.size() - 1);
Log(options_->info_log, "Recovering from manifest file:%s\n", Log(options_->info_log, "Recovering from manifest file:%s\n",
current.c_str()); manifest_filename.c_str());
std::string dscname = dbname_ + "/" + current; manifest_filename = dbname_ + "/" + manifest_filename;
unique_ptr<SequentialFile> file; unique_ptr<SequentialFile> manifest_file;
s = env_->NewSequentialFile(dscname, &file, storage_options_); s = env_->NewSequentialFile(
manifest_filename, &manifest_file, storage_options_
);
if (!s.ok()) { if (!s.ok()) {
return s; return s;
} }
uint64_t manifest_file_size; uint64_t manifest_file_size;
s = env_->GetFileSize(dscname, &manifest_file_size); s = env_->GetFileSize(manifest_filename, &manifest_file_size);
if (!s.ok()) { if (!s.ok()) {
return s; return s;
} }
@ -1448,7 +1453,7 @@ Status VersionSet::Recover() {
{ {
LogReporter reporter; LogReporter reporter;
reporter.status = &s; reporter.status = &s;
log::Reader reader(std::move(file), &reporter, true/*checksum*/, log::Reader reader(std::move(manifest_file), &reporter, true /*checksum*/,
0 /*initial_offset*/); 0 /*initial_offset*/);
Slice record; Slice record;
std::string scratch; std::string scratch;
@ -1489,7 +1494,6 @@ Status VersionSet::Recover() {
} }
} }
} }
file.reset();
if (s.ok()) { if (s.ok()) {
if (!have_next_file) { if (!have_next_file) {
@ -1529,7 +1533,7 @@ Status VersionSet::Recover() {
"manifest_file_number is %lu, next_file_number is %lu, " "manifest_file_number is %lu, next_file_number is %lu, "
"last_sequence is %lu, log_number is %lu," "last_sequence is %lu, log_number is %lu,"
"prev_log_number is %lu\n", "prev_log_number is %lu\n",
current.c_str(), manifest_filename.c_str(),
(unsigned long)manifest_file_number_, (unsigned long)manifest_file_number_,
(unsigned long)next_file_number_, (unsigned long)next_file_number_,
(unsigned long)last_sequence_, (unsigned long)last_sequence_,
@ -1844,9 +1848,9 @@ Status VersionSet::WriteSnapshot(log::Writer* log) {
// Save files // Save files
for (int level = 0; level < NumberLevels(); level++) { for (int level = 0; level < NumberLevels(); level++) {
const std::vector<FileMetaData*>& files = current_->files_[level]; const auto& files = current_->files_[level];
for (size_t i = 0; i < files.size(); i++) { for (size_t i = 0; i < files.size(); i++) {
const FileMetaData* f = files[i]; const auto f = files[i];
edit.AddFile(level, f->number, f->file_size, f->smallest, f->largest, edit.AddFile(level, f->number, f->file_size, f->smallest, f->largest,
f->smallest_seqno, f->largest_seqno); f->smallest_seqno, f->largest_seqno);
} }

@ -78,8 +78,8 @@ class Version {
}; };
void Get(const ReadOptions&, const LookupKey& key, std::string* val, void Get(const ReadOptions&, const LookupKey& key, std::string* val,
Status* status, MergeContext* merge_context, Status* status, MergeContext* merge_context,
GetStats* stats, const Options& db_option, bool* value_found = GetStats* stats, const Options& db_option,
nullptr); bool* value_found = nullptr);
// Adds "stats" into the current state. Returns true if a new // Adds "stats" into the current state. Returns true if a new
// compaction may need to be triggered, false otherwise. // compaction may need to be triggered, false otherwise.

Loading…
Cancel
Save