// Copyright (c) 2018 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.
|
|
|
|
#ifndef STORAGE_LEVELDB_PORT_PORT_STDCXX_H_
|
|
#define STORAGE_LEVELDB_PORT_PORT_STDCXX_H_
|
|
|
|
// port/port_config.h availability is automatically detected via __has_include
|
|
// in newer compilers. If LEVELDB_HAS_PORT_CONFIG_H is defined, it overrides the
|
|
// configuration detection.
|
|
#if defined(LEVELDB_HAS_PORT_CONFIG_H)
|
|
|
|
#if LEVELDB_HAS_PORT_CONFIG_H
|
|
#include "port/port_config.h"
|
|
#endif // LEVELDB_HAS_PORT_CONFIG_H
|
|
|
|
#elif defined(__has_include)
|
|
|
|
#if __has_include("port/port_config.h")
|
|
#include "port/port_config.h"
|
|
#endif // __has_include("port/port_config.h")
|
|
|
|
#endif // defined(LEVELDB_HAS_PORT_CONFIG_H)
|
|
|
|
#if HAVE_CRC32C
|
|
#include <crc32c/crc32c.h>
|
|
#endif // HAVE_CRC32C
|
|
#if HAVE_SNAPPY
|
|
#include <snappy.h>
|
|
#endif // HAVE_SNAPPY
|
|
#if HAVE_ZSTD
|
|
#define ZSTD_STATIC_LINKING_ONLY // For ZSTD_compressionParameters.
|
|
#include <zstd.h>
|
|
#endif // HAVE_ZSTD
|
|
|
|
#include <cassert>
|
|
#include <condition_variable> // NOLINT
|
|
#include <cstddef>
|
|
#include <cstdint>
|
|
#include <mutex> // NOLINT
|
|
#include <string>
|
|
|
|
#include "port/thread_annotations.h"
|
|
|
|
namespace leveldb {
|
|
namespace port {
|
|
|
|
class CondVar;
|
|
|
|
// Thinly wraps std::mutex.
|
|
class LOCKABLE Mutex {
|
|
public:
|
|
Mutex() = default;
|
|
~Mutex() = default;
|
|
|
|
Mutex(const Mutex&) = delete;
|
|
Mutex& operator=(const Mutex&) = delete;
|
|
|
|
void Lock() EXCLUSIVE_LOCK_FUNCTION() { mu_.lock(); }
|
|
void Unlock() UNLOCK_FUNCTION() { mu_.unlock(); }
|
|
void AssertHeld() ASSERT_EXCLUSIVE_LOCK() {}
|
|
|
|
private:
|
|
friend class CondVar;
|
|
std::mutex mu_;
|
|
};
|
|
|
|
// Thinly wraps std::condition_variable.
|
|
class CondVar {
|
|
public:
|
|
explicit CondVar(Mutex* mu) : mu_(mu) { assert(mu != nullptr); }
|
|
~CondVar() = default;
|
|
|
|
CondVar(const CondVar&) = delete;
|
|
CondVar& operator=(const CondVar&) = delete;
|
|
|
|
void Wait() {
|
|
std::unique_lock<std::mutex> lock(mu_->mu_, std::adopt_lock);
|
|
cv_.wait(lock);
|
|
lock.release();
|
|
}
|
|
void Signal() { cv_.notify_one(); }
|
|
void SignalAll() { cv_.notify_all(); }
|
|
|
|
private:
|
|
std::condition_variable cv_;
|
|
Mutex* const mu_;
|
|
};
|
|
|
|
inline bool Snappy_Compress(const char* input, size_t length,
|
|
std::string* output) {
|
|
#if HAVE_SNAPPY
|
|
output->resize(snappy::MaxCompressedLength(length));
|
|
size_t outlen;
|
|
snappy::RawCompress(input, length, &(*output)[0], &outlen);
|
|
output->resize(outlen);
|
|
return true;
|
|
#else
|
|
// Silence compiler warnings about unused arguments.
|
|
(void)input;
|
|
(void)length;
|
|
(void)output;
|
|
#endif // HAVE_SNAPPY
|
|
|
|
return false;
|
|
}
|
|
|
|
inline bool Snappy_GetUncompressedLength(const char* input, size_t length,
|
|
size_t* result) {
|
|
#if HAVE_SNAPPY
|
|
return snappy::GetUncompressedLength(input, length, result);
|
|
#else
|
|
// Silence compiler warnings about unused arguments.
|
|
(void)input;
|
|
(void)length;
|
|
(void)result;
|
|
return false;
|
|
#endif // HAVE_SNAPPY
|
|
}
|
|
|
|
inline bool Snappy_Uncompress(const char* input, size_t length, char* output) {
|
|
#if HAVE_SNAPPY
|
|
return snappy::RawUncompress(input, length, output);
|
|
#else
|
|
// Silence compiler warnings about unused arguments.
|
|
(void)input;
|
|
(void)length;
|
|
(void)output;
|
|
return false;
|
|
#endif // HAVE_SNAPPY
|
|
}
|
|
|
|
inline bool Zstd_Compress(int level, const char* input, size_t length,
|
|
std::string* output) {
|
|
#if HAVE_ZSTD
|
|
// Get the MaxCompressedLength.
|
|
size_t outlen = ZSTD_compressBound(length);
|
|
if (ZSTD_isError(outlen)) {
|
|
return false;
|
|
}
|
|
output->resize(outlen);
|
|
ZSTD_CCtx* ctx = ZSTD_createCCtx();
|
|
ZSTD_compressionParameters parameters =
|
|
ZSTD_getCParams(level, std::max(length, size_t{1}), /*dictSize=*/0);
|
|
ZSTD_CCtx_setCParams(ctx, parameters);
|
|
outlen = ZSTD_compress2(ctx, &(*output)[0], output->size(), input, length);
|
|
ZSTD_freeCCtx(ctx);
|
|
if (ZSTD_isError(outlen)) {
|
|
return false;
|
|
}
|
|
output->resize(outlen);
|
|
return true;
|
|
#else
|
|
// Silence compiler warnings about unused arguments.
|
|
(void)level;
|
|
(void)input;
|
|
(void)length;
|
|
(void)output;
|
|
return false;
|
|
#endif // HAVE_ZSTD
|
|
}
|
|
|
|
inline bool Zstd_GetUncompressedLength(const char* input, size_t length,
|
|
size_t* result) {
|
|
#if HAVE_ZSTD
|
|
size_t size = ZSTD_getFrameContentSize(input, length);
|
|
if (size == 0) return false;
|
|
*result = size;
|
|
return true;
|
|
#else
|
|
// Silence compiler warnings about unused arguments.
|
|
(void)input;
|
|
(void)length;
|
|
(void)result;
|
|
return false;
|
|
#endif // HAVE_ZSTD
|
|
}
|
|
|
|
inline bool Zstd_Uncompress(const char* input, size_t length, char* output) {
|
|
#if HAVE_ZSTD
|
|
size_t outlen;
|
|
if (!Zstd_GetUncompressedLength(input, length, &outlen)) {
|
|
return false;
|
|
}
|
|
ZSTD_DCtx* ctx = ZSTD_createDCtx();
|
|
outlen = ZSTD_decompressDCtx(ctx, output, outlen, input, length);
|
|
ZSTD_freeDCtx(ctx);
|
|
if (ZSTD_isError(outlen)) {
|
|
return false;
|
|
}
|
|
return true;
|
|
#else
|
|
// Silence compiler warnings about unused arguments.
|
|
(void)input;
|
|
(void)length;
|
|
(void)output;
|
|
return false;
|
|
#endif // HAVE_ZSTD
|
|
}
|
|
|
|
inline bool GetHeapProfile(void (*func)(void*, const char*, int), void* arg) {
|
|
// Silence compiler warnings about unused arguments.
|
|
(void)func;
|
|
(void)arg;
|
|
return false;
|
|
}
|
|
|
|
inline uint32_t AcceleratedCRC32C(uint32_t crc, const char* buf, size_t size) {
|
|
#if HAVE_CRC32C
|
|
return ::crc32c::Extend(crc, reinterpret_cast<const uint8_t*>(buf), size);
|
|
#else
|
|
// Silence compiler warnings about unused arguments.
|
|
(void)crc;
|
|
(void)buf;
|
|
(void)size;
|
|
return 0;
|
|
#endif // HAVE_CRC32C
|
|
}
|
|
|
|
} // namespace port
|
|
} // namespace leveldb
|
|
|
|
#endif // STORAGE_LEVELDB_PORT_PORT_STDCXX_H_
|