14c9044909
Summary: With quite frequent changes of serialization backend, it's getting really painful updating the C++ implementation. This diff defines the Symbol class via LCP, so that the C++ serialization is generated instead of written by hand. Reviewers: mtomic, llugovic Reviewed By: mtomic Subscribers: pullbot Differential Revision: https://phabricator.memgraph.io/D1741
209 lines
6.2 KiB
C++
209 lines
6.2 KiB
C++
#include <sstream>
|
|
#include <string>
|
|
|
|
#include <benchmark/benchmark.h>
|
|
|
|
#include <capnp/serialize.h>
|
|
#include <kj/std/iostream.h>
|
|
|
|
#include "query/frontend/semantic/symbol_serialization.capnp.h"
|
|
#include "query/frontend/semantic/symbol.hpp"
|
|
|
|
#include "communication/rpc/serialization.hpp"
|
|
|
|
class SymbolVectorFixture : public benchmark::Fixture {
|
|
protected:
|
|
std::vector<query::Symbol> symbols_;
|
|
|
|
void SetUp(const benchmark::State &state) override {
|
|
using Type = ::query::Symbol::Type;
|
|
std::vector<Type> types{Type::ANY, Type::VERTEX, Type::EDGE,
|
|
Type::PATH, Type::NUMBER, Type::EDGE_LIST};
|
|
symbols_.reserve(state.range(0));
|
|
for (int i = 0; i < state.range(0); ++i) {
|
|
std::string name = "Symbol " + std::to_string(i);
|
|
bool user_declared = i % 2;
|
|
auto type = types[i % types.size()];
|
|
symbols_.emplace_back(name, i, user_declared, type, i);
|
|
}
|
|
}
|
|
|
|
void TearDown(const benchmark::State &) override { symbols_.clear(); }
|
|
};
|
|
|
|
void SymbolVectorToCapnpMessage(const std::vector<query::Symbol> &symbols,
|
|
capnp::MessageBuilder &message) {
|
|
auto symbols_builder =
|
|
message.initRoot<capnp::List<query::capnp::Symbol>>(symbols.size());
|
|
for (int i = 0; i < symbols.size(); ++i) {
|
|
const auto &sym = symbols[i];
|
|
query::capnp::Symbol::Builder sym_builder = symbols_builder[i];
|
|
sym_builder.setName(sym.name());
|
|
sym_builder.setPosition(sym.position());
|
|
sym_builder.setType(query::capnp::Symbol::Type::ANY);
|
|
sym_builder.setUserDeclared(sym.user_declared());
|
|
sym_builder.setTokenPosition(sym.token_position());
|
|
}
|
|
}
|
|
|
|
std::stringstream SerializeCapnpSymbolVector(
|
|
const std::vector<query::Symbol> &symbols) {
|
|
std::stringstream stream(std::ios_base::in | std::ios_base::out |
|
|
std::ios_base::binary);
|
|
{
|
|
capnp::MallocMessageBuilder message;
|
|
SymbolVectorToCapnpMessage(symbols, message);
|
|
kj::std::StdOutputStream std_stream(stream);
|
|
kj::BufferedOutputStreamWrapper buffered_stream(std_stream);
|
|
writeMessage(buffered_stream, message);
|
|
}
|
|
return stream;
|
|
}
|
|
|
|
BENCHMARK_DEFINE_F(SymbolVectorFixture, CapnpSerial)(benchmark::State &state) {
|
|
while (state.KeepRunning()) {
|
|
SerializeCapnpSymbolVector(symbols_);
|
|
}
|
|
state.SetItemsProcessed(state.iterations());
|
|
}
|
|
|
|
BENCHMARK_DEFINE_F(SymbolVectorFixture, CapnpDeserial)
|
|
(benchmark::State &state) {
|
|
while (state.KeepRunning()) {
|
|
state.PauseTiming();
|
|
auto stream = SerializeCapnpSymbolVector(symbols_);
|
|
state.ResumeTiming();
|
|
kj::std::StdInputStream std_stream(stream);
|
|
capnp::InputStreamMessageReader message(std_stream);
|
|
auto symbols_reader = message.getRoot<capnp::List<query::capnp::Symbol>>();
|
|
std::vector<query::Symbol> symbols;
|
|
symbols.reserve(symbols_reader.size());
|
|
for (const auto &sym : symbols_reader) {
|
|
symbols.emplace_back(sym.getName().cStr(), sym.getPosition(),
|
|
sym.getUserDeclared(), query::Symbol::Type::ANY,
|
|
sym.getTokenPosition());
|
|
}
|
|
}
|
|
state.SetItemsProcessed(state.iterations());
|
|
}
|
|
|
|
BENCHMARK_REGISTER_F(SymbolVectorFixture, CapnpSerial)
|
|
->RangeMultiplier(4)
|
|
->Range(4, 1 << 12)
|
|
->Unit(benchmark::kNanosecond);
|
|
|
|
BENCHMARK_REGISTER_F(SymbolVectorFixture, CapnpDeserial)
|
|
->RangeMultiplier(4)
|
|
->Range(4, 1 << 12)
|
|
->Unit(benchmark::kNanosecond);
|
|
|
|
uint8_t Type2Int(query::Symbol::Type type) {
|
|
switch (type) {
|
|
case query::Symbol::Type::ANY:
|
|
return 1;
|
|
case query::Symbol::Type::VERTEX:
|
|
return 2;
|
|
case query::Symbol::Type::EDGE:
|
|
return 3;
|
|
case query::Symbol::Type::PATH:
|
|
return 4;
|
|
case query::Symbol::Type::NUMBER:
|
|
return 5;
|
|
case query::Symbol::Type::EDGE_LIST:
|
|
return 6;
|
|
}
|
|
}
|
|
|
|
query::Symbol::Type Int2Type(uint8_t value) {
|
|
switch (value) {
|
|
case 1:
|
|
return query::Symbol::Type::ANY;
|
|
case 2:
|
|
return query::Symbol::Type::VERTEX;
|
|
case 3:
|
|
return query::Symbol::Type::EDGE;
|
|
case 4:
|
|
return query::Symbol::Type::PATH;
|
|
case 5:
|
|
return query::Symbol::Type::NUMBER;
|
|
case 6:
|
|
return query::Symbol::Type::EDGE_LIST;
|
|
}
|
|
CHECK(false);
|
|
}
|
|
|
|
namespace slk {
|
|
void Save(const query::Symbol &obj, slk::Builder *builder) {
|
|
Save(obj.name(), builder);
|
|
Save(obj.position(), builder);
|
|
Save(Type2Int(obj.type()), builder);
|
|
Save(obj.user_declared(), builder);
|
|
Save(obj.token_position(), builder);
|
|
}
|
|
|
|
void Load(query::Symbol *obj, slk::Reader *reader) {
|
|
std::string name;
|
|
Load(&name, reader);
|
|
int position = 0;
|
|
Load(&position, reader);
|
|
uint8_t type = 0;
|
|
Load(&type, reader);
|
|
bool user_declared = false;
|
|
Load(&user_declared, reader);
|
|
int token_position = 0;
|
|
Load(&token_position, reader);
|
|
*obj = query::Symbol(std::move(name), position, user_declared, Int2Type(type),
|
|
token_position);
|
|
}
|
|
} // namespace slk
|
|
|
|
void SymbolVectorToCustom(const std::vector<query::Symbol> &symbols,
|
|
slk::Builder *builder) {
|
|
slk::Save(symbols.size(), builder);
|
|
for (int i = 0; i < symbols.size(); ++i) {
|
|
slk::Save(symbols[i], builder);
|
|
}
|
|
}
|
|
|
|
void CustomToSymbolVector(std::vector<query::Symbol> *symbols,
|
|
slk::Reader *reader) {
|
|
uint64_t size = 0;
|
|
slk::Load(&size, reader);
|
|
symbols->resize(size);
|
|
for (uint64_t i = 0; i < size; ++i) {
|
|
slk::Load(&(*symbols)[i], reader);
|
|
}
|
|
}
|
|
|
|
BENCHMARK_DEFINE_F(SymbolVectorFixture, CustomSerial)(benchmark::State &state) {
|
|
while (state.KeepRunning()) {
|
|
slk::Builder builder;
|
|
SymbolVectorToCustom(symbols_, &builder);
|
|
}
|
|
state.SetItemsProcessed(state.iterations());
|
|
}
|
|
|
|
BENCHMARK_DEFINE_F(SymbolVectorFixture, CustomDeserial)
|
|
(benchmark::State &state) {
|
|
slk::Builder builder;
|
|
SymbolVectorToCustom(symbols_, &builder);
|
|
while (state.KeepRunning()) {
|
|
slk::Reader reader(builder.data(), builder.size());
|
|
std::vector<query::Symbol> symbols;
|
|
CustomToSymbolVector(&symbols, &reader);
|
|
}
|
|
state.SetItemsProcessed(state.iterations());
|
|
}
|
|
|
|
BENCHMARK_REGISTER_F(SymbolVectorFixture, CustomSerial)
|
|
->RangeMultiplier(4)
|
|
->Range(4, 1 << 12)
|
|
->Unit(benchmark::kNanosecond);
|
|
|
|
BENCHMARK_REGISTER_F(SymbolVectorFixture, CustomDeserial)
|
|
->RangeMultiplier(4)
|
|
->Range(4, 1 << 12)
|
|
->Unit(benchmark::kNanosecond);
|
|
|
|
BENCHMARK_MAIN();
|