Commit e33b08f9 authored by Kenton Varda's avatar Kenton Varda

Add benchmarking mode for measuring improvement from a change. Also, fix a bug.

parent f966fc41
...@@ -242,19 +242,21 @@ struct BenchmarkMethods { ...@@ -242,19 +242,21 @@ struct BenchmarkMethods {
typename Compression::BufferedInput bufferedInput(inputStream); typename Compression::BufferedInput bufferedInput(inputStream);
CountingOutputStream output(outputFd); CountingOutputStream output(outputFd);
typename ReuseStrategy::ScratchSpace scratch; typename ReuseStrategy::ScratchSpace builderScratch;
typename ReuseStrategy::ScratchSpace readerScratch;
for (; iters > 0; --iters) { for (; iters > 0; --iters) {
typename TestCase::Expectation expected; typename TestCase::Expectation expected;
{ {
typename ReuseStrategy::MessageBuilder builder(scratch); typename ReuseStrategy::MessageBuilder builder(builderScratch);
expected = TestCase::setupRequest( expected = TestCase::setupRequest(
builder.template initRoot<typename TestCase::Request>()); builder.template initRoot<typename TestCase::Request>());
Compression::write(output, builder); Compression::write(output, builder);
} }
{ {
typename ReuseStrategy::template MessageReader<Compression> reader(bufferedInput, scratch); typename ReuseStrategy::template MessageReader<Compression> reader(
bufferedInput, readerScratch);
if (!TestCase::checkResponse( if (!TestCase::checkResponse(
reader.template getRoot<typename TestCase::Response>(), expected)) { reader.template getRoot<typename TestCase::Response>(), expected)) {
throw std::logic_error("Incorrect response."); throw std::logic_error("Incorrect response.");
......
...@@ -277,6 +277,15 @@ void reportComparisonHeader() { ...@@ -277,6 +277,15 @@ void reportComparisonHeader() {
cout << setfill('=') << setw(85) << "" << setfill(' ') << endl; cout << setfill('=') << setw(85) << "" << setfill(' ') << endl;
} }
void reportOldNewComparisonHeader() {
cout << setw(40) << left << "Measure"
<< setw(15) << right << "Old"
<< setw(15) << right << "New"
<< setw(15) << right << "Improvement"
<< endl;
cout << setfill('=') << setw(85) << "" << setfill(' ') << endl;
}
class Gain { class Gain {
public: public:
Gain(double oldValue, double newValue) Gain(double oldValue, double newValue)
...@@ -343,6 +352,11 @@ size_t fileSize(const std::string& name) { ...@@ -343,6 +352,11 @@ size_t fileSize(const std::string& name) {
int main(int argc, char* argv[]) { int main(int argc, char* argv[]) {
char* path = argv[0]; char* path = argv[0];
char* slashpos = strrchr(path, '/'); char* slashpos = strrchr(path, '/');
char origDir[1024];
if (getcwd(origDir, sizeof(origDir)) == nullptr) {
perror("getcwd");
return 1;
}
if (slashpos != nullptr) { if (slashpos != nullptr) {
*slashpos = '\0'; *slashpos = '\0';
if (chdir(path) < 0) { if (chdir(path) < 0) {
...@@ -356,6 +370,7 @@ int main(int argc, char* argv[]) { ...@@ -356,6 +370,7 @@ int main(int argc, char* argv[]) {
Mode mode = Mode::PIPE_SYNC; Mode mode = Mode::PIPE_SYNC;
Compression compression = Compression::NONE; Compression compression = Compression::NONE;
uint64_t iters = 1; uint64_t iters = 1;
const char* oldDir = nullptr;
for (int i = 1; i < argc; i++) { for (int i = 1; i < argc; i++) {
string arg = argv[i]; string arg = argv[i];
...@@ -371,6 +386,13 @@ int main(int argc, char* argv[]) { ...@@ -371,6 +386,13 @@ int main(int argc, char* argv[]) {
testCase = TestCase::CARSALES; testCase = TestCase::CARSALES;
} else if (arg == "snappy") { } else if (arg == "snappy") {
compression = Compression::SNAPPY; compression = Compression::SNAPPY;
} else if (arg == "-c") {
++i;
if (i == argc) {
fprintf(stderr, "-c requires argument.\n");
return 1;
}
oldDir = argv[i];
} else { } else {
fprintf(stderr, "Unknown option: %s\n", argv[i]); fprintf(stderr, "Unknown option: %s\n", argv[i]);
return 1; return 1;
...@@ -478,6 +500,7 @@ int main(int argc, char* argv[]) { ...@@ -478,6 +500,7 @@ int main(int argc, char* argv[]) {
Product::PROTOBUF, testCase, mode, Reuse::YES, compression, iters); Product::PROTOBUF, testCase, mode, Reuse::YES, compression, iters);
protobuf.objectSize = protobufBase.objectSize; protobuf.objectSize = protobufBase.objectSize;
reportResults("Protobuf I/O", iters, protobuf); reportResults("Protobuf I/O", iters, protobuf);
TestResult capnp = runTest( TestResult capnp = runTest(
Product::CAPNPROTO, testCase, mode, Reuse::YES, compression, iters); Product::CAPNPROTO, testCase, mode, Reuse::YES, compression, iters);
capnp.objectSize = capnpBase.objectSize; capnp.objectSize = capnpBase.objectSize;
...@@ -487,6 +510,59 @@ int main(int argc, char* argv[]) { ...@@ -487,6 +510,59 @@ int main(int argc, char* argv[]) {
capnpPacked.objectSize = capnpBase.objectSize; capnpPacked.objectSize = capnpBase.objectSize;
reportResults("Cap'n Proto packed I/O", iters, capnpPacked); reportResults("Cap'n Proto packed I/O", iters, capnpPacked);
size_t protobufBinarySize = fileSize("protobuf-" + std::string(testCaseName(testCase)));
size_t capnpBinarySize = fileSize("capnproto-" + std::string(testCaseName(testCase)));
size_t protobufCodeSize = fileSize(std::string(testCaseName(testCase)) + ".pb.cc")
+ fileSize(std::string(testCaseName(testCase)) + ".pb.h");
size_t capnpCodeSize = fileSize(std::string(testCaseName(testCase)) + ".capnp.c++")
+ fileSize(std::string(testCaseName(testCase)) + ".capnp.h");
size_t protobufObjSize = fileSize(std::string(testCaseName(testCase)) + ".pb.o");
size_t capnpObjSize = fileSize(std::string(testCaseName(testCase)) + ".capnp.o");
TestResult oldCapnpBase;
TestResult oldCapnpNoReuse;
TestResult oldCapnp;
TestResult oldCapnpPacked;
size_t oldCapnpBinarySize = 0;
size_t oldCapnpCodeSize = 0;
size_t oldCapnpObjSize = 0;
if (oldDir != nullptr) {
if (chdir(origDir) < 0) {
perror("chdir");
return 1;
}
if (chdir(oldDir) < 0) {
perror(oldDir);
return 1;
}
oldCapnpBase = runTest(
Product::CAPNPROTO, testCase, Mode::OBJECTS, Reuse::YES, compression, iters);
oldCapnpBase.objectSize = runTest(
Product::CAPNPROTO, testCase, Mode::OBJECT_SIZE, Reuse::YES, compression, iters)
.objectSize;
reportResults("Old Cap'n Proto pass-by-object", iters, oldCapnpBase);
oldCapnpNoReuse = runTest(
Product::CAPNPROTO, testCase, Mode::OBJECTS, Reuse::NO, compression, iters);
oldCapnpNoReuse.objectSize = runTest(
Product::CAPNPROTO, testCase, Mode::OBJECT_SIZE, Reuse::NO, compression, iters).objectSize;
reportResults("Old Cap'n Proto w/o object reuse", iters, oldCapnpNoReuse);
oldCapnp = runTest(
Product::CAPNPROTO, testCase, mode, Reuse::YES, compression, iters);
oldCapnp.objectSize = oldCapnpBase.objectSize;
reportResults("Old Cap'n Proto I/O", iters, oldCapnp);
oldCapnpPacked = runTest(
Product::CAPNPROTO, testCase, mode, Reuse::YES, Compression::PACKED, iters);
oldCapnpPacked.objectSize = oldCapnpBase.objectSize;
reportResults("Old Cap'n Proto packed I/O", iters, oldCapnpPacked);
oldCapnpBinarySize = fileSize("capnproto-" + std::string(testCaseName(testCase)));
oldCapnpCodeSize = fileSize(std::string(testCaseName(testCase)) + ".capnp.c++")
+ fileSize(std::string(testCaseName(testCase)) + ".capnp.h");
oldCapnpObjSize = fileSize(std::string(testCaseName(testCase)) + ".capnp.o");
}
cout << endl; cout << endl;
reportComparisonHeader(); reportComparisonHeader();
...@@ -512,16 +588,43 @@ int main(int argc, char* argv[]) { ...@@ -512,16 +588,43 @@ int main(int argc, char* argv[]) {
protobuf.messageSize, capnpPacked.messageSize, iters); protobuf.messageSize, capnpPacked.messageSize, iters);
reportComparison("binary size (KiB)", "", reportComparison("binary size (KiB)", "",
fileSize("protobuf-" + std::string(testCaseName(testCase))) / 1024.0, protobufBinarySize / 1024.0, capnpBinarySize / 1024.0, 1);
fileSize("capnproto-" + std::string(testCaseName(testCase))) / 1024.0, 1);
reportComparison("generated code size (KiB)", "", reportComparison("generated code size (KiB)", "",
fileSize(std::string(testCaseName(testCase)) + ".pb.cc") / 1024.0 protobufCodeSize / 1024.0, capnpCodeSize / 1024.0, 1);
+ fileSize(std::string(testCaseName(testCase)) + ".pb.h") / 1024.0,
fileSize(std::string(testCaseName(testCase)) + ".capnp.c++") / 1024.0
+ fileSize(std::string(testCaseName(testCase)) + ".capnp.h") / 1024.0, 1);
reportComparison("generated obj size (KiB)", "", reportComparison("generated obj size (KiB)", "",
fileSize(std::string(testCaseName(testCase)) + ".pb.o") / 1024.0, protobufObjSize / 1024.0, capnpObjSize / 1024.0, 1);
fileSize(std::string(testCaseName(testCase)) + ".capnp.o") / 1024.0, 1);
cout << endl;
reportOldNewComparisonHeader();
if (oldDir != nullptr) {
reportComparison("memory overhead",
nullCase.objectSize, oldCapnpBase.objectSize, capnpBase.objectSize, iters);
reportComparison("memory overhead w/o object reuse",
nullCaseNoReuse.objectSize, oldCapnpNoReuse.objectSize, capnpNoReuse.objectSize, iters);
reportComparison("object manipulation time (us)", "",
((int64_t)oldCapnpBase.time.user - (int64_t)nullCase.time.user) / 1000.0,
((int64_t)capnpBase.time.user - (int64_t)nullCase.time.user) / 1000.0, iters);
reportComparison("object manipulation time w/o reuse (us)", "",
((int64_t)oldCapnpNoReuse.time.user - (int64_t)nullCaseNoReuse.time.user) / 1000.0,
((int64_t)capnpNoReuse.time.user - (int64_t)nullCaseNoReuse.time.user) / 1000.0, iters);
reportComparison("I/O time (us)", "",
((int64_t)oldCapnp.time.user - (int64_t)oldCapnpBase.time.user) / 1000.0,
((int64_t)capnp.time.user - (int64_t)capnpBase.time.user) / 1000.0, iters);
reportComparison("packed I/O time (us)", "",
((int64_t)oldCapnpPacked.time.user - (int64_t)oldCapnpBase.time.user) / 1000.0,
((int64_t)capnpPacked.time.user - (int64_t)capnpBase.time.user) / 1000.0, iters);
reportIntComparison("message size (bytes)", "", oldCapnp.messageSize, capnp.messageSize, iters);
reportIntComparison("packed message size (bytes)", "",
oldCapnpPacked.messageSize, capnpPacked.messageSize, iters);
reportComparison("binary size (KiB)", "",
oldCapnpBinarySize / 1024.0, capnpBinarySize / 1024.0, 1);
reportComparison("generated code size (KiB)", "",
oldCapnpCodeSize / 1024.0, capnpCodeSize / 1024.0, 1);
reportComparison("generated obj size (KiB)", "",
oldCapnpObjSize / 1024.0, capnpObjSize / 1024.0, 1);
}
return 0; return 0;
} }
......
...@@ -195,42 +195,52 @@ struct MallocMessageBuilder::MoreSegments { ...@@ -195,42 +195,52 @@ struct MallocMessageBuilder::MoreSegments {
MallocMessageBuilder::MallocMessageBuilder( MallocMessageBuilder::MallocMessageBuilder(
uint firstSegmentWords, AllocationStrategy allocationStrategy) uint firstSegmentWords, AllocationStrategy allocationStrategy)
: nextSize(firstSegmentWords), allocationStrategy(allocationStrategy), : nextSize(firstSegmentWords), allocationStrategy(allocationStrategy),
ownFirstSegment(true), firstSegment(nullptr) {} ownFirstSegment(true), returnedFirstSegment(false), firstSegment(nullptr) {}
MallocMessageBuilder::MallocMessageBuilder( MallocMessageBuilder::MallocMessageBuilder(
ArrayPtr<word> firstSegment, AllocationStrategy allocationStrategy) ArrayPtr<word> firstSegment, AllocationStrategy allocationStrategy)
: nextSize(firstSegment.size()), allocationStrategy(allocationStrategy), : nextSize(firstSegment.size()), allocationStrategy(allocationStrategy),
ownFirstSegment(false), firstSegment(firstSegment.begin()) {} ownFirstSegment(false), returnedFirstSegment(false), firstSegment(firstSegment.begin()) {
CAPNPROTO_ASSERT(firstSegment.size() > 0, "First segment size must be non-zero.");
// Checking just the first word should catch most cases of failing to zero the segment.
CAPNPROTO_ASSERT(*reinterpret_cast<uint64_t*>(firstSegment.begin()) == 0,
"First segment must be zeroed.");
}
MallocMessageBuilder::~MallocMessageBuilder() { MallocMessageBuilder::~MallocMessageBuilder() {
if (ownFirstSegment) { if (returnedFirstSegment) {
free(firstSegment); if (ownFirstSegment) {
} else { free(firstSegment);
ArrayPtr<const ArrayPtr<const word>> segments = getSegmentsForOutput(); } else {
if (segments.size() > 0) { // Must zero first segment.
CAPNPROTO_ASSERT(segments[0].begin() == firstSegment, ArrayPtr<const ArrayPtr<const word>> segments = getSegmentsForOutput();
"First segment in getSegmentsForOutput() is not the first segment allocated?"); if (segments.size() > 0) {
memset(firstSegment, 0, segments[0].size() * sizeof(word)); CAPNPROTO_ASSERT(segments[0].begin() == firstSegment,
"First segment in getSegmentsForOutput() is not the first segment allocated?");
memset(firstSegment, 0, segments[0].size() * sizeof(word));
}
} }
}
if (moreSegments != nullptr) { if (moreSegments != nullptr) {
for (void* ptr: moreSegments->segments) { for (void* ptr: moreSegments->segments) {
free(ptr); free(ptr);
}
} }
} }
} }
ArrayPtr<word> MallocMessageBuilder::allocateSegment(uint minimumSize) { ArrayPtr<word> MallocMessageBuilder::allocateSegment(uint minimumSize) {
if (!ownFirstSegment) { if (!returnedFirstSegment && !ownFirstSegment) {
ArrayPtr<word> result = arrayPtr(reinterpret_cast<word*>(firstSegment), nextSize); ArrayPtr<word> result = arrayPtr(reinterpret_cast<word*>(firstSegment), nextSize);
firstSegment = nullptr;
ownFirstSegment = true;
if (result.size() >= minimumSize) { if (result.size() >= minimumSize) {
returnedFirstSegment = true;
return result; return result;
} }
// If the provided first segment wasn't big enough, we discard it and proceed to allocate // If the provided first segment wasn't big enough, we discard it and proceed to allocate
// our own. This never happens in practice since minimumSize is always 1 for the first // our own. This never happens in practice since minimumSize is always 1 for the first
// segment. // segment.
ownFirstSegment = true;
} }
uint size = std::max(minimumSize, nextSize); uint size = std::max(minimumSize, nextSize);
...@@ -240,8 +250,11 @@ ArrayPtr<word> MallocMessageBuilder::allocateSegment(uint minimumSize) { ...@@ -240,8 +250,11 @@ ArrayPtr<word> MallocMessageBuilder::allocateSegment(uint minimumSize) {
throw std::bad_alloc(); throw std::bad_alloc();
} }
if (firstSegment == nullptr) { if (!returnedFirstSegment) {
firstSegment = result; firstSegment = result;
returnedFirstSegment = true;
// After the first segment, we want nextSize to equal the total size allocated so far.
if (allocationStrategy == AllocationStrategy::GROW_HEURISTICALLY) nextSize = size; if (allocationStrategy == AllocationStrategy::GROW_HEURISTICALLY) nextSize = size;
} else { } else {
if (moreSegments == nullptr) { if (moreSegments == nullptr) {
......
...@@ -282,6 +282,8 @@ private: ...@@ -282,6 +282,8 @@ private:
AllocationStrategy allocationStrategy; AllocationStrategy allocationStrategy;
bool ownFirstSegment; bool ownFirstSegment;
bool returnedFirstSegment;
void* firstSegment; void* firstSegment;
struct MoreSegments; struct MoreSegments;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment