// Copyright (c) 2013, Facebook, Inc. All rights reserved. // This source code is licensed under the BSD-style license found in the // LICENSE file in the root directory of this source tree. An additional grant // of patent rights can be found in the PATENTS file in the same directory. // // Copyright (c) 2011 The LevelDB Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. See the AUTHORS file for names of contributors. #include "db/file_indexer.h" #include #include "rocksdb/comparator.h" #include "db/version_edit.h" namespace rocksdb { FileIndexer::FileIndexer(const uint32_t num_levels, const Comparator* ucmp) : num_levels_(num_levels), ucmp_(ucmp), next_level_index_(num_levels), level_rb_(num_levels, -1) { } uint32_t FileIndexer::NumLevelIndex() { return next_level_index_.size(); } uint32_t FileIndexer::LevelIndexSize(uint32_t level) { return next_level_index_[level].size(); } void FileIndexer::GetNextLevelIndex( const uint32_t level, const uint32_t file_index, const int cmp_smallest, const int cmp_largest, int32_t* left_bound, int32_t* right_bound) { assert(level > 0); // Last level, no hint if (level == num_levels_ - 1) { *left_bound = 0; *right_bound = -1; return; } assert(level < num_levels_ - 1); assert(static_cast(file_index) <= level_rb_[level]); const auto& index = next_level_index_[level][file_index]; if (cmp_smallest < 0) { *left_bound = (level > 0 && file_index > 0) ? next_level_index_[level][file_index - 1].largest_lb : 0; *right_bound = index.smallest_rb; } else if (cmp_smallest == 0) { *left_bound = index.smallest_lb; *right_bound = index.smallest_rb; } else if (cmp_smallest > 0 && cmp_largest < 0) { *left_bound = index.smallest_lb; *right_bound = index.largest_rb; } else if (cmp_largest == 0) { *left_bound = index.largest_lb; *right_bound = index.largest_rb; } else if (cmp_largest > 0) { *left_bound = index.largest_lb; *right_bound = level_rb_[level + 1]; } else { assert(false); } assert(*left_bound >= 0); assert(*left_bound <= *right_bound + 1); assert(*right_bound <= level_rb_[level + 1]); } void FileIndexer::ClearIndex() { for (uint32_t level = 1; level < num_levels_; ++level) { next_level_index_[level].clear(); } } void FileIndexer::UpdateIndex(std::vector* const files) { if (files == nullptr) { return; } // L1 - Ln-1 for (uint32_t level = 1; level < num_levels_ - 1; ++level) { const auto& upper_files = files[level]; const int32_t upper_size = upper_files.size(); const auto& lower_files = files[level + 1]; level_rb_[level] = upper_files.size() - 1; if (upper_size == 0) { continue; } auto& index = next_level_index_[level]; index.resize(upper_size); CalculateLB(upper_files, lower_files, &index, [this](const FileMetaData* a, const FileMetaData* b) -> int { return ucmp_->Compare(a->smallest.user_key(), b->largest.user_key()); }, [](IndexUnit* index, int32_t f_idx) { index->smallest_lb = f_idx; }); CalculateLB(upper_files, lower_files, &index, [this](const FileMetaData* a, const FileMetaData* b) -> int { return ucmp_->Compare(a->largest.user_key(), b->largest.user_key()); }, [](IndexUnit* index, int32_t f_idx) { index->largest_lb = f_idx; }); CalculateRB(upper_files, lower_files, &index, [this](const FileMetaData* a, const FileMetaData* b) -> int { return ucmp_->Compare(a->smallest.user_key(), b->smallest.user_key()); }, [](IndexUnit* index, int32_t f_idx) { index->smallest_rb = f_idx; }); CalculateRB(upper_files, lower_files, &index, [this](const FileMetaData* a, const FileMetaData* b) -> int { return ucmp_->Compare(a->largest.user_key(), b->smallest.user_key()); }, [](IndexUnit* index, int32_t f_idx) { index->largest_rb = f_idx; }); } level_rb_[num_levels_ - 1] = files[num_levels_ - 1].size() - 1; } void FileIndexer::CalculateLB(const std::vector& upper_files, const std::vector& lower_files, std::vector* index, std::function cmp_op, std::function set_index) { const int32_t upper_size = upper_files.size(); const int32_t lower_size = lower_files.size(); int32_t upper_idx = 0; int32_t lower_idx = 0; while (upper_idx < upper_size && lower_idx < lower_size) { int cmp = cmp_op(upper_files[upper_idx], lower_files[lower_idx]); if (cmp == 0) { set_index(&(*index)[upper_idx], lower_idx); ++upper_idx; ++lower_idx; } else if (cmp > 0) { // Lower level's file (largest) is smaller, a key won't hit in that // file. Move to next lower file ++lower_idx; } else { // Lower level's file becomes larger, update the index, and // move to the next upper file set_index(&(*index)[upper_idx], lower_idx); ++upper_idx; } } while (upper_idx < upper_size) { // Lower files are exhausted, that means the remaining upper files are // greater than any lower files. Set the index to be the lower level size. set_index(&(*index)[upper_idx], lower_size); ++upper_idx; } } void FileIndexer::CalculateRB(const std::vector& upper_files, const std::vector& lower_files, std::vector* index, std::function cmp_op, std::function set_index) { const int32_t upper_size = upper_files.size(); const int32_t lower_size = lower_files.size(); int32_t upper_idx = upper_size - 1; int32_t lower_idx = lower_size - 1; while (upper_idx >= 0 && lower_idx >= 0) { int cmp = cmp_op(upper_files[upper_idx], lower_files[lower_idx]); if (cmp == 0) { set_index(&(*index)[upper_idx], lower_idx); --upper_idx; --lower_idx; } else if (cmp < 0) { // Lower level's file (smallest) is larger, a key won't hit in that // file. Move to next lower file. --lower_idx; } else { // Lower level's file becomes smaller, update the index, and move to // the next the upper file set_index(&(*index)[upper_idx], lower_idx); --upper_idx; } } while (upper_idx >= 0) { // Lower files are exhausted, that means the remaining upper files are // smaller than any lower files. Set it to -1. set_index(&(*index)[upper_idx], -1); --upper_idx; } } } // namespace rocksdb