2016-02-09 15:12:00 -08:00
|
|
|
// Copyright (c) 2011-present, Facebook, Inc. All rights reserved.
|
2013-10-16 14:59:46 -07:00
|
|
|
// 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.
|
|
|
|
//
|
2013-08-13 14:04:56 -07: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.
|
|
|
|
|
2015-07-01 16:13:49 -07:00
|
|
|
#include <algorithm>
|
2013-08-23 08:38:13 -07:00
|
|
|
#include "rocksdb/slice_transform.h"
|
|
|
|
#include "rocksdb/slice.h"
|
2014-11-24 20:44:49 -08:00
|
|
|
#include "util/string_util.h"
|
2015-07-07 16:58:20 -07:00
|
|
|
#include <stdio.h>
|
2013-08-13 14:04:56 -07:00
|
|
|
|
2013-10-03 21:49:15 -07:00
|
|
|
namespace rocksdb {
|
2013-08-13 14:04:56 -07:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
class FixedPrefixTransform : public SliceTransform {
|
|
|
|
private:
|
|
|
|
size_t prefix_len_;
|
2014-05-13 14:42:31 -07:00
|
|
|
std::string name_;
|
2013-08-13 14:04:56 -07:00
|
|
|
|
|
|
|
public:
|
2014-05-13 14:42:31 -07:00
|
|
|
explicit FixedPrefixTransform(size_t prefix_len)
|
|
|
|
: prefix_len_(prefix_len),
|
2015-10-02 15:35:32 -07:00
|
|
|
// 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.
|
2014-11-24 20:44:49 -08:00
|
|
|
name_("rocksdb.FixedPrefix." + ToString(prefix_len_)) {}
|
2013-08-13 14:04:56 -07:00
|
|
|
|
2015-02-26 11:28:41 -08:00
|
|
|
virtual const char* Name() const override { return name_.c_str(); }
|
2013-08-13 14:04:56 -07:00
|
|
|
|
2015-02-26 11:28:41 -08:00
|
|
|
virtual Slice Transform(const Slice& src) const override {
|
2013-08-13 14:04:56 -07:00
|
|
|
assert(InDomain(src));
|
|
|
|
return Slice(src.data(), prefix_len_);
|
|
|
|
}
|
|
|
|
|
2015-02-26 11:28:41 -08:00
|
|
|
virtual bool InDomain(const Slice& src) const override {
|
2013-08-13 14:04:56 -07:00
|
|
|
return (src.size() >= prefix_len_);
|
|
|
|
}
|
|
|
|
|
2015-02-26 11:28:41 -08:00
|
|
|
virtual bool InRange(const Slice& dst) const override {
|
2013-08-13 14:04:56 -07:00
|
|
|
return (dst.size() == prefix_len_);
|
|
|
|
}
|
2015-01-21 11:09:56 -08:00
|
|
|
|
2015-02-26 11:28:41 -08:00
|
|
|
virtual bool SameResultWhenAppended(const Slice& prefix) const override {
|
2015-01-21 11:09:56 -08:00
|
|
|
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),
|
2015-10-02 15:35:32 -07:00
|
|
|
// 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.
|
2015-01-21 11:09:56 -08:00
|
|
|
name_("rocksdb.CappedPrefix." + ToString(cap_len_)) {}
|
|
|
|
|
2015-02-26 11:28:41 -08:00
|
|
|
virtual const char* Name() const override { return name_.c_str(); }
|
2015-01-21 11:09:56 -08:00
|
|
|
|
2015-02-26 11:28:41 -08:00
|
|
|
virtual Slice Transform(const Slice& src) const override {
|
2015-01-21 11:09:56 -08:00
|
|
|
assert(InDomain(src));
|
|
|
|
return Slice(src.data(), std::min(cap_len_, src.size()));
|
|
|
|
}
|
|
|
|
|
2015-02-26 11:28:41 -08:00
|
|
|
virtual bool InDomain(const Slice& src) const override { return true; }
|
2015-01-21 11:09:56 -08:00
|
|
|
|
2015-02-26 11:28:41 -08:00
|
|
|
virtual bool InRange(const Slice& dst) const override {
|
2015-01-21 11:09:56 -08:00
|
|
|
return (dst.size() <= cap_len_);
|
|
|
|
}
|
|
|
|
|
2015-02-26 11:28:41 -08:00
|
|
|
virtual bool SameResultWhenAppended(const Slice& prefix) const override {
|
2015-01-21 11:09:56 -08:00
|
|
|
return prefix.size() >= cap_len_;
|
|
|
|
}
|
2013-08-13 14:04:56 -07:00
|
|
|
};
|
2013-08-22 23:10:02 -07:00
|
|
|
|
|
|
|
class NoopTransform : public SliceTransform {
|
|
|
|
public:
|
|
|
|
explicit NoopTransform() { }
|
|
|
|
|
2015-02-26 11:28:41 -08:00
|
|
|
virtual const char* Name() const override { return "rocksdb.Noop"; }
|
2013-08-22 23:10:02 -07:00
|
|
|
|
2015-02-26 11:28:41 -08:00
|
|
|
virtual Slice Transform(const Slice& src) const override { return src; }
|
2013-08-22 23:10:02 -07:00
|
|
|
|
2015-02-26 11:28:41 -08:00
|
|
|
virtual bool InDomain(const Slice& src) const override { return true; }
|
2013-08-22 23:10:02 -07:00
|
|
|
|
2015-02-26 11:28:41 -08:00
|
|
|
virtual bool InRange(const Slice& dst) const override { return true; }
|
2015-01-21 11:09:56 -08:00
|
|
|
|
2015-02-26 11:28:41 -08:00
|
|
|
virtual bool SameResultWhenAppended(const Slice& prefix) const override {
|
2015-01-21 11:09:56 -08:00
|
|
|
return false;
|
|
|
|
}
|
2013-08-22 23:10:02 -07:00
|
|
|
};
|
|
|
|
|
2013-08-13 14:04:56 -07:00
|
|
|
}
|
|
|
|
|
2016-03-29 21:25:12 -07:00
|
|
|
// 2 small internal utility functions, for efficient hex conversions
|
|
|
|
// and no need for snprintf, toupper etc...
|
|
|
|
// Originally from wdt/util/EncryptionUtils.cpp - for ToString(true)/DecodeHex:
|
|
|
|
char toHex(unsigned char v) {
|
|
|
|
if (v <= 9) {
|
|
|
|
return '0' + v;
|
|
|
|
}
|
|
|
|
return 'A' + v - 10;
|
|
|
|
}
|
|
|
|
// most of the code is for validation/error check
|
|
|
|
int fromHex(char c) {
|
|
|
|
// toupper:
|
|
|
|
if (c >= 'a' && c <= 'f') {
|
|
|
|
c -= ('a' - 'A'); // aka 0x20
|
|
|
|
}
|
|
|
|
// validation
|
|
|
|
if (c < '0' || (c > '9' && (c < 'A' || c > 'F'))) {
|
|
|
|
return -1; // invalid not 0-9A-F hex char
|
|
|
|
}
|
|
|
|
if (c <= '9') {
|
|
|
|
return c - '0';
|
|
|
|
}
|
|
|
|
return c - 'A' + 10;
|
|
|
|
}
|
2015-07-07 16:58:20 -07:00
|
|
|
|
|
|
|
// Return a string that contains the copy of the referenced data.
|
|
|
|
std::string Slice::ToString(bool hex) const {
|
2015-07-13 12:11:05 -07:00
|
|
|
std::string result; // RVO/NRVO/move
|
2015-07-07 16:58:20 -07:00
|
|
|
if (hex) {
|
2016-03-29 21:25:12 -07:00
|
|
|
result.reserve(2 * size_);
|
|
|
|
for (size_t i = 0; i < size_; ++i) {
|
|
|
|
unsigned char c = data_[i];
|
|
|
|
result.push_back(toHex(c >> 4));
|
|
|
|
result.push_back(toHex(c & 0xf));
|
2015-07-07 16:58:20 -07:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
} else {
|
|
|
|
result.assign(data_, size_);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-29 21:25:12 -07:00
|
|
|
// Originally from tools/ldb_cmd.h
|
|
|
|
bool Slice::DecodeHex(std::string* result) const {
|
|
|
|
std::string::size_type len = size_;
|
|
|
|
if (len % 2) {
|
|
|
|
// Hex string must be even number of hex digits to get complete bytes back
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!result) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
result->clear();
|
|
|
|
result->reserve(len / 2);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < len;) {
|
|
|
|
int h1 = fromHex(data_[i++]);
|
|
|
|
if (h1 < 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
int h2 = fromHex(data_[i++]);
|
|
|
|
if (h2 < 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
result->push_back((h1 << 4) | h2);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-08-13 14:04:56 -07:00
|
|
|
const SliceTransform* NewFixedPrefixTransform(size_t prefix_len) {
|
|
|
|
return new FixedPrefixTransform(prefix_len);
|
|
|
|
}
|
|
|
|
|
2015-01-21 11:09:56 -08:00
|
|
|
const SliceTransform* NewCappedPrefixTransform(size_t cap_len) {
|
|
|
|
return new CappedPrefixTransform(cap_len);
|
|
|
|
}
|
|
|
|
|
2013-08-22 23:10:02 -07:00
|
|
|
const SliceTransform* NewNoopTransform() {
|
|
|
|
return new NoopTransform;
|
|
|
|
}
|
|
|
|
|
2013-10-03 21:49:15 -07:00
|
|
|
} // namespace rocksdb
|