Skip to content

Commit

Permalink
Introduce InternalIteratorBase::NextAndGetResult() (facebook#5197)
Browse files Browse the repository at this point in the history
Summary:
In long scans, virtual function calls of Next(), Valid(), key() and value() are not trivial. By introducing NextAndGetResult(), Some of the Next(), Valid() and key() calls are consolidated into one virtual function call to reduce CPU.
Also did some inline tricks and add some "final" randomly in some functions. Even without the "final" annotation, most Next() calls are inlined with -O3, but sometimes with a final it is inlined by O2 too. It doesn't hurt to add those final annotations.
Pull Request resolved: facebook#5197

Differential Revision: D14945977

Pulled By: siying

fbshipit-source-id: 7003969f9a5f1d5717f0bda503b91d19ba75ed88
  • Loading branch information
siying authored and facebook-github-bot committed Apr 18, 2019
1 parent 6c2bf9e commit 992dfc7
Show file tree
Hide file tree
Showing 7 changed files with 60 additions and 12 deletions.
2 changes: 1 addition & 1 deletion db/db_iter.cc
Original file line number Diff line number Diff line change
Expand Up @@ -231,7 +231,7 @@ class DBIter final: public Iterator {
return Status::InvalidArgument("Unidentified property.");
}

void Next() override;
void Next() final override;
void Prev() override;
void Seek(const Slice& target) override;
void SeekForPrev(const Slice& target) override;
Expand Down
23 changes: 18 additions & 5 deletions db/version_set.cc
Original file line number Diff line number Diff line change
Expand Up @@ -906,7 +906,8 @@ class LevelIterator final : public InternalIterator {
void SeekForPrev(const Slice& target) override;
void SeekToFirst() override;
void SeekToLast() override;
void Next() override;
void Next() final override;
bool NextAndGetResult(Slice* ret_key) override;
void Prev() override;

bool Valid() const override { return file_iter_.Valid(); }
Expand Down Expand Up @@ -942,6 +943,13 @@ class LevelIterator final : public InternalIterator {
void SetFileIterator(InternalIterator* iter);
void InitFileIterator(size_t new_file_index);

// Called by both of Next() and NextAndGetResult(). Force inline.
void NextImpl() {
assert(Valid());
file_iter_.Next();
SkipEmptyFileForward();
}

const Slice& file_smallest_key(size_t file_index) {
assert(file_index < flevel_->num_files);
return flevel_->files[file_index].smallest_key;
Expand Down Expand Up @@ -1037,10 +1045,15 @@ void LevelIterator::SeekToLast() {
SkipEmptyFileBackward();
}

void LevelIterator::Next() {
assert(Valid());
file_iter_.Next();
SkipEmptyFileForward();
void LevelIterator::Next() { NextImpl(); }

bool LevelIterator::NextAndGetResult(Slice* ret_key) {
NextImpl();
bool is_valid = Valid();
if (is_valid) {
*ret_key = key();
}
return is_valid;
}

void LevelIterator::Prev() {
Expand Down
2 changes: 1 addition & 1 deletion table/block.h
Original file line number Diff line number Diff line change
Expand Up @@ -392,7 +392,7 @@ class DataBlockIter final : public BlockIter<Slice> {

virtual void Prev() override;

virtual void Next() override;
virtual void Next() final override;

// Try to advance to the next entry in the block. If there is data corruption
// or error, report it to the caller instead of aborting the process. May
Expand Down
24 changes: 22 additions & 2 deletions table/block_based_table_reader.cc
Original file line number Diff line number Diff line change
Expand Up @@ -2435,6 +2435,17 @@ void BlockBasedTableIterator<TBlockIter, TValue>::Next() {
FindKeyForward();
}

template <class TBlockIter, typename TValue>
bool BlockBasedTableIterator<TBlockIter, TValue>::NextAndGetResult(
Slice* ret_key) {
Next();
bool is_valid = Valid();
if (is_valid) {
*ret_key = key();
}
return is_valid;
}

template <class TBlockIter, typename TValue>
void BlockBasedTableIterator<TBlockIter, TValue>::Prev() {
assert(block_iter_points_to_real_block_);
Expand Down Expand Up @@ -2493,10 +2504,10 @@ void BlockBasedTableIterator<TBlockIter, TValue>::InitDataBlock() {
}

template <class TBlockIter, typename TValue>
void BlockBasedTableIterator<TBlockIter, TValue>::FindKeyForward() {
void BlockBasedTableIterator<TBlockIter, TValue>::FindBlockForward() {
// TODO the while loop inherits from two-level-iterator. We don't know
// whether a block can be empty so it can be replaced by an "if".
while (!block_iter_.Valid()) {
do {
if (!block_iter_.status().ok()) {
return;
}
Expand Down Expand Up @@ -2528,6 +2539,15 @@ void BlockBasedTableIterator<TBlockIter, TValue>::FindKeyForward() {
} else {
return;
}
} while (!block_iter_.Valid());
}

template <class TBlockIter, typename TValue>
void BlockBasedTableIterator<TBlockIter, TValue>::FindKeyForward() {
assert(!is_out_of_bound_);

if (!block_iter_.Valid()) {
FindBlockForward();
}
}

Expand Down
6 changes: 4 additions & 2 deletions table/block_based_table_reader.h
Original file line number Diff line number Diff line change
Expand Up @@ -611,7 +611,8 @@ class BlockBasedTableIterator : public InternalIteratorBase<TValue> {
void SeekForPrev(const Slice& target) override;
void SeekToFirst() override;
void SeekToLast() override;
void Next() override;
void Next() final override;
bool NextAndGetResult(Slice* ret_key) override;
void Prev() override;
bool Valid() const override {
return !is_out_of_bound_ && block_iter_points_to_real_block_ &&
Expand Down Expand Up @@ -688,7 +689,8 @@ class BlockBasedTableIterator : public InternalIteratorBase<TValue> {
}

void InitDataBlock();
void FindKeyForward();
inline void FindKeyForward();
void FindBlockForward();
void FindKeyBackward();
void CheckOutOfBound();

Expand Down
9 changes: 9 additions & 0 deletions table/internal_iterator.h
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,15 @@ class InternalIteratorBase : public Cleanable {
// REQUIRES: Valid()
virtual void Next() = 0;

virtual bool NextAndGetResult(Slice* ret_key) {
Next();
bool is_valid = Valid();
if (is_valid) {
*ret_key = key();
}
return is_valid;
}

// Moves to the previous entry in the source. After this call, Valid() is
// true iff the iterator was not positioned at the first entry in source.
// REQUIRES: Valid()
Expand Down
6 changes: 5 additions & 1 deletion table/iterator_wrapper.h
Original file line number Diff line number Diff line change
Expand Up @@ -63,7 +63,11 @@ class IteratorWrapperBase {
}
// Methods below require iter() != nullptr
Status status() const { assert(iter_); return iter_->status(); }
void Next() { assert(iter_); iter_->Next(); Update(); }
void Next() {
assert(iter_);
valid_ = iter_->NextAndGetResult(&key_);
assert(!valid_ || iter_->status().ok());
}
void Prev() { assert(iter_); iter_->Prev(); Update(); }
void Seek(const Slice& k) { assert(iter_); iter_->Seek(k); Update(); }
void SeekForPrev(const Slice& k) {
Expand Down

0 comments on commit 992dfc7

Please sign in to comment.