2017-04-12 22:49:58 +08:00
|
|
|
#include <chrono>
|
|
|
|
#include <memory>
|
|
|
|
#include <thread>
|
|
|
|
|
2017-06-21 17:29:13 +08:00
|
|
|
#include <gflags/gflags.h>
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
|
2017-06-07 16:15:08 +08:00
|
|
|
#include "data_structures/concurrent/skiplist_gc.hpp"
|
2017-04-12 22:49:58 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* FakeItem class which increments a variable in the destructor.
|
|
|
|
* Used to keep track of the number of destroyed elements in GC.
|
|
|
|
*/
|
|
|
|
class FakeItem {
|
|
|
|
public:
|
|
|
|
FakeItem(std::atomic<int> &count, int value) : count(count), value(value) {}
|
|
|
|
~FakeItem() { count.fetch_add(1); }
|
|
|
|
|
|
|
|
bool operator<(const FakeItem &item) const {
|
|
|
|
return this->value < item.value;
|
|
|
|
}
|
|
|
|
bool operator>(const FakeItem &item) const {
|
|
|
|
return this->value > item.value;
|
|
|
|
}
|
|
|
|
|
2017-06-07 16:15:08 +08:00
|
|
|
static void destroy(FakeItem *item) { delete item; }
|
|
|
|
|
2017-04-12 22:49:58 +08:00
|
|
|
private:
|
|
|
|
std::atomic<int> &count;
|
|
|
|
int value;
|
|
|
|
};
|
|
|
|
|
2017-06-07 16:15:08 +08:00
|
|
|
DECLARE_int32(skiplist_gc_interval);
|
2017-05-16 21:54:40 +08:00
|
|
|
|
2017-06-07 16:15:08 +08:00
|
|
|
TEST(SkipListGC, CreateNewAccessors) {
|
2017-12-18 17:20:35 +08:00
|
|
|
FLAGS_skiplist_gc_interval = -1;
|
2017-06-07 16:15:08 +08:00
|
|
|
SkipListGC<FakeItem> gc;
|
|
|
|
auto &accessor1 = gc.CreateNewAccessor();
|
|
|
|
auto &accessor2 = gc.CreateNewAccessor();
|
|
|
|
auto &accessor3 = gc.CreateNewAccessor();
|
|
|
|
|
|
|
|
EXPECT_EQ(accessor1.id_, 1);
|
|
|
|
EXPECT_EQ(accessor2.id_, 2);
|
|
|
|
EXPECT_EQ(accessor3.id_, 3);
|
|
|
|
|
|
|
|
accessor1.alive_ = false;
|
|
|
|
accessor2.alive_ = false;
|
|
|
|
accessor3.alive_ = false;
|
2017-04-12 22:49:58 +08:00
|
|
|
}
|
|
|
|
|
2017-06-07 16:15:08 +08:00
|
|
|
TEST(SkipListGC, DeleteItem) {
|
2017-12-18 17:20:35 +08:00
|
|
|
FLAGS_skiplist_gc_interval = -1;
|
2017-06-07 16:15:08 +08:00
|
|
|
SkipListGC<FakeItem> gc;
|
|
|
|
auto &accessor1 = gc.CreateNewAccessor();
|
|
|
|
std::atomic<int> count{0};
|
|
|
|
auto item1 = new FakeItem(count, 1);
|
|
|
|
gc.Collect(item1);
|
|
|
|
|
|
|
|
// Kill the accesssor
|
|
|
|
accessor1.alive_ = false;
|
|
|
|
gc.GarbageCollect();
|
|
|
|
EXPECT_EQ(count, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(SkipListGC, DontDeleteItem) {
|
2017-12-18 17:20:35 +08:00
|
|
|
FLAGS_skiplist_gc_interval = -1;
|
2017-06-07 16:15:08 +08:00
|
|
|
SkipListGC<FakeItem> gc;
|
|
|
|
auto &accessor1 = gc.CreateNewAccessor();
|
|
|
|
auto &accessor2 = gc.CreateNewAccessor();
|
|
|
|
std::atomic<int> count{0};
|
|
|
|
auto item1 = new FakeItem(count, 1);
|
|
|
|
gc.Collect(item1);
|
|
|
|
|
|
|
|
// Kill the accesssor
|
|
|
|
accessor2.alive_ = false;
|
|
|
|
|
|
|
|
// Nothing deleted because accessor1 is blocking.
|
|
|
|
gc.GarbageCollect();
|
|
|
|
EXPECT_EQ(count, 0);
|
|
|
|
|
|
|
|
// Accessor 1 is not blocking anymore.
|
|
|
|
accessor1.alive_ = false;
|
|
|
|
gc.GarbageCollect();
|
|
|
|
EXPECT_EQ(count, 1);
|
2017-04-12 22:49:58 +08:00
|
|
|
}
|
|
|
|
|
2017-06-07 16:15:08 +08:00
|
|
|
TEST(SkipListGC, Destructor) {
|
2017-12-18 17:20:35 +08:00
|
|
|
FLAGS_skiplist_gc_interval = -1;
|
2017-06-07 16:15:08 +08:00
|
|
|
std::atomic<int> count{0};
|
|
|
|
auto item1 = new FakeItem(count, 1);
|
2017-04-12 22:49:58 +08:00
|
|
|
{
|
2017-06-07 16:15:08 +08:00
|
|
|
SkipListGC<FakeItem> gc;
|
|
|
|
gc.Collect(item1);
|
|
|
|
EXPECT_EQ(count, 0);
|
2017-04-12 22:49:58 +08:00
|
|
|
}
|
2017-06-07 16:15:08 +08:00
|
|
|
EXPECT_EQ(count, 1);
|
2017-04-12 22:49:58 +08:00
|
|
|
}
|
|
|
|
|
2017-06-07 16:15:08 +08:00
|
|
|
TEST(SkipListGC, MultipleDeletes) {
|
2017-12-18 17:20:35 +08:00
|
|
|
FLAGS_skiplist_gc_interval = -1;
|
2017-06-07 16:15:08 +08:00
|
|
|
SkipListGC<FakeItem> gc;
|
|
|
|
std::atomic<int> count{0};
|
|
|
|
auto &accessor1 = gc.CreateNewAccessor();
|
|
|
|
auto item1 = new FakeItem(count, 1);
|
|
|
|
gc.Collect(item1);
|
|
|
|
|
|
|
|
auto &accessor2 = gc.CreateNewAccessor();
|
|
|
|
auto item2 = new FakeItem(count, 1);
|
|
|
|
gc.Collect(item2);
|
|
|
|
|
|
|
|
auto &accessor3 = gc.CreateNewAccessor();
|
|
|
|
auto item3 = new FakeItem(count, 1);
|
|
|
|
gc.Collect(item3);
|
|
|
|
|
|
|
|
accessor1.alive_ = false;
|
|
|
|
accessor2.alive_ = false;
|
|
|
|
gc.GarbageCollect();
|
|
|
|
EXPECT_EQ(count, 2);
|
|
|
|
|
|
|
|
accessor3.alive_ = false;
|
|
|
|
gc.GarbageCollect();
|
|
|
|
EXPECT_EQ(count, 3);
|
2017-04-12 22:49:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char **argv) {
|
2017-06-21 17:29:13 +08:00
|
|
|
google::InitGoogleLogging(argv[0]);
|
2017-04-12 22:49:58 +08:00
|
|
|
::testing::InitGoogleTest(&argc, argv);
|
|
|
|
return RUN_ALL_TESTS();
|
|
|
|
}
|