2016-02-09 15:12:00 -08:00
|
|
|
// Copyright (c) 2011-present, Facebook, Inc. All rights reserved.
|
2017-07-15 16:03:42 -07:00
|
|
|
// This source code is licensed under both the GPLv2 (found in the
|
|
|
|
// COPYING file in the root directory) and Apache 2.0 License
|
|
|
|
// (found in the LICENSE.Apache file in the root directory).
|
2012-11-05 19:18:49 -08:00
|
|
|
|
|
|
|
#include "db/db_impl_readonly.h"
|
2015-07-13 18:10:31 -07:00
|
|
|
|
|
|
|
#include "db/compacted_db_impl.h"
|
2012-11-05 19:18:49 -08:00
|
|
|
#include "db/db_impl.h"
|
2014-09-25 11:14:01 -07:00
|
|
|
#include "db/db_iter.h"
|
2016-11-03 18:40:23 -07:00
|
|
|
#include "db/merge_context.h"
|
|
|
|
#include "db/range_del_aggregator.h"
|
2017-04-05 19:02:00 -07:00
|
|
|
#include "monitoring/perf_context_imp.h"
|
2012-11-05 19:18:49 -08:00
|
|
|
|
2013-10-03 21:49:15 -07:00
|
|
|
namespace rocksdb {
|
2012-11-05 19:18:49 -08:00
|
|
|
|
2014-11-26 11:37:59 -08:00
|
|
|
#ifndef ROCKSDB_LITE
|
|
|
|
|
2014-09-08 18:46:52 -07:00
|
|
|
DBImplReadOnly::DBImplReadOnly(const DBOptions& db_options,
|
2014-02-05 13:12:23 -08:00
|
|
|
const std::string& dbname)
|
2014-09-08 18:46:52 -07:00
|
|
|
: DBImpl(db_options, dbname) {
|
2017-03-15 19:22:52 -07:00
|
|
|
ROCKS_LOG_INFO(immutable_db_options_.info_log,
|
|
|
|
"Opening the db in read only mode");
|
2016-09-23 16:34:04 -07:00
|
|
|
LogFlush(immutable_db_options_.info_log);
|
2012-11-05 19:18:49 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
DBImplReadOnly::~DBImplReadOnly() {
|
|
|
|
}
|
|
|
|
|
|
|
|
// Implementations of the DB interface
|
2014-09-08 18:46:52 -07:00
|
|
|
Status DBImplReadOnly::Get(const ReadOptions& read_options,
|
2014-02-10 17:04:44 -08:00
|
|
|
ColumnFamilyHandle* column_family, const Slice& key,
|
2017-03-13 11:44:50 -07:00
|
|
|
PinnableSlice* pinnable_val) {
|
|
|
|
assert(pinnable_val != nullptr);
|
2012-11-05 19:18:49 -08:00
|
|
|
Status s;
|
|
|
|
SequenceNumber snapshot = versions_->LastSequence();
|
2014-02-10 17:04:44 -08:00
|
|
|
auto cfh = reinterpret_cast<ColumnFamilyHandleImpl*>(column_family);
|
|
|
|
auto cfd = cfh->cfd();
|
2014-02-03 15:28:03 -08:00
|
|
|
SuperVersion* super_version = cfd->GetSuperVersion();
|
2013-12-02 18:34:05 -08:00
|
|
|
MergeContext merge_context;
|
2016-11-18 16:54:09 -08:00
|
|
|
RangeDelAggregator range_del_agg(cfd->internal_comparator(), snapshot);
|
2012-11-05 19:18:49 -08:00
|
|
|
LookupKey lkey(key, snapshot);
|
2017-03-13 11:44:50 -07:00
|
|
|
if (super_version->mem->Get(lkey, pinnable_val->GetSelf(), &s, &merge_context,
|
|
|
|
&range_del_agg, read_options)) {
|
|
|
|
pinnable_val->PinSelf();
|
2013-02-15 15:28:24 -08:00
|
|
|
} else {
|
2014-10-02 17:02:30 -07:00
|
|
|
PERF_TIMER_GUARD(get_from_output_files_time);
|
2017-03-13 11:44:50 -07:00
|
|
|
super_version->current->Get(read_options, lkey, pinnable_val, &s,
|
|
|
|
&merge_context, &range_del_agg);
|
2013-02-15 15:28:24 -08:00
|
|
|
}
|
2012-11-05 19:18:49 -08:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2014-09-08 15:04:34 -07:00
|
|
|
Iterator* DBImplReadOnly::NewIterator(const ReadOptions& read_options,
|
2014-02-10 17:04:44 -08:00
|
|
|
ColumnFamilyHandle* column_family) {
|
|
|
|
auto cfh = reinterpret_cast<ColumnFamilyHandleImpl*>(column_family);
|
|
|
|
auto cfd = cfh->cfd();
|
2014-02-03 15:28:03 -08:00
|
|
|
SuperVersion* super_version = cfd->GetSuperVersion()->Ref();
|
|
|
|
SequenceNumber latest_snapshot = versions_->LastSequence();
|
2014-07-23 16:52:11 -04:00
|
|
|
auto db_iter = NewArenaWrappedDbIterator(
|
2017-07-24 10:28:17 -07:00
|
|
|
env_, read_options, *cfd->ioptions(),
|
2014-09-08 15:04:34 -07:00
|
|
|
(read_options.snapshot != nullptr
|
2016-02-29 18:38:03 -08:00
|
|
|
? reinterpret_cast<const SnapshotImpl*>(read_options.snapshot)
|
|
|
|
->number_
|
2014-09-08 15:04:34 -07:00
|
|
|
: latest_snapshot),
|
2016-02-29 18:38:03 -08:00
|
|
|
super_version->mutable_cf_options.max_sequential_skip_in_iterations,
|
2017-04-10 11:13:23 -07:00
|
|
|
super_version->version_number);
|
2016-11-04 11:53:38 -07:00
|
|
|
auto internal_iter =
|
|
|
|
NewInternalIterator(read_options, cfd, super_version, db_iter->GetArena(),
|
|
|
|
db_iter->GetRangeDelAggregator());
|
2014-07-23 16:52:11 -04:00
|
|
|
db_iter->SetIterUnderDBIter(internal_iter);
|
|
|
|
return db_iter;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status DBImplReadOnly::NewIterators(
|
2014-09-08 15:04:34 -07:00
|
|
|
const ReadOptions& read_options,
|
2014-07-23 16:52:11 -04:00
|
|
|
const std::vector<ColumnFamilyHandle*>& column_families,
|
|
|
|
std::vector<Iterator*>* iterators) {
|
|
|
|
if (iterators == nullptr) {
|
|
|
|
return Status::InvalidArgument("iterators not allowed to be nullptr");
|
|
|
|
}
|
|
|
|
iterators->clear();
|
|
|
|
iterators->reserve(column_families.size());
|
|
|
|
SequenceNumber latest_snapshot = versions_->LastSequence();
|
|
|
|
|
|
|
|
for (auto cfh : column_families) {
|
2014-10-23 15:34:21 -07:00
|
|
|
auto* cfd = reinterpret_cast<ColumnFamilyHandleImpl*>(cfh)->cfd();
|
|
|
|
auto* sv = cfd->GetSuperVersion()->Ref();
|
|
|
|
auto* db_iter = NewArenaWrappedDbIterator(
|
2017-07-24 10:28:17 -07:00
|
|
|
env_, read_options, *cfd->ioptions(),
|
2014-09-08 15:04:34 -07:00
|
|
|
(read_options.snapshot != nullptr
|
2016-02-29 18:38:03 -08:00
|
|
|
? reinterpret_cast<const SnapshotImpl*>(read_options.snapshot)
|
|
|
|
->number_
|
|
|
|
: latest_snapshot),
|
|
|
|
sv->mutable_cf_options.max_sequential_skip_in_iterations,
|
2017-04-10 11:13:23 -07:00
|
|
|
sv->version_number);
|
2016-11-04 11:53:38 -07:00
|
|
|
auto* internal_iter =
|
|
|
|
NewInternalIterator(read_options, cfd, sv, db_iter->GetArena(),
|
|
|
|
db_iter->GetRangeDelAggregator());
|
2014-07-23 16:52:11 -04:00
|
|
|
db_iter->SetIterUnderDBIter(internal_iter);
|
|
|
|
iterators->push_back(db_iter);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status::OK();
|
2012-11-05 19:18:49 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
Status DB::OpenForReadOnly(const Options& options, const std::string& dbname,
|
2017-07-21 18:13:59 -07:00
|
|
|
DB** dbptr, bool error_if_log_file_exist) {
|
2013-02-15 15:28:24 -08:00
|
|
|
*dbptr = nullptr;
|
2012-11-05 19:18:49 -08:00
|
|
|
|
2014-09-25 11:14:01 -07:00
|
|
|
// Try to first open DB as fully compacted DB
|
|
|
|
Status s;
|
|
|
|
s = CompactedDBImpl::Open(options, dbname, dbptr);
|
|
|
|
if (s.ok()) {
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2014-01-06 13:31:06 -08:00
|
|
|
DBOptions db_options(options);
|
|
|
|
ColumnFamilyOptions cf_options(options);
|
|
|
|
std::vector<ColumnFamilyDescriptor> column_families;
|
|
|
|
column_families.push_back(
|
2014-04-09 09:56:17 -07:00
|
|
|
ColumnFamilyDescriptor(kDefaultColumnFamilyName, cf_options));
|
|
|
|
std::vector<ColumnFamilyHandle*> handles;
|
|
|
|
|
2014-09-25 11:14:01 -07:00
|
|
|
s = DB::OpenForReadOnly(db_options, dbname, column_families, &handles, dbptr);
|
2014-04-09 09:56:17 -07:00
|
|
|
if (s.ok()) {
|
|
|
|
assert(handles.size() == 1);
|
|
|
|
// i can delete the handle since DBImpl is always holding a
|
|
|
|
// reference to default column family
|
|
|
|
delete handles[0];
|
|
|
|
}
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status DB::OpenForReadOnly(
|
|
|
|
const DBOptions& db_options, const std::string& dbname,
|
|
|
|
const std::vector<ColumnFamilyDescriptor>& column_families,
|
|
|
|
std::vector<ColumnFamilyHandle*>* handles, DB** dbptr,
|
|
|
|
bool error_if_log_file_exist) {
|
|
|
|
*dbptr = nullptr;
|
|
|
|
handles->clear();
|
2014-02-05 13:12:23 -08:00
|
|
|
|
|
|
|
DBImplReadOnly* impl = new DBImplReadOnly(db_options, dbname);
|
|
|
|
impl->mutex_.Lock();
|
2014-01-24 14:30:28 -08:00
|
|
|
Status s = impl->Recover(column_families, true /* read only */,
|
|
|
|
error_if_log_file_exist);
|
2014-04-09 09:56:17 -07:00
|
|
|
if (s.ok()) {
|
|
|
|
// set column family handles
|
|
|
|
for (auto cf : column_families) {
|
|
|
|
auto cfd =
|
|
|
|
impl->versions_->GetColumnFamilySet()->GetColumnFamily(cf.name);
|
|
|
|
if (cfd == nullptr) {
|
|
|
|
s = Status::InvalidArgument("Column family not found: ", cf.name);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
handles->push_back(new ColumnFamilyHandleImpl(cfd, impl, &impl->mutex_));
|
|
|
|
}
|
|
|
|
}
|
2014-02-03 13:13:36 -08:00
|
|
|
if (s.ok()) {
|
2014-02-03 13:44:47 -08:00
|
|
|
for (auto cfd : *impl->versions_->GetColumnFamilySet()) {
|
2014-03-03 17:54:04 -08:00
|
|
|
delete cfd->InstallSuperVersion(new SuperVersion(), &impl->mutex_);
|
2014-02-03 13:44:47 -08:00
|
|
|
}
|
2014-02-03 13:13:36 -08:00
|
|
|
}
|
2012-11-05 19:18:49 -08:00
|
|
|
impl->mutex_.Unlock();
|
|
|
|
if (s.ok()) {
|
|
|
|
*dbptr = impl;
|
2014-11-20 10:49:32 -08:00
|
|
|
for (auto* h : *handles) {
|
|
|
|
impl->NewThreadStatusCfInfo(
|
|
|
|
reinterpret_cast<ColumnFamilyHandleImpl*>(h)->cfd());
|
|
|
|
}
|
2012-11-05 19:18:49 -08:00
|
|
|
} else {
|
2014-04-09 09:56:17 -07:00
|
|
|
for (auto h : *handles) {
|
|
|
|
delete h;
|
|
|
|
}
|
|
|
|
handles->clear();
|
2012-11-05 19:18:49 -08:00
|
|
|
delete impl;
|
|
|
|
}
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2014-11-26 11:37:59 -08:00
|
|
|
#else // !ROCKSDB_LITE
|
|
|
|
|
|
|
|
Status DB::OpenForReadOnly(const Options& options, const std::string& dbname,
|
|
|
|
DB** dbptr, bool error_if_log_file_exist) {
|
|
|
|
return Status::NotSupported("Not supported in ROCKSDB_LITE.");
|
|
|
|
}
|
|
|
|
|
|
|
|
Status DB::OpenForReadOnly(
|
|
|
|
const DBOptions& db_options, const std::string& dbname,
|
|
|
|
const std::vector<ColumnFamilyDescriptor>& column_families,
|
|
|
|
std::vector<ColumnFamilyHandle*>* handles, DB** dbptr,
|
|
|
|
bool error_if_log_file_exist) {
|
|
|
|
return Status::NotSupported("Not supported in ROCKSDB_LITE.");
|
|
|
|
}
|
|
|
|
#endif // !ROCKSDB_LITE
|
2014-04-09 09:56:17 -07:00
|
|
|
|
2014-03-20 13:42:45 -07:00
|
|
|
} // namespace rocksdb
|