blob: e7549216ede1beac2e3b5148fb3c4cdd79f20887 [file] [log] [blame]
// Copyright 2018 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <memory>
#include <string>
#include <utility>
#include "base/run_loop.h"
#include "base/test/scoped_task_environment.h"
#include "mojo/public/cpp/bindings/binding.h"
#include "services/tracing/perfetto/perfetto_service.h"
#include "services/tracing/perfetto/producer_host.h"
#include "services/tracing/perfetto/test_utils.h"
#include "services/tracing/public/cpp/perfetto/producer_client.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace tracing {
namespace {
class PerfettoIntegrationTest : public testing::Test {
public:
void SetUp() override {
perfetto_service_ = std::make_unique<PerfettoService>(
base::SequencedTaskRunnerHandle::Get());
// The actual Perfetto service is created async on the given task_runner;
// wait until that's done.
RunUntilIdle();
ProducerClient::ResetTaskRunnerForTesting();
}
void TearDown() override { perfetto_service_.reset(); }
PerfettoService* perfetto_service() const { return perfetto_service_.get(); }
void RunUntilIdle() { scoped_task_environment_.RunUntilIdle(); }
private:
std::unique_ptr<PerfettoService> perfetto_service_;
base::test::ScopedTaskEnvironment scoped_task_environment_;
};
TEST_F(PerfettoIntegrationTest, ProducerDatasourceInitialized) {
auto dummy_client =
std::make_unique<MockProducerClient>(0 /* send_packet_count */);
base::RunLoop producer_initialized_runloop;
auto new_producer = std::make_unique<MockProducerHost>(
kPerfettoTestDataSourceName, perfetto_service()->GetService(),
dummy_client.get(), producer_initialized_runloop.QuitClosure());
producer_initialized_runloop.Run();
ProducerClient::DeleteSoonForTesting(std::move(dummy_client));
}
TEST_F(PerfettoIntegrationTest, ClientEnabledAndDisabled) {
base::RunLoop on_trace_packets;
MockConsumer consumer(kPerfettoTestDataSourceName,
perfetto_service()->GetService(),
[&on_trace_packets](bool has_more) {
EXPECT_FALSE(has_more);
on_trace_packets.Quit();
});
base::RunLoop client_enabled_callback;
base::RunLoop client_disabled_callback;
auto client = std::make_unique<MockProducerClient>(
0 /* send_packet_count */, client_enabled_callback.QuitClosure(),
client_disabled_callback.QuitClosure());
auto producer = std::make_unique<MockProducerHost>(
kPerfettoTestDataSourceName, perfetto_service()->GetService(),
client.get());
client_enabled_callback.Run();
RunUntilIdle();
consumer.StopTracing();
client_disabled_callback.Run();
on_trace_packets.Run();
EXPECT_EQ(0u, consumer.received_packets());
ProducerClient::DeleteSoonForTesting(std::move(client));
}
TEST_F(PerfettoIntegrationTest, PacketsEndToEndProducerFirst) {
const size_t kNumPackets = 10;
base::RunLoop client_enabled_callback;
base::RunLoop client_disabled_callback;
auto client = std::make_unique<MockProducerClient>(
kNumPackets, client_enabled_callback.QuitClosure(),
client_disabled_callback.QuitClosure());
auto producer = std::make_unique<MockProducerHost>(
kPerfettoTestDataSourceName, perfetto_service()->GetService(),
client.get());
base::RunLoop no_more_packets_runloop;
MockConsumer consumer(kPerfettoTestDataSourceName,
perfetto_service()->GetService(),
[&no_more_packets_runloop](bool has_more) {
if (!has_more) {
no_more_packets_runloop.Quit();
}
});
client_enabled_callback.Run();
RunUntilIdle();
consumer.StopTracing();
client_disabled_callback.Run();
no_more_packets_runloop.Run();
EXPECT_EQ(kNumPackets, consumer.received_packets());
ProducerClient::DeleteSoonForTesting(std::move(client));
}
TEST_F(PerfettoIntegrationTest, PacketsEndToEndConsumerFirst) {
const size_t kNumPackets = 10;
base::RunLoop no_more_packets_runloop;
MockConsumer consumer(kPerfettoTestDataSourceName,
perfetto_service()->GetService(),
[&no_more_packets_runloop](bool has_more) {
if (!has_more) {
no_more_packets_runloop.Quit();
}
});
base::RunLoop client_enabled_callback;
auto client = std::make_unique<MockProducerClient>(
kNumPackets, client_enabled_callback.QuitClosure());
auto new_producer = std::make_unique<MockProducerHost>(
kPerfettoTestDataSourceName, perfetto_service()->GetService(),
client.get());
client_enabled_callback.Run();
RunUntilIdle();
consumer.StopTracing();
no_more_packets_runloop.Run();
EXPECT_EQ(kNumPackets, consumer.received_packets());
ProducerClient::DeleteSoonForTesting(std::move(client));
}
TEST_F(PerfettoIntegrationTest, CommitDataRequestIsMaybeComplete) {
const size_t kNumPackets = 100;
base::RunLoop no_more_packets_runloop;
MockConsumer consumer(kPerfettoTestDataSourceName,
perfetto_service()->GetService(),
[&no_more_packets_runloop](bool has_more) {
if (!has_more) {
no_more_packets_runloop.Quit();
}
});
base::RunLoop client_enabled_callback;
auto client = std::make_unique<MockProducerClient>(
kNumPackets, client_enabled_callback.QuitClosure());
auto new_producer = std::make_unique<MockProducerHost>(
kPerfettoTestDataSourceName, perfetto_service()->GetService(),
client.get());
client_enabled_callback.Run();
base::RunLoop wait_for_packet_write;
client->GetTaskRunner()->PostTaskAndReply(
FROM_HERE,
base::BindOnce(&TestDataSource::WritePacketBigly,
base::Unretained(client->data_source())),
wait_for_packet_write.QuitClosure());
wait_for_packet_write.Run();
RunUntilIdle();
consumer.StopTracing();
no_more_packets_runloop.Run();
EXPECT_EQ(client->all_client_commit_data_requests(),
new_producer->all_host_commit_data_requests());
ProducerClient::DeleteSoonForTesting(std::move(client));
}
TEST_F(PerfettoIntegrationTest, TracingRestarted) {
const size_t kNumPackets = 10;
base::RunLoop no_more_packets_runloop;
MockConsumer consumer(kPerfettoTestDataSourceName,
perfetto_service()->GetService(),
[&no_more_packets_runloop](bool has_more) {
if (!has_more) {
no_more_packets_runloop.Quit();
}
});
base::RunLoop client_enabled_callback;
auto client = std::make_unique<MockProducerClient>(
kNumPackets, client_enabled_callback.QuitClosure());
auto new_producer = std::make_unique<MockProducerHost>(
kPerfettoTestDataSourceName, perfetto_service()->GetService(),
client.get());
client_enabled_callback.Run();
RunUntilIdle();
perfetto::SharedMemory* first_session_shm = client->shared_memory();
consumer.StopTracing();
no_more_packets_runloop.Run();
EXPECT_EQ(kNumPackets, consumer.received_packets());
consumer.FreeBuffers();
base::RunLoop client_reenabled_callback;
client->SetAgentEnabledCallback(client_reenabled_callback.QuitClosure());
consumer.StartTracing();
client_reenabled_callback.Run();
RunUntilIdle();
// We should still be using the same shared memory.
EXPECT_EQ(first_session_shm, client->shared_memory());
base::RunLoop client_redisabled_callback;
client->SetAgentDisabledCallback(client_redisabled_callback.QuitClosure());
consumer.StopTracing();
client_redisabled_callback.Run();
EXPECT_EQ(kNumPackets * 2, consumer.received_packets());
ProducerClient::DeleteSoonForTesting(std::move(client));
}
TEST_F(PerfettoIntegrationTest, NoPacketsReceivedOnWrongSourceName) {
const size_t kNumPackets = 10;
base::RunLoop client_enabled_callback;
base::RunLoop client_disabled_callback;
auto client = std::make_unique<MockProducerClient>(
kNumPackets, client_enabled_callback.QuitClosure(),
client_disabled_callback.QuitClosure());
base::RunLoop producer_initialized_runloop;
auto new_producer = std::make_unique<MockProducerHost>(
"fake_data_source", perfetto_service()->GetService(), client.get());
base::RunLoop no_more_packets_runloop;
MockConsumer consumer(kPerfettoTestDataSourceName,
perfetto_service()->GetService(),
[&no_more_packets_runloop](bool has_more) {
if (!has_more) {
no_more_packets_runloop.Quit();
}
});
RunUntilIdle();
consumer.StopTracing();
no_more_packets_runloop.Run();
EXPECT_EQ(0u, consumer.received_packets());
ProducerClient::DeleteSoonForTesting(std::move(client));
}
TEST_F(PerfettoIntegrationTest,
DifferentSharedMemoryBuffersForDifferentAgents) {
base::RunLoop client1_enabled_callback;
base::RunLoop client2_enabled_callback;
auto client1 = std::make_unique<MockProducerClient>(
0 /* send_packet_count */, client1_enabled_callback.QuitClosure());
auto client2 = std::make_unique<MockProducerClient>(
0 /* send_packet_count */, client2_enabled_callback.QuitClosure());
auto producer1 = std::make_unique<MockProducerHost>(
kPerfettoTestDataSourceName, perfetto_service()->GetService(),
client1.get());
auto producer2 = std::make_unique<MockProducerHost>(
kPerfettoTestDataSourceName, perfetto_service()->GetService(),
client2.get());
MockConsumer consumer(kPerfettoTestDataSourceName,
perfetto_service()->GetService(), nullptr);
client1_enabled_callback.Run();
client2_enabled_callback.Run();
EXPECT_TRUE(client1->shared_memory());
EXPECT_TRUE(client2->shared_memory());
EXPECT_NE(client1->shared_memory(), client2->shared_memory());
ProducerClient::DeleteSoonForTesting(std::move(client1));
ProducerClient::DeleteSoonForTesting(std::move(client2));
}
} // namespace
} // namespace tracing