apply ReadOptions.iterate_upper_bound to transaction iterator (#4656)

Summary:
Currently transaction iterator does not apply `ReadOptions.iterate_upper_bound` when iterating. This PR attempts to fix the problem by having `BaseDeltaIterator` enforcing the upper bound check when iterator state is changed.
Pull Request resolved: https://github.com/facebook/rocksdb/pull/4656

Differential Revision: D13039257

Pulled By: miasantreble

fbshipit-source-id: 909eb9f6b4597a4d80418fb139f32ec82c6ec1d1
main
Zhongyi Xie 6 years ago committed by Facebook Github Bot
parent a2de8e52bb
commit b76398a82b
  1. 28
      db/c.cc
  2. 5
      db/c_test.c
  3. 9
      include/rocksdb/c.h
  4. 6
      include/rocksdb/utilities/write_batch_with_index.h
  5. 7
      java/rocksjni/write_batch_with_index.cc
  6. 69
      java/src/main/java/org/rocksdb/WriteBatchWithIndex.java
  7. 5
      java/src/test/java/org/rocksdb/WriteBatchWithIndexTest.java
  8. 45
      utilities/transactions/optimistic_transaction_test.cc
  9. 4
      utilities/transactions/transaction_base.cc
  10. 4
      utilities/transactions/write_prepared_txn.cc
  11. 2
      utilities/transactions/write_unprepared_txn.cc
  12. 62
      utilities/write_batch_with_index/write_batch_with_index.cc

@ -1777,18 +1777,38 @@ void rocksdb_writebatch_wi_rollback_to_save_point(rocksdb_writebatch_wi_t* b,
rocksdb_iterator_t* rocksdb_writebatch_wi_create_iterator_with_base( rocksdb_iterator_t* rocksdb_writebatch_wi_create_iterator_with_base(
rocksdb_writebatch_wi_t* wbwi, rocksdb_writebatch_wi_t* wbwi,
rocksdb_iterator_t* base_iterator) { rocksdb_iterator_t* base_iterator) {
rocksdb_readoptions_t options;
return rocksdb_writebatch_wi_create_iterator_with_base_and_readoptions(
&options, wbwi, base_iterator);
}
rocksdb_iterator_t* rocksdb_writebatch_wi_create_iterator_with_base_cf(
rocksdb_writebatch_wi_t* wbwi, rocksdb_iterator_t* base_iterator,
rocksdb_column_family_handle_t* column_family) {
rocksdb_readoptions_t options;
return rocksdb_writebatch_wi_create_iterator_with_base_cf_and_readoptions(
&options, wbwi, base_iterator, column_family);
}
rocksdb_iterator_t*
rocksdb_writebatch_wi_create_iterator_with_base_and_readoptions(
const rocksdb_readoptions_t* options, rocksdb_writebatch_wi_t* wbwi,
rocksdb_iterator_t* base_iterator) {
rocksdb_iterator_t* result = new rocksdb_iterator_t; rocksdb_iterator_t* result = new rocksdb_iterator_t;
result->rep = wbwi->rep->NewIteratorWithBase(base_iterator->rep); result->rep =
wbwi->rep->NewIteratorWithBase(options->rep, base_iterator->rep);
delete base_iterator; delete base_iterator;
return result; return result;
} }
rocksdb_iterator_t* rocksdb_writebatch_wi_create_iterator_with_base_cf( rocksdb_iterator_t*
rocksdb_writebatch_wi_t* wbwi, rocksdb_writebatch_wi_create_iterator_with_base_cf_and_readoptions(
const rocksdb_readoptions_t* options, rocksdb_writebatch_wi_t* wbwi,
rocksdb_iterator_t* base_iterator, rocksdb_iterator_t* base_iterator,
rocksdb_column_family_handle_t* column_family) { rocksdb_column_family_handle_t* column_family) {
rocksdb_iterator_t* result = new rocksdb_iterator_t; rocksdb_iterator_t* result = new rocksdb_iterator_t;
result->rep = wbwi->rep->NewIteratorWithBase(column_family->rep, base_iterator->rep); result->rep = wbwi->rep->NewIteratorWithBase(options->rep, column_family->rep,
base_iterator->rep);
delete base_iterator; delete base_iterator;
return result; return result;
} }

@ -917,7 +917,9 @@ int main(int argc, char** argv) {
rocksdb_writebatch_wi_t* wbi = rocksdb_writebatch_wi_create(0, 1); rocksdb_writebatch_wi_t* wbi = rocksdb_writebatch_wi_create(0, 1);
rocksdb_writebatch_wi_put(wbi, "bar", 3, "b", 1); rocksdb_writebatch_wi_put(wbi, "bar", 3, "b", 1);
rocksdb_writebatch_wi_delete(wbi, "foo", 3); rocksdb_writebatch_wi_delete(wbi, "foo", 3);
rocksdb_iterator_t* iter = rocksdb_writebatch_wi_create_iterator_with_base(wbi, base_iter); rocksdb_iterator_t* iter =
rocksdb_writebatch_wi_create_iterator_with_base_and_readoptions(
roptions, wbi, base_iter);
CheckCondition(!rocksdb_iter_valid(iter)); CheckCondition(!rocksdb_iter_valid(iter));
rocksdb_iter_seek_to_first(iter); rocksdb_iter_seek_to_first(iter);
CheckCondition(rocksdb_iter_valid(iter)); CheckCondition(rocksdb_iter_valid(iter));
@ -1643,6 +1645,7 @@ int main(int argc, char** argv) {
// Check iterator with column family // Check iterator with column family
rocksdb_transaction_put_cf(txn, cfh1, "key1_cf", 7, "val1_cf", 7, &err); rocksdb_transaction_put_cf(txn, cfh1, "key1_cf", 7, "val1_cf", 7, &err);
CheckNoError(err); CheckNoError(err);
rocksdb_readoptions_set_iterate_upper_bound(roptions, "key2", 4);
rocksdb_iterator_t* iter = rocksdb_iterator_t* iter =
rocksdb_transaction_create_iterator_cf(txn, roptions, cfh1); rocksdb_transaction_create_iterator_cf(txn, roptions, cfh1);
CheckCondition(!rocksdb_iter_valid(iter)); CheckCondition(!rocksdb_iter_valid(iter));

@ -636,7 +636,14 @@ extern ROCKSDB_LIBRARY_API rocksdb_iterator_t* rocksdb_writebatch_wi_create_iter
rocksdb_writebatch_wi_t* wbwi, rocksdb_writebatch_wi_t* wbwi,
rocksdb_iterator_t* base_iterator, rocksdb_iterator_t* base_iterator,
rocksdb_column_family_handle_t* cf); rocksdb_column_family_handle_t* cf);
extern ROCKSDB_LIBRARY_API rocksdb_iterator_t*
rocksdb_writebatch_wi_create_iterator_with_base_and_readoptions(
const rocksdb_readoptions_t* options, rocksdb_writebatch_wi_t* wbwi,
rocksdb_iterator_t* base_iterator);
extern ROCKSDB_LIBRARY_API rocksdb_iterator_t*
rocksdb_writebatch_wi_create_iterator_with_base_cf_and_readoptions(
const rocksdb_readoptions_t* options, rocksdb_writebatch_wi_t* wbwi,
rocksdb_iterator_t* base_iterator, rocksdb_column_family_handle_t* cf);
/* Block based table options */ /* Block based table options */

@ -161,9 +161,15 @@ class WriteBatchWithIndex : public WriteBatchBase {
// key() and value() of the iterator. This invalidation happens even before // key() and value() of the iterator. This invalidation happens even before
// the write batch update finishes. The state may recover after Next() is // the write batch update finishes. The state may recover after Next() is
// called. // called.
Iterator* NewIteratorWithBase(const ReadOptions& read_options,
ColumnFamilyHandle* column_family,
Iterator* base_iterator);
Iterator* NewIteratorWithBase(ColumnFamilyHandle* column_family, Iterator* NewIteratorWithBase(ColumnFamilyHandle* column_family,
Iterator* base_iterator); Iterator* base_iterator);
// default column family // default column family
Iterator* NewIteratorWithBase(const ReadOptions& read_options,
Iterator* base_iterator);
Iterator* NewIteratorWithBase(Iterator* base_iterator); Iterator* NewIteratorWithBase(Iterator* base_iterator);
// Similar to DB::Get() but will only read the key from this batch. // Similar to DB::Get() but will only read the key from this batch.

@ -457,11 +457,14 @@ jlong Java_org_rocksdb_WriteBatchWithIndex_iteratorWithBase(JNIEnv* /*env*/,
jobject /*jobj*/, jobject /*jobj*/,
jlong jwbwi_handle, jlong jwbwi_handle,
jlong jcf_handle, jlong jcf_handle,
jlong jbi_handle) { jlong jbi_handle,
jlong jreadopt_handle) {
auto* wbwi = reinterpret_cast<rocksdb::WriteBatchWithIndex*>(jwbwi_handle); auto* wbwi = reinterpret_cast<rocksdb::WriteBatchWithIndex*>(jwbwi_handle);
auto* cf_handle = reinterpret_cast<rocksdb::ColumnFamilyHandle*>(jcf_handle); auto* cf_handle = reinterpret_cast<rocksdb::ColumnFamilyHandle*>(jcf_handle);
auto* base_iterator = reinterpret_cast<rocksdb::Iterator*>(jbi_handle); auto* base_iterator = reinterpret_cast<rocksdb::Iterator*>(jbi_handle);
auto* iterator = wbwi->NewIteratorWithBase(cf_handle, base_iterator); auto* readopt = reinterpret_cast<rocksdb::ReadOptions*>(jreadopt_handle);
auto* iterator =
wbwi->NewIteratorWithBase(*readopt, cf_handle, base_iterator);
return reinterpret_cast<jlong>(iterator); return reinterpret_cast<jlong>(iterator);
} }

@ -14,8 +14,9 @@ package org.rocksdb;
* *
* A user can call {@link org.rocksdb.WriteBatchWithIndex#newIterator()} to * A user can call {@link org.rocksdb.WriteBatchWithIndex#newIterator()} to
* create an iterator over the write batch or * create an iterator over the write batch or
* {@link org.rocksdb.WriteBatchWithIndex#newIteratorWithBase(org.rocksdb.RocksIterator)} * {@link org.rocksdb.WriteBatchWithIndex#newIteratorWithBase(org.rocksdb.ReadOptions, org.rocksdb.RocksIterator)}
* to get an iterator for the database with Read-Your-Own-Writes like capability * to get an iterator for the database with Read-Your-Own-Writes like
* capability
*/ */
public class WriteBatchWithIndex extends AbstractWriteBatch { public class WriteBatchWithIndex extends AbstractWriteBatch {
/** /**
@ -120,25 +121,65 @@ public class WriteBatchWithIndex extends AbstractWriteBatch {
* the write batch update finishes. The state may recover after Next() is * the write batch update finishes. The state may recover after Next() is
* called. * called.
* *
* @param read_opts The read options to use
* @param columnFamilyHandle The column family to iterate over * @param columnFamilyHandle The column family to iterate over
* @param baseIterator The base iterator, * @param baseIterator The base iterator,
* e.g. {@link org.rocksdb.RocksDB#newIterator()} * e.g. {@link org.rocksdb.RocksDB#newIterator()}
* @return An iterator which shows a view comprised of both the database * @return An iterator which shows a view comprised of both the database
* point-in-time from baseIterator and modifications made in this write batch. * point-in-time from baseIterator and modifications made in this write batch.
*/ */
public RocksIterator newIteratorWithBase( public RocksIterator newIteratorWithBase(final ReadOptions read_opts,
final ColumnFamilyHandle columnFamilyHandle, final ColumnFamilyHandle columnFamilyHandle, final RocksIterator baseIterator) {
final RocksIterator baseIterator) { RocksIterator iterator = new RocksIterator(baseIterator.parent_,
RocksIterator iterator = new RocksIterator( iteratorWithBase(nativeHandle_, columnFamilyHandle.nativeHandle_,
baseIterator.parent_, baseIterator.nativeHandle_, read_opts.nativeHandle_));
iteratorWithBase(nativeHandle_,
columnFamilyHandle.nativeHandle_,
baseIterator.nativeHandle_));
//when the iterator is deleted it will also delete the baseIterator //when the iterator is deleted it will also delete the baseIterator
baseIterator.disOwnNativeHandle(); baseIterator.disOwnNativeHandle();
return iterator; return iterator;
} }
/**
* Provides Read-Your-Own-Writes like functionality by
* creating a new Iterator that will use {@link org.rocksdb.WBWIRocksIterator}
* as a delta and baseIterator as a base
*
* Updating write batch with the current key of the iterator is not safe.
* We strongly recommand users not to do it. It will invalidate the current
* key() and value() of the iterator. This invalidation happens even before
* the write batch update finishes. The state may recover after Next() is
* called.
*
* @param columnFamilyHandle The column family to iterate over
* @param baseIterator The base iterator,
* e.g. {@link org.rocksdb.RocksDB#newIterator()}
* @return An iterator which shows a view comprised of both the database
* point-in-time from baseIterator and modifications made in this write batch.
*/
public RocksIterator newIteratorWithBase(
final ColumnFamilyHandle columnFamilyHandle,
final RocksIterator baseIterator) {
ReadOptions read_opts = new ReadOptions();
return newIteratorWithBase(read_opts, columnFamilyHandle, baseIterator);
}
/**
* Provides Read-Your-Own-Writes like functionality by
* creating a new Iterator that will use {@link org.rocksdb.WBWIRocksIterator}
* as a delta and baseIterator as a base. Operates on the default column
* family.
*
* @param read_opts The read options to use
* @param baseIterator The base iterator,
* e.g. {@link org.rocksdb.RocksDB#newIterator()}
* @return An iterator which shows a view comprised of both the database
* point-in-timefrom baseIterator and modifications made in this write batch.
*/
public RocksIterator newIteratorWithBase(
final ReadOptions read_opts, final RocksIterator baseIterator) {
return newIteratorWithBase(
read_opts, baseIterator.parent_.getDefaultColumnFamily(), baseIterator);
}
/** /**
* Provides Read-Your-Own-Writes like functionality by * Provides Read-Your-Own-Writes like functionality by
* creating a new Iterator that will use {@link org.rocksdb.WBWIRocksIterator} * creating a new Iterator that will use {@link org.rocksdb.WBWIRocksIterator}
@ -151,8 +192,8 @@ public class WriteBatchWithIndex extends AbstractWriteBatch {
* point-in-timefrom baseIterator and modifications made in this write batch. * point-in-timefrom baseIterator and modifications made in this write batch.
*/ */
public RocksIterator newIteratorWithBase(final RocksIterator baseIterator) { public RocksIterator newIteratorWithBase(final RocksIterator baseIterator) {
return newIteratorWithBase(baseIterator.parent_.getDefaultColumnFamily(), ReadOptions read_opts = new ReadOptions();
baseIterator); return newIteratorWithBase(read_opts, baseIterator);
} }
/** /**
@ -295,8 +336,8 @@ public class WriteBatchWithIndex extends AbstractWriteBatch {
final boolean overwriteKey); final boolean overwriteKey);
private native long iterator0(final long handle); private native long iterator0(final long handle);
private native long iterator1(final long handle, final long cfHandle); private native long iterator1(final long handle, final long cfHandle);
private native long iteratorWithBase(final long handle, private native long iteratorWithBase(final long handle, final long baseIteratorHandle,
final long baseIteratorHandle, final long cfHandle); final long cfHandle, final long jreadopt_handle);
private native byte[] getFromBatch(final long handle, final long optHandle, private native byte[] getFromBatch(final long handle, final long optHandle,
final byte[] key, final int keyLen); final byte[] key, final int keyLen);
private native byte[] getFromBatch(final long handle, final long optHandle, private native byte[] getFromBatch(final long handle, final long optHandle,

@ -47,7 +47,6 @@ public class WriteBatchWithIndexTest {
try (final WriteBatchWithIndex wbwi = new WriteBatchWithIndex(true); try (final WriteBatchWithIndex wbwi = new WriteBatchWithIndex(true);
final RocksIterator base = db.newIterator(); final RocksIterator base = db.newIterator();
final RocksIterator it = wbwi.newIteratorWithBase(base)) { final RocksIterator it = wbwi.newIteratorWithBase(base)) {
it.seek(k1); it.seek(k1);
assertThat(it.isValid()).isTrue(); assertThat(it.isValid()).isTrue();
assertThat(it.key()).isEqualTo(k1); assertThat(it.key()).isEqualTo(k1);
@ -421,8 +420,8 @@ public class WriteBatchWithIndexTest {
final ReadOptions readOptions, final WriteBatchWithIndex wbwi, final ReadOptions readOptions, final WriteBatchWithIndex wbwi,
final String skey) { final String skey) {
final byte[] key = skey.getBytes(); final byte[] key = skey.getBytes();
try(final RocksIterator baseIterator = db.newIterator(readOptions); try (final RocksIterator baseIterator = db.newIterator(readOptions);
final RocksIterator iterator = wbwi.newIteratorWithBase(baseIterator)) { final RocksIterator iterator = wbwi.newIteratorWithBase(baseIterator)) {
iterator.seek(key); iterator.seek(key);
// Arrays.equals(key, iterator.key()) ensures an exact match in Rocks, // Arrays.equals(key, iterator.key()) ensures an exact match in Rocks,

@ -852,6 +852,51 @@ TEST_F(OptimisticTransactionTest, UntrackedWrites) {
delete txn; delete txn;
} }
TEST_F(OptimisticTransactionTest, IteratorUpperBoundTest) {
WriteOptions write_options;
ReadOptions read_options;
auto txn = unique_ptr<Transaction>(txn_db->BeginTransaction(write_options));
string key1 = "a1";
string key2 = "a3";
string key3 = "b1";
string val = "123";
txn->Put(key1, val);
txn->Put(key2, val);
Status s = txn->Commit();
ASSERT_OK(s);
txn = unique_ptr<Transaction>(txn_db->BeginTransaction(write_options));
txn->Put(key3, val);
string ubKey("a2");
Slice upperbound(ubKey);
read_options.iterate_upper_bound = &upperbound;
auto it = unique_ptr<Iterator>(txn->GetIterator(read_options));
for (it->SeekToFirst(); it->Valid(); it->Next()) {
EXPECT_LT(it->key().ToString(), ubKey);
}
EXPECT_GE(it->key().ToString(), ubKey);
int key_count = 0;
for (it->SeekToFirst(); it->Valid(); it->Next()) {
EXPECT_LT(it->key().ToString(), ubKey);
key_count++;
}
ASSERT_EQ(key_count, 1);
// Test Seek to a key equal or over upper bound
it->Seek("a2");
ASSERT_FALSE(it->Valid());
it->Seek("a3");
ASSERT_FALSE(it->Valid());
it->Seek("a1");
ASSERT_TRUE(it->Valid());
it.reset();
s = txn->Commit();
ASSERT_OK(s);
txn.reset();
}
TEST_F(OptimisticTransactionTest, IteratorTest) { TEST_F(OptimisticTransactionTest, IteratorTest) {
WriteOptions write_options; WriteOptions write_options;
ReadOptions read_options, snapshot_read_options; ReadOptions read_options, snapshot_read_options;

@ -291,7 +291,7 @@ Iterator* TransactionBaseImpl::GetIterator(const ReadOptions& read_options) {
Iterator* db_iter = db_->NewIterator(read_options); Iterator* db_iter = db_->NewIterator(read_options);
assert(db_iter); assert(db_iter);
return write_batch_.NewIteratorWithBase(db_iter); return write_batch_.NewIteratorWithBase(read_options, db_iter);
} }
Iterator* TransactionBaseImpl::GetIterator(const ReadOptions& read_options, Iterator* TransactionBaseImpl::GetIterator(const ReadOptions& read_options,
@ -299,7 +299,7 @@ Iterator* TransactionBaseImpl::GetIterator(const ReadOptions& read_options,
Iterator* db_iter = db_->NewIterator(read_options, column_family); Iterator* db_iter = db_->NewIterator(read_options, column_family);
assert(db_iter); assert(db_iter);
return write_batch_.NewIteratorWithBase(column_family, db_iter); return write_batch_.NewIteratorWithBase(read_options, column_family, db_iter);
} }
Status TransactionBaseImpl::Put(ColumnFamilyHandle* column_family, Status TransactionBaseImpl::Put(ColumnFamilyHandle* column_family,

@ -62,7 +62,7 @@ Iterator* WritePreparedTxn::GetIterator(const ReadOptions& options) {
Iterator* db_iter = wpt_db_->NewIterator(options); Iterator* db_iter = wpt_db_->NewIterator(options);
assert(db_iter); assert(db_iter);
return write_batch_.NewIteratorWithBase(db_iter); return write_batch_.NewIteratorWithBase(options, db_iter);
} }
Iterator* WritePreparedTxn::GetIterator(const ReadOptions& options, Iterator* WritePreparedTxn::GetIterator(const ReadOptions& options,
@ -71,7 +71,7 @@ Iterator* WritePreparedTxn::GetIterator(const ReadOptions& options,
Iterator* db_iter = wpt_db_->NewIterator(options, column_family); Iterator* db_iter = wpt_db_->NewIterator(options, column_family);
assert(db_iter); assert(db_iter);
return write_batch_.NewIteratorWithBase(column_family, db_iter); return write_batch_.NewIteratorWithBase(options, column_family, db_iter);
} }
Status WritePreparedTxn::PrepareInternal() { Status WritePreparedTxn::PrepareInternal() {

@ -506,7 +506,7 @@ Iterator* WriteUnpreparedTxn::GetIterator(const ReadOptions& options,
Iterator* db_iter = wupt_db_->NewIterator(options, column_family, this); Iterator* db_iter = wupt_db_->NewIterator(options, column_family, this);
assert(db_iter); assert(db_iter);
return write_batch_.NewIteratorWithBase(column_family, db_iter); return write_batch_.NewIteratorWithBase(options, column_family, db_iter);
} }
const std::map<SequenceNumber, size_t>& const std::map<SequenceNumber, size_t>&

@ -32,11 +32,13 @@ namespace rocksdb {
// * equal_keys_ <=> base_iterator == delta_iterator // * equal_keys_ <=> base_iterator == delta_iterator
class BaseDeltaIterator : public Iterator { class BaseDeltaIterator : public Iterator {
public: public:
BaseDeltaIterator(Iterator* base_iterator, WBWIIterator* delta_iterator, BaseDeltaIterator(const ReadOptions& read_options, Iterator* base_iterator,
const Comparator* comparator) WBWIIterator* delta_iterator, const Comparator* comparator)
: forward_(true), : read_options_(read_options),
forward_(true),
current_at_base_(true), current_at_base_(true),
equal_keys_(false), equal_keys_(false),
current_over_upper_bound_(false),
status_(Status::OK()), status_(Status::OK()),
base_iterator_(base_iterator), base_iterator_(base_iterator),
delta_iterator_(delta_iterator), delta_iterator_(delta_iterator),
@ -45,7 +47,9 @@ class BaseDeltaIterator : public Iterator {
virtual ~BaseDeltaIterator() {} virtual ~BaseDeltaIterator() {}
bool Valid() const override { bool Valid() const override {
return current_at_base_ ? BaseValid() : DeltaValid(); return current_over_upper_bound_
? false
: (current_at_base_ ? BaseValid() : DeltaValid());
} }
void SeekToFirst() override { void SeekToFirst() override {
@ -216,9 +220,15 @@ class BaseDeltaIterator : public Iterator {
} }
// equal_keys_ <=> compare == 0 // equal_keys_ <=> compare == 0
assert((equal_keys_ || compare != 0) && (!equal_keys_ || compare == 0)); assert((equal_keys_ || compare != 0) && (!equal_keys_ || compare == 0));
#endif #endif
} }
bool IsOverUpperBound() {
return read_options_.iterate_upper_bound != nullptr &&
comparator_->Compare(key(), *read_options_.iterate_upper_bound) >= 0;
}
void Advance() { void Advance() {
if (equal_keys_) { if (equal_keys_) {
assert(BaseValid() && DeltaValid()); assert(BaseValid() && DeltaValid());
@ -264,32 +274,32 @@ class BaseDeltaIterator : public Iterator {
} else if (!delta_iterator_->status().ok()) { } else if (!delta_iterator_->status().ok()) {
// Expose the error status and stop. // Expose the error status and stop.
current_at_base_ = false; current_at_base_ = false;
return; break;
} }
equal_keys_ = false; equal_keys_ = false;
if (!BaseValid()) { if (!BaseValid()) {
if (!base_iterator_->status().ok()) { if (!base_iterator_->status().ok()) {
// Expose the error status and stop. // Expose the error status and stop.
current_at_base_ = true; current_at_base_ = true;
return; break;
} }
// Base has finished. // Base has finished.
if (!DeltaValid()) { if (!DeltaValid()) {
// Finished // Finished
return; break;
} }
if (delta_entry.type == kDeleteRecord || if (delta_entry.type == kDeleteRecord ||
delta_entry.type == kSingleDeleteRecord) { delta_entry.type == kSingleDeleteRecord) {
AdvanceDelta(); AdvanceDelta();
} else { } else {
current_at_base_ = false; current_at_base_ = false;
return; break;
} }
} else if (!DeltaValid()) { } else if (!DeltaValid()) {
// Delta has finished. // Delta has finished.
current_at_base_ = true; current_at_base_ = true;
return; break;
} else { } else {
int compare = int compare =
(forward_ ? 1 : -1) * (forward_ ? 1 : -1) *
@ -301,7 +311,7 @@ class BaseDeltaIterator : public Iterator {
if (delta_entry.type != kDeleteRecord && if (delta_entry.type != kDeleteRecord &&
delta_entry.type != kSingleDeleteRecord) { delta_entry.type != kSingleDeleteRecord) {
current_at_base_ = false; current_at_base_ = false;
return; break;
} }
// Delta is less advanced and is delete. // Delta is less advanced and is delete.
AdvanceDelta(); AdvanceDelta();
@ -310,18 +320,24 @@ class BaseDeltaIterator : public Iterator {
} }
} else { } else {
current_at_base_ = true; current_at_base_ = true;
return; break;
} }
} }
current_over_upper_bound_ = IsOverUpperBound();
if (current_over_upper_bound_) {
break;
}
} }
AssertInvariants(); current_over_upper_bound_ = IsOverUpperBound();
#endif // __clang_analyzer__ #endif // __clang_analyzer__
} }
ReadOptions read_options_;
bool forward_; bool forward_;
bool current_at_base_; bool current_at_base_;
bool equal_keys_; bool equal_keys_;
bool current_over_upper_bound_;
Status status_; Status status_;
std::unique_ptr<Iterator> base_iterator_; std::unique_ptr<Iterator> base_iterator_;
std::unique_ptr<WBWIIterator> delta_iterator_; std::unique_ptr<WBWIIterator> delta_iterator_;
@ -642,25 +658,39 @@ WBWIIterator* WriteBatchWithIndex::NewIterator(
} }
Iterator* WriteBatchWithIndex::NewIteratorWithBase( Iterator* WriteBatchWithIndex::NewIteratorWithBase(
ColumnFamilyHandle* column_family, Iterator* base_iterator) { const ReadOptions& read_options, ColumnFamilyHandle* column_family,
Iterator* base_iterator) {
if (rep->overwrite_key == false) { if (rep->overwrite_key == false) {
assert(false); assert(false);
return nullptr; return nullptr;
} }
return new BaseDeltaIterator(base_iterator, NewIterator(column_family), return new BaseDeltaIterator(read_options, base_iterator,
NewIterator(column_family),
GetColumnFamilyUserComparator(column_family)); GetColumnFamilyUserComparator(column_family));
} }
Iterator* WriteBatchWithIndex::NewIteratorWithBase(Iterator* base_iterator) { Iterator* WriteBatchWithIndex::NewIteratorWithBase(
ColumnFamilyHandle* column_family, Iterator* base_iterator) {
ReadOptions read_options;
return NewIteratorWithBase(read_options, column_family, base_iterator);
}
Iterator* WriteBatchWithIndex::NewIteratorWithBase(
const ReadOptions& read_options, Iterator* base_iterator) {
if (rep->overwrite_key == false) { if (rep->overwrite_key == false) {
assert(false); assert(false);
return nullptr; return nullptr;
} }
// default column family's comparator // default column family's comparator
return new BaseDeltaIterator(base_iterator, NewIterator(), return new BaseDeltaIterator(read_options, base_iterator, NewIterator(),
rep->comparator.default_comparator()); rep->comparator.default_comparator());
} }
Iterator* WriteBatchWithIndex::NewIteratorWithBase(Iterator* base_iterator) {
ReadOptions read_options;
return NewIteratorWithBase(read_options, base_iterator);
}
Status WriteBatchWithIndex::Put(ColumnFamilyHandle* column_family, Status WriteBatchWithIndex::Put(ColumnFamilyHandle* column_family,
const Slice& key, const Slice& value) { const Slice& key, const Slice& value) {
rep->SetLastEntryOffset(); rep->SetLastEntryOffset();

Loading…
Cancel
Save