// 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) 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. #include #include "rocksdb/slice_transform.h" #include "rocksdb/slice.h" #include "util/string_util.h" #include namespace rocksdb { namespace { class FixedPrefixTransform : public SliceTransform { private: size_t prefix_len_; std::string name_; public: explicit FixedPrefixTransform(size_t prefix_len) : prefix_len_(prefix_len), // Note that if any part of the name format changes, it will require // changes on options_helper in order to make RocksDBOptionsParser work // for the new change. // TODO(yhchiang): move serialization / deserializaion code inside // the class implementation itself. name_("rocksdb.FixedPrefix." + ToString(prefix_len_)) {} virtual const char* Name() const override { return name_.c_str(); } virtual Slice Transform(const Slice& src) const override { assert(InDomain(src)); return Slice(src.data(), prefix_len_); } virtual bool InDomain(const Slice& src) const override { return (src.size() >= prefix_len_); } virtual bool InRange(const Slice& dst) const override { return (dst.size() == prefix_len_); } virtual bool SameResultWhenAppended(const Slice& prefix) const override { return InDomain(prefix); } }; class CappedPrefixTransform : public SliceTransform { private: size_t cap_len_; std::string name_; public: explicit CappedPrefixTransform(size_t cap_len) : cap_len_(cap_len), // Note that if any part of the name format changes, it will require // changes on options_helper in order to make RocksDBOptionsParser work // for the new change. // TODO(yhchiang): move serialization / deserializaion code inside // the class implementation itself. name_("rocksdb.CappedPrefix." + ToString(cap_len_)) {} virtual const char* Name() const override { return name_.c_str(); } virtual Slice Transform(const Slice& src) const override { assert(InDomain(src)); return Slice(src.data(), std::min(cap_len_, src.size())); } virtual bool InDomain(const Slice& src) const override { return true; } virtual bool InRange(const Slice& dst) const override { return (dst.size() <= cap_len_); } virtual bool SameResultWhenAppended(const Slice& prefix) const override { return prefix.size() >= cap_len_; } }; class NoopTransform : public SliceTransform { public: explicit NoopTransform() { } virtual const char* Name() const override { return "rocksdb.Noop"; } virtual Slice Transform(const Slice& src) const override { return src; } virtual bool InDomain(const Slice& src) const override { return true; } virtual bool InRange(const Slice& dst) const override { return true; } virtual bool SameResultWhenAppended(const Slice& prefix) const override { return false; } }; } // Do not want to include the whole /port/port.h here for one define #ifdef OS_WIN #define snprintf _snprintf #endif // Return a string that contains the copy of the referenced data. std::string Slice::ToString(bool hex) const { std::string result; // RVO/NRVO/move if (hex) { char buf[10]; for (size_t i = 0; i < size_; i++) { snprintf(buf, 10, "%02X", (unsigned char)data_[i]); result += buf; } return result; } else { result.assign(data_, size_); return result; } } const SliceTransform* NewFixedPrefixTransform(size_t prefix_len) { return new FixedPrefixTransform(prefix_len); } const SliceTransform* NewCappedPrefixTransform(size_t cap_len) { return new CappedPrefixTransform(cap_len); } const SliceTransform* NewNoopTransform() { return new NoopTransform; } } // namespace rocksdb