blob: 7eaa89405cb23782b5c070ca97947a64edb9f3ef [file] [log] [blame]
// Copyright (c) 2013 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "content/browser/indexed_db/leveldb/leveldb_database.h"
#include <inttypes.h>
#include <stdint.h>
#include <cerrno>
#include <memory>
#include <utility>
#include "base/files/file.h"
#include "base/logging.h"
#include "base/macros.h"
#include "base/memory/ptr_util.h"
#include "base/metrics/histogram_macros.h"
#include "base/strings/string16.h"
#include "base/strings/string_number_conversions.h"
#include "base/strings/string_piece.h"
#include "base/strings/stringprintf.h"
#include "base/strings/utf_string_conversions.h"
#include "base/sys_info.h"
#include "base/trace_event/memory_dump_manager.h"
#include "base/trace_event/process_memory_dump.h"
#include "build/build_config.h"
#include "content/browser/indexed_db/indexed_db_class_factory.h"
#include "content/browser/indexed_db/indexed_db_tracing.h"
#include "content/browser/indexed_db/leveldb/leveldb_comparator.h"
#include "content/browser/indexed_db/leveldb/leveldb_env.h"
#include "content/browser/indexed_db/leveldb/leveldb_iterator_impl.h"
#include "content/browser/indexed_db/leveldb/leveldb_write_batch.h"
#include "third_party/leveldatabase/env_chromium.h"
#include "third_party/leveldatabase/src/helpers/memenv/memenv.h"
#include "third_party/leveldatabase/src/include/leveldb/db.h"
#include "third_party/leveldatabase/src/include/leveldb/env.h"
#include "third_party/leveldatabase/src/include/leveldb/filter_policy.h"
#include "third_party/leveldatabase/src/include/leveldb/slice.h"
using base::StringPiece;
namespace content {
// Forcing flushes to disk at the end of a transaction guarantees that the
// data hit disk, but drastically impacts throughput when the filesystem is
// busy with background compactions. Not syncing trades off reliability for
// performance. Note that background compactions which move data from the
// log to SSTs are always done with reliable writes.
//
// Sync writes are necessary on Windows for quota calculations; POSIX
// calculates file sizes correctly even when not synced to disk.
#if defined(OS_WIN)
static const bool kSyncWrites = true;
#else
// TODO(dgrogan): Either remove the #if block or change this back to false.
// See http://crbug.com/338385.
static const bool kSyncWrites = true;
#endif
static leveldb::Slice MakeSlice(const StringPiece& s) {
return leveldb::Slice(s.begin(), s.size());
}
static StringPiece MakeStringPiece(const leveldb::Slice& s) {
return StringPiece(s.data(), s.size());
}
LevelDBDatabase::ComparatorAdapter::ComparatorAdapter(
const LevelDBComparator* comparator)
: comparator_(comparator) {}
int LevelDBDatabase::ComparatorAdapter::Compare(const leveldb::Slice& a,
const leveldb::Slice& b) const {
return comparator_->Compare(MakeStringPiece(a), MakeStringPiece(b));
}
const char* LevelDBDatabase::ComparatorAdapter::Name() const {
return comparator_->Name();
}
// TODO(jsbell): Support the methods below in the future.
void LevelDBDatabase::ComparatorAdapter::FindShortestSeparator(
std::string* start,
const leveldb::Slice& limit) const {}
void LevelDBDatabase::ComparatorAdapter::FindShortSuccessor(
std::string* key) const {}
LevelDBSnapshot::LevelDBSnapshot(LevelDBDatabase* db)
: db_(db->db_.get()), snapshot_(db_->GetSnapshot()) {}
LevelDBSnapshot::~LevelDBSnapshot() { db_->ReleaseSnapshot(snapshot_); }
LevelDBDatabase::LevelDBDatabase() {}
LevelDBDatabase::~LevelDBDatabase() {
base::trace_event::MemoryDumpManager::GetInstance()->UnregisterDumpProvider(
this);
// db_'s destructor uses comparator_adapter_; order of deletion is important.
CloseDatabase();
comparator_adapter_.reset();
env_.reset();
}
void LevelDBDatabase::CloseDatabase() {
if (db_) {
base::TimeTicks begin_time = base::TimeTicks::Now();
db_.reset();
UMA_HISTOGRAM_MEDIUM_TIMES("WebCore.IndexedDB.LevelDB.CloseTime",
base::TimeTicks::Now() - begin_time);
}
}
static leveldb::Status OpenDB(
leveldb::Comparator* comparator,
leveldb::Env* env,
const base::FilePath& path,
std::unique_ptr<leveldb::DB>* db,
std::unique_ptr<const leveldb::FilterPolicy>* filter_policy) {
filter_policy->reset(leveldb::NewBloomFilterPolicy(10));
leveldb::Options options;
options.comparator = comparator;
options.create_if_missing = true;
options.paranoid_checks = true;
options.filter_policy = filter_policy->get();
options.reuse_logs = leveldb_env::kDefaultLogReuseOptionValue;
options.compression = leveldb::kSnappyCompression;
options.write_buffer_size =
leveldb_env::WriteBufferSize(base::SysInfo::AmountOfTotalDiskSpace(path));
// For info about the troubles we've run into with this parameter, see:
// https://code.google.com/p/chromium/issues/detail?id=227313#c11
options.max_open_files = 80;
options.env = env;
// ChromiumEnv assumes UTF8, converts back to FilePath before using.
leveldb::DB* db_ptr = nullptr;
leveldb::Status s = leveldb::DB::Open(options, path.AsUTF8Unsafe(), &db_ptr);
db->reset(db_ptr);
return s;
}
leveldb::Status LevelDBDatabase::Destroy(const base::FilePath& file_name) {
leveldb::Options options;
options.env = LevelDBEnv::Get();
// ChromiumEnv assumes UTF8, converts back to FilePath before using.
return leveldb::DestroyDB(file_name.AsUTF8Unsafe(), options);
}
namespace {
class LockImpl : public LevelDBLock {
public:
explicit LockImpl(leveldb::Env* env, leveldb::FileLock* lock)
: env_(env), lock_(lock) {}
~LockImpl() override { env_->UnlockFile(lock_); }
private:
leveldb::Env* env_;
leveldb::FileLock* lock_;
DISALLOW_COPY_AND_ASSIGN(LockImpl);
};
} // namespace
std::unique_ptr<LevelDBLock> LevelDBDatabase::LockForTesting(
const base::FilePath& file_name) {
leveldb::Env* env = LevelDBEnv::Get();
base::FilePath lock_path = file_name.AppendASCII("LOCK");
leveldb::FileLock* lock = NULL;
leveldb::Status status = env->LockFile(lock_path.AsUTF8Unsafe(), &lock);
if (!status.ok())
return std::unique_ptr<LevelDBLock>();
DCHECK(lock);
return base::MakeUnique<LockImpl>(env, lock);
}
static int CheckFreeSpace(const char* const type,
const base::FilePath& file_name) {
std::string name =
std::string("WebCore.IndexedDB.LevelDB.Open") + type + "FreeDiskSpace";
int64_t free_disk_space_in_k_bytes =
base::SysInfo::AmountOfFreeDiskSpace(file_name) / 1024;
if (free_disk_space_in_k_bytes < 0) {
base::Histogram::FactoryGet(
"WebCore.IndexedDB.LevelDB.FreeDiskSpaceFailure",
1,
2 /*boundary*/,
2 /*boundary*/ + 1,
base::HistogramBase::kUmaTargetedHistogramFlag)->Add(1 /*sample*/);
return -1;
}
int clamped_disk_space_k_bytes = free_disk_space_in_k_bytes > INT_MAX
? INT_MAX
: free_disk_space_in_k_bytes;
const uint64_t histogram_max = static_cast<uint64_t>(1e9);
static_assert(histogram_max <= INT_MAX, "histogram_max too big");
base::Histogram::FactoryGet(name,
1,
histogram_max,
11 /*buckets*/,
base::HistogramBase::kUmaTargetedHistogramFlag)
->Add(clamped_disk_space_k_bytes);
return clamped_disk_space_k_bytes;
}
static void ParseAndHistogramIOErrorDetails(const std::string& histogram_name,
const leveldb::Status& s) {
leveldb_env::MethodID method;
base::File::Error error = base::File::FILE_OK;
leveldb_env::ErrorParsingResult result =
leveldb_env::ParseMethodAndError(s, &method, &error);
if (result == leveldb_env::NONE)
return;
std::string method_histogram_name(histogram_name);
method_histogram_name.append(".EnvMethod");
base::LinearHistogram::FactoryGet(
method_histogram_name,
1,
leveldb_env::kNumEntries,
leveldb_env::kNumEntries + 1,
base::HistogramBase::kUmaTargetedHistogramFlag)->Add(method);
std::string error_histogram_name(histogram_name);
if (result == leveldb_env::METHOD_AND_BFE) {
DCHECK_LT(error, 0);
error_histogram_name.append(std::string(".BFE.") +
leveldb_env::MethodIDToString(method));
base::LinearHistogram::FactoryGet(
error_histogram_name,
1,
-base::File::FILE_ERROR_MAX,
-base::File::FILE_ERROR_MAX + 1,
base::HistogramBase::kUmaTargetedHistogramFlag)->Add(-error);
}
}
static void ParseAndHistogramCorruptionDetails(
const std::string& histogram_name,
const leveldb::Status& status) {
int error = leveldb_env::GetCorruptionCode(status);
DCHECK_GE(error, 0);
std::string corruption_histogram_name(histogram_name);
corruption_histogram_name.append(".Corruption");
const int kNumPatterns = leveldb_env::GetNumCorruptionCodes();
base::LinearHistogram::FactoryGet(
corruption_histogram_name,
1,
kNumPatterns,
kNumPatterns + 1,
base::HistogramBase::kUmaTargetedHistogramFlag)->Add(error);
}
static void HistogramLevelDBError(const std::string& histogram_name,
const leveldb::Status& s) {
if (s.ok()) {
NOTREACHED();
return;
}
enum {
LEVEL_DB_NOT_FOUND,
LEVEL_DB_CORRUPTION,
LEVEL_DB_IO_ERROR,
LEVEL_DB_OTHER,
LEVEL_DB_MAX_ERROR
};
int leveldb_error = LEVEL_DB_OTHER;
if (s.IsNotFound())
leveldb_error = LEVEL_DB_NOT_FOUND;
else if (s.IsCorruption())
leveldb_error = LEVEL_DB_CORRUPTION;
else if (s.IsIOError())
leveldb_error = LEVEL_DB_IO_ERROR;
base::Histogram::FactoryGet(histogram_name,
1,
LEVEL_DB_MAX_ERROR,
LEVEL_DB_MAX_ERROR + 1,
base::HistogramBase::kUmaTargetedHistogramFlag)
->Add(leveldb_error);
if (s.IsIOError())
ParseAndHistogramIOErrorDetails(histogram_name, s);
else
ParseAndHistogramCorruptionDetails(histogram_name, s);
}
leveldb::Status LevelDBDatabase::Open(const base::FilePath& file_name,
const LevelDBComparator* comparator,
std::unique_ptr<LevelDBDatabase>* result,
bool* is_disk_full) {
IDB_TRACE("LevelDBDatabase::Open");
base::TimeTicks begin_time = base::TimeTicks::Now();
std::unique_ptr<ComparatorAdapter> comparator_adapter(
base::MakeUnique<ComparatorAdapter>(comparator));
std::unique_ptr<leveldb::DB> db;
std::unique_ptr<const leveldb::FilterPolicy> filter_policy;
const leveldb::Status s = OpenDB(comparator_adapter.get(), LevelDBEnv::Get(),
file_name, &db, &filter_policy);
if (!s.ok()) {
HistogramLevelDBError("WebCore.IndexedDB.LevelDBOpenErrors", s);
int free_space_k_bytes = CheckFreeSpace("Failure", file_name);
// Disks with <100k of free space almost never succeed in opening a
// leveldb database.
if (is_disk_full)
*is_disk_full = free_space_k_bytes >= 0 && free_space_k_bytes < 100;
LOG(ERROR) << "Failed to open LevelDB database from "
<< file_name.AsUTF8Unsafe() << "," << s.ToString();
return s;
}
UMA_HISTOGRAM_MEDIUM_TIMES("WebCore.IndexedDB.LevelDB.OpenTime",
base::TimeTicks::Now() - begin_time);
CheckFreeSpace("Success", file_name);
(*result) = base::WrapUnique(new LevelDBDatabase());
(*result)->db_ = std::move(db);
(*result)->comparator_adapter_ = std::move(comparator_adapter);
(*result)->comparator_ = comparator;
(*result)->filter_policy_ = std::move(filter_policy);
(*result)->file_name_for_tracing = file_name.BaseName().AsUTF8Unsafe();
return s;
}
std::unique_ptr<LevelDBDatabase> LevelDBDatabase::OpenInMemory(
const LevelDBComparator* comparator) {
std::unique_ptr<ComparatorAdapter> comparator_adapter(
base::MakeUnique<ComparatorAdapter>(comparator));
std::unique_ptr<leveldb::Env> in_memory_env(
leveldb::NewMemEnv(LevelDBEnv::Get()));
std::unique_ptr<leveldb::DB> db;
std::unique_ptr<const leveldb::FilterPolicy> filter_policy;
const leveldb::Status s = OpenDB(comparator_adapter.get(),
in_memory_env.get(),
base::FilePath(),
&db,
&filter_policy);
if (!s.ok()) {
LOG(ERROR) << "Failed to open in-memory LevelDB database: " << s.ToString();
return std::unique_ptr<LevelDBDatabase>();
}
std::unique_ptr<LevelDBDatabase> result =
base::WrapUnique(new LevelDBDatabase());
result->env_ = std::move(in_memory_env);
result->db_ = std::move(db);
result->comparator_adapter_ = std::move(comparator_adapter);
result->comparator_ = comparator;
result->filter_policy_ = std::move(filter_policy);
result->file_name_for_tracing = "in-memory-database";
return result;
}
leveldb::Status LevelDBDatabase::Put(const StringPiece& key,
std::string* value) {
base::TimeTicks begin_time = base::TimeTicks::Now();
leveldb::WriteOptions write_options;
write_options.sync = kSyncWrites;
const leveldb::Status s =
db_->Put(write_options, MakeSlice(key), MakeSlice(*value));
if (!s.ok())
LOG(ERROR) << "LevelDB put failed: " << s.ToString();
else
UMA_HISTOGRAM_TIMES("WebCore.IndexedDB.LevelDB.PutTime",
base::TimeTicks::Now() - begin_time);
return s;
}
leveldb::Status LevelDBDatabase::Remove(const StringPiece& key) {
leveldb::WriteOptions write_options;
write_options.sync = kSyncWrites;
const leveldb::Status s = db_->Delete(write_options, MakeSlice(key));
if (!s.IsNotFound())
LOG(ERROR) << "LevelDB remove failed: " << s.ToString();
return s;
}
leveldb::Status LevelDBDatabase::Get(const StringPiece& key,
std::string* value,
bool* found,
const LevelDBSnapshot* snapshot) {
*found = false;
leveldb::ReadOptions read_options;
read_options.verify_checksums = true; // TODO(jsbell): Disable this if the
// performance impact is too great.
read_options.snapshot = snapshot ? snapshot->snapshot_ : 0;
const leveldb::Status s = db_->Get(read_options, MakeSlice(key), value);
if (s.ok()) {
*found = true;
return s;
}
if (s.IsNotFound())
return leveldb::Status::OK();
HistogramLevelDBError("WebCore.IndexedDB.LevelDBReadErrors", s);
LOG(ERROR) << "LevelDB get failed: " << s.ToString();
return s;
}
leveldb::Status LevelDBDatabase::Write(const LevelDBWriteBatch& write_batch) {
base::TimeTicks begin_time = base::TimeTicks::Now();
leveldb::WriteOptions write_options;
write_options.sync = kSyncWrites;
const leveldb::Status s =
db_->Write(write_options, write_batch.write_batch_.get());
if (!s.ok()) {
HistogramLevelDBError("WebCore.IndexedDB.LevelDBWriteErrors", s);
LOG(ERROR) << "LevelDB write failed: " << s.ToString();
} else {
UMA_HISTOGRAM_TIMES("WebCore.IndexedDB.LevelDB.WriteTime",
base::TimeTicks::Now() - begin_time);
}
return s;
}
std::unique_ptr<LevelDBIterator> LevelDBDatabase::CreateIterator(
const LevelDBSnapshot* snapshot) {
leveldb::ReadOptions read_options;
read_options.verify_checksums = true; // TODO(jsbell): Disable this if the
// performance impact is too great.
read_options.snapshot = snapshot ? snapshot->snapshot_ : 0;
std::unique_ptr<leveldb::Iterator> i(db_->NewIterator(read_options));
return std::unique_ptr<LevelDBIterator>(
IndexedDBClassFactory::Get()->CreateIteratorImpl(std::move(i)));
}
const LevelDBComparator* LevelDBDatabase::Comparator() const {
return comparator_;
}
void LevelDBDatabase::Compact(const base::StringPiece& start,
const base::StringPiece& stop) {
IDB_TRACE("LevelDBDatabase::Compact");
const leveldb::Slice start_slice = MakeSlice(start);
const leveldb::Slice stop_slice = MakeSlice(stop);
// NULL batch means just wait for earlier writes to be done
db_->Write(leveldb::WriteOptions(), NULL);
db_->CompactRange(&start_slice, &stop_slice);
}
void LevelDBDatabase::CompactAll() { db_->CompactRange(NULL, NULL); }
bool LevelDBDatabase::OnMemoryDump(
const base::trace_event::MemoryDumpArgs& args,
base::trace_event::ProcessMemoryDump* pmd) {
if (!db_)
return false;
std::string value;
uint64_t size;
bool res = db_->GetProperty("leveldb.approximate-memory-usage", &value);
DCHECK(res);
base::StringToUint64(value, &size);
auto* dump = pmd->CreateAllocatorDump(base::StringPrintf(
"leveldb/index_db/0x%" PRIXPTR, reinterpret_cast<uintptr_t>(db_.get())));
dump->AddScalar(base::trace_event::MemoryAllocatorDump::kNameSize,
base::trace_event::MemoryAllocatorDump::kUnitsBytes, size);
dump->AddString("file_name", "", file_name_for_tracing);
// Memory is allocated from system allocator (malloc).
pmd->AddSuballocation(dump->guid(),
base::trace_event::MemoryDumpManager::GetInstance()
->system_allocator_pool_name());
return true;
}
} // namespace content