2018-09-11 00:22:53 +00:00
|
|
|
//===- FDRRecords.cpp - Unit Tests for XRay FDR Record Loading ------------===//
|
|
|
|
//
|
2019-01-19 08:50:56 +00:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2018-09-11 00:22:53 +00:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "gmock/gmock.h"
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
|
|
|
#include "llvm/XRay/BlockIndexer.h"
|
|
|
|
#include "llvm/XRay/BlockPrinter.h"
|
|
|
|
#include "llvm/XRay/BlockVerifier.h"
|
|
|
|
#include "llvm/XRay/FDRLogBuilder.h"
|
|
|
|
#include "llvm/XRay/FDRRecords.h"
|
|
|
|
#include "llvm/XRay/RecordPrinter.h"
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
namespace xray {
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
using ::testing::Eq;
|
|
|
|
using ::testing::Not;
|
|
|
|
|
|
|
|
TEST(XRayFDRTest, BuilderAndBlockIndexer) {
|
|
|
|
// We recreate a single block of valid records, then ensure that we find all
|
|
|
|
// of them belonging in the same index. We do this for three blocks, and
|
|
|
|
// ensure we find the same records in the blocks we deduce.
|
|
|
|
auto Block0 = LogBuilder()
|
|
|
|
.add<BufferExtents>(100)
|
|
|
|
.add<NewBufferRecord>(1)
|
|
|
|
.add<WallclockRecord>(1, 1)
|
|
|
|
.add<PIDRecord>(1)
|
|
|
|
.add<FunctionRecord>(RecordTypes::ENTER, 1, 1)
|
|
|
|
.add<FunctionRecord>(RecordTypes::EXIT, 1, 100)
|
2018-11-07 04:37:42 +00:00
|
|
|
.add<CustomEventRecordV5>(1, 4, "XRAY")
|
|
|
|
.add<TypedEventRecord>(1, 4, 2, "XRAY")
|
2018-09-11 00:22:53 +00:00
|
|
|
.consume();
|
|
|
|
auto Block1 = LogBuilder()
|
|
|
|
.add<BufferExtents>(100)
|
|
|
|
.add<NewBufferRecord>(1)
|
|
|
|
.add<WallclockRecord>(1, 2)
|
|
|
|
.add<PIDRecord>(1)
|
|
|
|
.add<FunctionRecord>(RecordTypes::ENTER, 1, 1)
|
|
|
|
.add<FunctionRecord>(RecordTypes::EXIT, 1, 100)
|
2018-11-07 04:37:42 +00:00
|
|
|
.add<CustomEventRecordV5>(1, 4, "XRAY")
|
|
|
|
.add<TypedEventRecord>(1, 4, 2, "XRAY")
|
2018-09-11 00:22:53 +00:00
|
|
|
.consume();
|
|
|
|
auto Block2 = LogBuilder()
|
|
|
|
.add<BufferExtents>(100)
|
|
|
|
.add<NewBufferRecord>(2)
|
|
|
|
.add<WallclockRecord>(1, 3)
|
|
|
|
.add<PIDRecord>(1)
|
|
|
|
.add<FunctionRecord>(RecordTypes::ENTER, 1, 1)
|
|
|
|
.add<FunctionRecord>(RecordTypes::EXIT, 1, 100)
|
2018-11-07 04:37:42 +00:00
|
|
|
.add<CustomEventRecordV5>(1, 4, "XRAY")
|
|
|
|
.add<TypedEventRecord>(1, 4, 2, "XRAY")
|
2018-09-11 00:22:53 +00:00
|
|
|
.consume();
|
|
|
|
BlockIndexer::Index Index;
|
|
|
|
BlockIndexer Indexer(Index);
|
|
|
|
for (auto B : {std::ref(Block0), std::ref(Block1), std::ref(Block2)}) {
|
|
|
|
for (auto &R : B.get())
|
|
|
|
ASSERT_FALSE(errorToBool(R->apply(Indexer)));
|
|
|
|
ASSERT_FALSE(errorToBool(Indexer.flush()));
|
|
|
|
}
|
|
|
|
|
|
|
|
// We have two threads worth of blocks.
|
|
|
|
ASSERT_THAT(Index.size(), Eq(2u));
|
|
|
|
auto T1Blocks = Index.find({1, 1});
|
|
|
|
ASSERT_THAT(T1Blocks, Not(Eq(Index.end())));
|
|
|
|
ASSERT_THAT(T1Blocks->second.size(), Eq(2u));
|
|
|
|
auto T2Blocks = Index.find({1, 2});
|
|
|
|
ASSERT_THAT(T2Blocks, Not(Eq(Index.end())));
|
|
|
|
ASSERT_THAT(T2Blocks->second.size(), Eq(1u));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(XRayFDRTest, BuilderAndBlockVerifier) {
|
|
|
|
auto Block = LogBuilder()
|
|
|
|
.add<BufferExtents>(48)
|
|
|
|
.add<NewBufferRecord>(1)
|
|
|
|
.add<WallclockRecord>(1, 1)
|
|
|
|
.add<PIDRecord>(1)
|
2018-09-11 06:36:51 +00:00
|
|
|
.add<NewCPUIDRecord>(1, 2)
|
2018-09-11 00:22:53 +00:00
|
|
|
.consume();
|
|
|
|
BlockVerifier Verifier;
|
|
|
|
for (auto &R : Block)
|
|
|
|
ASSERT_FALSE(errorToBool(R->apply(Verifier)));
|
|
|
|
ASSERT_FALSE(errorToBool(Verifier.verify()));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(XRayFDRTest, IndexAndVerifyBlocks) {
|
|
|
|
auto Block0 = LogBuilder()
|
|
|
|
.add<BufferExtents>(64)
|
|
|
|
.add<NewBufferRecord>(1)
|
|
|
|
.add<WallclockRecord>(1, 1)
|
|
|
|
.add<PIDRecord>(1)
|
2018-09-11 06:36:51 +00:00
|
|
|
.add<NewCPUIDRecord>(1, 2)
|
2018-09-11 00:22:53 +00:00
|
|
|
.add<FunctionRecord>(RecordTypes::ENTER, 1, 1)
|
|
|
|
.add<FunctionRecord>(RecordTypes::EXIT, 1, 100)
|
2018-11-07 04:37:42 +00:00
|
|
|
.add<CustomEventRecordV5>(1, 4, "XRAY")
|
|
|
|
.add<TypedEventRecord>(1, 4, 2, "XRAY")
|
2018-09-11 00:22:53 +00:00
|
|
|
.consume();
|
|
|
|
auto Block1 = LogBuilder()
|
|
|
|
.add<BufferExtents>(64)
|
|
|
|
.add<NewBufferRecord>(1)
|
|
|
|
.add<WallclockRecord>(1, 1)
|
|
|
|
.add<PIDRecord>(1)
|
2018-09-11 06:36:51 +00:00
|
|
|
.add<NewCPUIDRecord>(1, 2)
|
2018-09-11 00:22:53 +00:00
|
|
|
.add<FunctionRecord>(RecordTypes::ENTER, 1, 1)
|
|
|
|
.add<FunctionRecord>(RecordTypes::EXIT, 1, 100)
|
2018-11-07 04:37:42 +00:00
|
|
|
.add<CustomEventRecordV5>(1, 4, "XRAY")
|
|
|
|
.add<TypedEventRecord>(1, 4, 2, "XRAY")
|
2018-09-11 00:22:53 +00:00
|
|
|
.consume();
|
|
|
|
auto Block2 = LogBuilder()
|
|
|
|
.add<BufferExtents>(64)
|
|
|
|
.add<NewBufferRecord>(1)
|
|
|
|
.add<WallclockRecord>(1, 1)
|
|
|
|
.add<PIDRecord>(1)
|
2018-09-11 06:36:51 +00:00
|
|
|
.add<NewCPUIDRecord>(1, 2)
|
2018-09-11 00:22:53 +00:00
|
|
|
.add<FunctionRecord>(RecordTypes::ENTER, 1, 1)
|
|
|
|
.add<FunctionRecord>(RecordTypes::EXIT, 1, 100)
|
2018-11-07 04:37:42 +00:00
|
|
|
.add<CustomEventRecordV5>(1, 4, "XRAY")
|
|
|
|
.add<TypedEventRecord>(1, 4, 2, "XRAY")
|
2018-09-11 00:22:53 +00:00
|
|
|
.consume();
|
|
|
|
|
|
|
|
// First, index the records in different blocks.
|
|
|
|
BlockIndexer::Index Index;
|
|
|
|
BlockIndexer Indexer(Index);
|
|
|
|
for (auto B : {std::ref(Block0), std::ref(Block1), std::ref(Block2)}) {
|
|
|
|
for (auto &R : B.get())
|
|
|
|
ASSERT_FALSE(errorToBool(R->apply(Indexer)));
|
|
|
|
ASSERT_FALSE(errorToBool(Indexer.flush()));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Next, verify that each block is consistently defined.
|
|
|
|
BlockVerifier Verifier;
|
|
|
|
for (auto &ProcessThreadBlocks : Index) {
|
|
|
|
auto &Blocks = ProcessThreadBlocks.second;
|
|
|
|
for (auto &B : Blocks) {
|
|
|
|
for (auto *R : B.Records)
|
|
|
|
ASSERT_FALSE(errorToBool(R->apply(Verifier)));
|
|
|
|
ASSERT_FALSE(errorToBool(Verifier.verify()));
|
|
|
|
Verifier.reset();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Then set up the printing mechanisms.
|
|
|
|
std::string Output;
|
|
|
|
raw_string_ostream OS(Output);
|
|
|
|
RecordPrinter RP(OS);
|
|
|
|
BlockPrinter BP(OS, RP);
|
|
|
|
for (auto &ProcessThreadBlocks : Index) {
|
|
|
|
auto &Blocks = ProcessThreadBlocks.second;
|
|
|
|
for (auto &B : Blocks) {
|
|
|
|
for (auto *R : B.Records)
|
|
|
|
ASSERT_FALSE(errorToBool(R->apply(BP)));
|
|
|
|
BP.reset();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
OS.flush();
|
|
|
|
EXPECT_THAT(Output, Not(Eq("")));
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
} // namespace xray
|
|
|
|
} // namespace llvm
|