2016-02-09 23:12:00 +00:00
|
|
|
// Copyright (c) 2011-present, Facebook, Inc. All rights reserved.
|
2017-07-15 23:03:42 +00: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).
|
2013-10-16 21:59:46 +00:00
|
|
|
//
|
2012-04-17 15:36:46 +00:00
|
|
|
// Copyright (c) 2012 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.
|
|
|
|
//
|
|
|
|
// A filter block is stored near the end of a Table file. It contains
|
|
|
|
// filters (e.g., bloom filters) for all data blocks in the table combined
|
|
|
|
// into a single filter block.
|
2014-09-08 17:37:05 +00:00
|
|
|
//
|
|
|
|
// It is a base class for BlockBasedFilter and FullFilter.
|
|
|
|
// These two are both used in BlockBasedTable. The first one contain filter
|
|
|
|
// For a part of keys in sst file, the second contain filter for all keys
|
|
|
|
// in sst file.
|
2012-04-17 15:36:46 +00:00
|
|
|
|
2013-10-05 05:32:05 +00:00
|
|
|
#pragma once
|
2013-11-13 06:46:51 +00:00
|
|
|
|
2014-08-15 22:05:09 +00:00
|
|
|
#include <memory>
|
2012-04-17 15:36:46 +00:00
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
2014-09-08 17:37:05 +00:00
|
|
|
#include "rocksdb/options.h"
|
2013-08-23 15:38:13 +00:00
|
|
|
#include "rocksdb/slice.h"
|
2014-08-15 22:05:09 +00:00
|
|
|
#include "rocksdb/slice_transform.h"
|
2014-08-25 21:22:05 +00:00
|
|
|
#include "rocksdb/table.h"
|
2014-08-15 22:05:09 +00:00
|
|
|
#include "util/hash.h"
|
|
|
|
#include "format.h"
|
2012-04-17 15:36:46 +00:00
|
|
|
|
2013-10-04 04:49:15 +00:00
|
|
|
namespace rocksdb {
|
2012-04-17 15:36:46 +00:00
|
|
|
|
2014-09-08 17:37:05 +00:00
|
|
|
const uint64_t kNotValid = ULLONG_MAX;
|
2014-08-15 22:05:09 +00:00
|
|
|
class FilterPolicy;
|
2012-04-17 15:36:46 +00:00
|
|
|
|
|
|
|
// A FilterBlockBuilder is used to construct all of the filters for a
|
|
|
|
// particular Table. It generates a single string which is stored as
|
|
|
|
// a special block in the Table.
|
|
|
|
//
|
|
|
|
// The sequence of calls to FilterBlockBuilder must match the regexp:
|
2014-09-08 17:37:05 +00:00
|
|
|
// (StartBlock Add*)* Finish
|
|
|
|
//
|
|
|
|
// BlockBased/Full FilterBlock would be called in the same way.
|
2012-04-17 15:36:46 +00:00
|
|
|
class FilterBlockBuilder {
|
|
|
|
public:
|
2014-09-08 17:37:05 +00:00
|
|
|
explicit FilterBlockBuilder() {}
|
|
|
|
virtual ~FilterBlockBuilder() {}
|
2012-04-17 15:36:46 +00:00
|
|
|
|
2014-09-08 17:37:05 +00:00
|
|
|
virtual bool IsBlockBased() = 0; // If is blockbased filter
|
|
|
|
virtual void StartBlock(uint64_t block_offset) = 0; // Start new block filter
|
|
|
|
virtual void Add(const Slice& key) = 0; // Add a key to current filter
|
2018-03-22 05:56:48 +00:00
|
|
|
virtual size_t NumAdded() const = 0; // Number of keys added
|
2017-03-07 21:48:02 +00:00
|
|
|
Slice Finish() { // Generate Filter
|
|
|
|
const BlockHandle empty_handle;
|
|
|
|
Status dont_care_status;
|
|
|
|
auto ret = Finish(empty_handle, &dont_care_status);
|
|
|
|
assert(dont_care_status.ok());
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
virtual Slice Finish(const BlockHandle& tmp, Status* status) = 0;
|
2012-04-17 15:36:46 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
// No copying allowed
|
|
|
|
FilterBlockBuilder(const FilterBlockBuilder&);
|
|
|
|
void operator=(const FilterBlockBuilder&);
|
|
|
|
};
|
|
|
|
|
2014-09-08 17:37:05 +00:00
|
|
|
// A FilterBlockReader is used to parse filter from SST table.
|
|
|
|
// KeyMayMatch and PrefixMayMatch would trigger filter checking
|
|
|
|
//
|
|
|
|
// BlockBased/Full FilterBlock would be called in the same way.
|
2012-04-17 15:36:46 +00:00
|
|
|
class FilterBlockReader {
|
|
|
|
public:
|
2016-06-09 22:48:45 +00:00
|
|
|
explicit FilterBlockReader()
|
|
|
|
: whole_key_filtering_(true), size_(0), statistics_(nullptr) {}
|
|
|
|
explicit FilterBlockReader(size_t s, Statistics* stats,
|
|
|
|
bool _whole_key_filtering)
|
|
|
|
: whole_key_filtering_(_whole_key_filtering),
|
|
|
|
size_(s),
|
|
|
|
statistics_(stats) {}
|
2014-09-08 17:37:05 +00:00
|
|
|
virtual ~FilterBlockReader() {}
|
2013-11-13 06:46:51 +00:00
|
|
|
|
2014-09-08 17:37:05 +00:00
|
|
|
virtual bool IsBlockBased() = 0; // If is blockbased filter
|
2017-03-22 16:11:23 +00:00
|
|
|
/**
|
|
|
|
* If no_io is set, then it returns true if it cannot answer the query without
|
|
|
|
* reading data from disk. This is used in PartitionedFilterBlockReader to
|
|
|
|
* avoid reading partitions that are not in block cache already
|
|
|
|
*
|
|
|
|
* Normally filters are built on only the user keys and the InternalKey is not
|
|
|
|
* needed for a query. The index in PartitionedFilterBlockReader however is
|
|
|
|
* built upon InternalKey and must be provided via const_ikey_ptr when running
|
|
|
|
* queries.
|
|
|
|
*/
|
|
|
|
virtual bool KeyMayMatch(const Slice& key, uint64_t block_offset = kNotValid,
|
|
|
|
const bool no_io = false,
|
|
|
|
const Slice* const const_ikey_ptr = nullptr) = 0;
|
|
|
|
/**
|
|
|
|
* no_io and const_ikey_ptr here means the same as in KeyMayMatch
|
|
|
|
*/
|
2014-09-08 17:37:05 +00:00
|
|
|
virtual bool PrefixMayMatch(const Slice& prefix,
|
2017-03-22 16:11:23 +00:00
|
|
|
uint64_t block_offset = kNotValid,
|
|
|
|
const bool no_io = false,
|
|
|
|
const Slice* const const_ikey_ptr = nullptr) = 0;
|
2014-09-08 17:37:05 +00:00
|
|
|
virtual size_t ApproximateMemoryUsage() const = 0;
|
2016-06-03 17:47:47 +00:00
|
|
|
virtual size_t size() const { return size_; }
|
|
|
|
virtual Statistics* statistics() const { return statistics_; }
|
2013-08-13 21:04:56 +00:00
|
|
|
|
2016-06-09 22:48:45 +00:00
|
|
|
bool whole_key_filtering() const { return whole_key_filtering_; }
|
|
|
|
|
2014-12-23 21:24:07 +00:00
|
|
|
// convert this object to a human readable form
|
|
|
|
virtual std::string ToString() const {
|
|
|
|
std::string error_msg("Unsupported filter \n");
|
|
|
|
return error_msg;
|
|
|
|
}
|
|
|
|
|
2018-03-05 21:08:17 +00:00
|
|
|
virtual void CacheDependencies(bool /*pin*/) {}
|
2017-08-23 14:48:54 +00:00
|
|
|
|
2016-06-09 22:48:45 +00:00
|
|
|
protected:
|
|
|
|
bool whole_key_filtering_;
|
|
|
|
|
2014-09-08 17:37:05 +00:00
|
|
|
private:
|
|
|
|
// No copying allowed
|
|
|
|
FilterBlockReader(const FilterBlockReader&);
|
|
|
|
void operator=(const FilterBlockReader&);
|
2016-06-03 17:47:47 +00:00
|
|
|
size_t size_;
|
|
|
|
Statistics* statistics_;
|
2017-03-22 16:11:23 +00:00
|
|
|
int level_ = -1;
|
2012-04-17 15:36:46 +00:00
|
|
|
};
|
|
|
|
|
2014-09-08 17:37:05 +00:00
|
|
|
} // namespace rocksdb
|