2015-02-02 23:49:22 +01:00
|
|
|
// Copyright (c) 2014, 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.
|
|
|
|
#pragma once
|
|
|
|
|
2015-02-27 00:19:17 +01:00
|
|
|
#include <cstdlib>
|
2015-02-02 23:49:22 +01:00
|
|
|
#include <string>
|
|
|
|
|
|
|
|
namespace rocksdb {
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If ROCKSDB_XFTEST_FORCE has a value of 1, XFUNC is forced to be defined.
|
|
|
|
* If ROCKSDB_XFTEST_FORCE has a value other than 1,
|
|
|
|
* XFUNC is forced to be undefined.
|
|
|
|
* If ROCKSDB_XFTEST_FORCE is undefined, XFUNC is defined based on NDEBUG,
|
|
|
|
* with XFUNC only being set for debug builds.
|
|
|
|
*/
|
|
|
|
#if defined(ROCKSDB_XFTEST_FORCE)
|
2015-04-10 06:05:09 +02:00
|
|
|
#ifndef ROCKSDB_LITE
|
2015-02-02 23:49:22 +01:00
|
|
|
#if (ROCKSDB_XFTEST_FORCE == 1)
|
|
|
|
#define XFUNC
|
2015-04-10 06:05:09 +02:00
|
|
|
#endif // ROCKSDB_XFTEST_FORCE == 1
|
2015-03-17 12:04:05 +01:00
|
|
|
#elif defined(NDEBUG)
|
2015-02-02 23:49:22 +01:00
|
|
|
#else
|
|
|
|
#define XFUNC
|
2015-04-10 06:05:09 +02:00
|
|
|
#endif // defined(ROCKSDB_XFTEST_FORCE)
|
|
|
|
#endif // !ROCKSDB_LITE
|
2015-02-02 23:49:22 +01:00
|
|
|
|
|
|
|
#ifndef XFUNC
|
|
|
|
#define XFUNC_TEST(condition, location, lfname, fname, ...)
|
|
|
|
#else
|
|
|
|
|
2015-02-03 00:20:19 +01:00
|
|
|
struct Options;
|
2015-07-03 02:23:41 +02:00
|
|
|
struct WriteOptions;
|
2015-02-18 20:49:31 +01:00
|
|
|
class ManagedIterator;
|
|
|
|
class DBImpl;
|
2015-02-02 23:49:22 +01:00
|
|
|
void GetXFTestOptions(Options* options, int skip_policy);
|
2015-02-18 20:49:31 +01:00
|
|
|
void xf_manage_release(ManagedIterator* iter);
|
|
|
|
void xf_manage_new(DBImpl* db, ReadOptions* readoptions,
|
|
|
|
bool is_snapshot_supported);
|
|
|
|
void xf_manage_create(ManagedIterator* iter);
|
|
|
|
void xf_manage_options(ReadOptions* read_options);
|
2015-07-03 02:23:41 +02:00
|
|
|
void xf_transaction_set_memtable_history(
|
|
|
|
int32_t* max_write_buffer_number_to_maintain);
|
|
|
|
void xf_transaction_clear_memtable_history(
|
|
|
|
int32_t* max_write_buffer_number_to_maintain);
|
|
|
|
void xf_transaction_write(const WriteOptions& write_options,
|
|
|
|
const DBOptions& db_options,
|
|
|
|
class WriteBatch* my_batch,
|
|
|
|
class WriteCallback* callback, DBImpl* db_impl,
|
|
|
|
Status* success, bool* write_attempted);
|
2015-02-02 23:49:22 +01:00
|
|
|
|
|
|
|
// This class provides the facility to run custom code to test a specific
|
|
|
|
// feature typically with all existing unit tests.
|
|
|
|
// A developer could specify cross functional test points in the codebase
|
|
|
|
// via XFUNC_TEST.
|
|
|
|
// Each xfunc test represents a position in the execution stream of a thread.
|
|
|
|
// Whenever that particular piece of code is called, the given cross-functional
|
|
|
|
// test point is executed.
|
|
|
|
// eg. on DBOpen, a particular option can be set.
|
|
|
|
// on Get, a particular option can be set, or a specific check can be invoked.
|
|
|
|
// XFUNC_TEST(TestName, location, lfname, FunctionName, Args)
|
|
|
|
// Turn on a specific cross functional test by setting the environment variable
|
|
|
|
// ROCKSDB_XFUNC_TEST
|
|
|
|
|
|
|
|
class XFuncPoint {
|
|
|
|
public:
|
|
|
|
// call once at the beginning of a test to get the test name
|
|
|
|
static void Init() {
|
|
|
|
char* s = getenv("ROCKSDB_XFUNC_TEST");
|
|
|
|
if (s == nullptr) {
|
|
|
|
xfunc_test_ = "";
|
|
|
|
enabled_ = false;
|
|
|
|
} else {
|
|
|
|
xfunc_test_ = s;
|
|
|
|
enabled_ = true;
|
|
|
|
}
|
|
|
|
initialized_ = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool Initialized() { return initialized_; }
|
|
|
|
|
|
|
|
static bool Check(std::string test) {
|
|
|
|
return (enabled_ &&
|
|
|
|
((test.compare("") == 0) || (test.compare(xfunc_test_) == 0)));
|
|
|
|
}
|
|
|
|
|
2015-02-18 20:49:31 +01:00
|
|
|
static void SetSkip(int skip) { skip_policy_ = skip; }
|
|
|
|
static int GetSkip(void) { return skip_policy_; }
|
|
|
|
|
2015-02-02 23:49:22 +01:00
|
|
|
private:
|
|
|
|
static std::string xfunc_test_;
|
|
|
|
static bool initialized_;
|
|
|
|
static bool enabled_;
|
2015-02-18 20:49:31 +01:00
|
|
|
static int skip_policy_;
|
2015-02-02 23:49:22 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
// Use XFUNC_TEST to specify cross functional test points inside the code base.
|
|
|
|
// By setting ROCKSDB_XFUNC_TEST, all XFUNC_TEST having that
|
|
|
|
// value in the condition field will be executed.
|
|
|
|
// The second argument specifies a string representing the calling location
|
|
|
|
// The third argument, lfname, is the name of the function which will be created
|
|
|
|
// and called.
|
|
|
|
// The fourth argument fname represents the function to be called
|
|
|
|
// The arguments following that are the arguments to fname
|
|
|
|
// See Options::Options in options.h for an example use case.
|
|
|
|
// XFUNC_TEST is no op in release build.
|
|
|
|
#define XFUNC_TEST(condition, location, lfname, fname, ...) \
|
|
|
|
{ \
|
|
|
|
if (!XFuncPoint::Initialized()) { \
|
|
|
|
XFuncPoint::Init(); \
|
|
|
|
} \
|
|
|
|
if (XFuncPoint::Check(condition)) { \
|
|
|
|
std::function<void()> lfname = std::bind(fname, __VA_ARGS__); \
|
|
|
|
lfname(); \
|
|
|
|
} \
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // XFUNC
|
|
|
|
|
|
|
|
enum SkipPolicy { kSkipNone = 0, kSkipNoSnapshot = 1, kSkipNoPrefix = 2 };
|
|
|
|
} // namespace rocksdb
|