262 lines
9.0 KiB
C++
262 lines
9.0 KiB
C++
|
|
/*
|
||
|
|
* Copyright (C) 2022 The Android Open Source Project
|
||
|
|
*
|
||
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
|
* you may not use this file except in compliance with the License.
|
||
|
|
* You may obtain a copy of the License at
|
||
|
|
*
|
||
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||
|
|
*
|
||
|
|
* Unless required by applicable law or agreed to in writing, software
|
||
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
|
* See the License for the specific language governing permissions and
|
||
|
|
* limitations under the License.
|
||
|
|
*/
|
||
|
|
|
||
|
|
#include "perfetto/base/build_config.h"
|
||
|
|
|
||
|
|
#if PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID)
|
||
|
|
|
||
|
|
#include "perfetto/base/logging.h"
|
||
|
|
#include "perfetto/ext/base/file_utils.h"
|
||
|
|
#include "perfetto/ext/base/pipe.h"
|
||
|
|
#include "perfetto/ext/base/scoped_file.h"
|
||
|
|
#include "perfetto/ext/base/string_utils.h"
|
||
|
|
#include "perfetto/ext/base/utils.h"
|
||
|
|
#include "perfetto/ext/tracing/core/commit_data_request.h"
|
||
|
|
#include "perfetto/ext/tracing/core/trace_packet.h"
|
||
|
|
#include "perfetto/ext/tracing/core/tracing_service.h"
|
||
|
|
#include "perfetto/protozero/scattered_heap_buffer.h"
|
||
|
|
#include "src/base/test/test_task_runner.h"
|
||
|
|
#include "src/base/test/utils.h"
|
||
|
|
#include "test/gtest_and_gmock.h"
|
||
|
|
#include "test/test_helper.h"
|
||
|
|
|
||
|
|
#include "protos/perfetto/config/power/android_power_config.pbzero.h"
|
||
|
|
#include "protos/perfetto/config/test_config.gen.h"
|
||
|
|
#include "protos/perfetto/config/trace_config.gen.h"
|
||
|
|
#include "protos/perfetto/trace/ftrace/ftrace.gen.h"
|
||
|
|
#include "protos/perfetto/trace/ftrace/ftrace_event.gen.h"
|
||
|
|
#include "protos/perfetto/trace/ftrace/ftrace_event_bundle.gen.h"
|
||
|
|
#include "protos/perfetto/trace/ftrace/ftrace_stats.gen.h"
|
||
|
|
#include "protos/perfetto/trace/perfetto/tracing_service_event.gen.h"
|
||
|
|
#include "protos/perfetto/trace/power/battery_counters.gen.h"
|
||
|
|
#include "protos/perfetto/trace/test_event.gen.h"
|
||
|
|
#include "protos/perfetto/trace/trace.gen.h"
|
||
|
|
#include "protos/perfetto/trace/trace_packet.gen.h"
|
||
|
|
#include "protos/perfetto/trace/trace_packet.pbzero.h"
|
||
|
|
#include "protos/perfetto/trace/trigger.gen.h"
|
||
|
|
|
||
|
|
#include "protos/perfetto/common/sys_stats_counters.gen.h"
|
||
|
|
#include "protos/perfetto/config/sys_stats/sys_stats_config.gen.h"
|
||
|
|
#include "protos/perfetto/trace/sys_stats/sys_stats.gen.h"
|
||
|
|
|
||
|
|
namespace perfetto {
|
||
|
|
|
||
|
|
namespace {
|
||
|
|
|
||
|
|
using ::testing::ContainsRegex;
|
||
|
|
using ::testing::Each;
|
||
|
|
using ::testing::ElementsAreArray;
|
||
|
|
using ::testing::HasSubstr;
|
||
|
|
using ::testing::Property;
|
||
|
|
using ::testing::SizeIs;
|
||
|
|
|
||
|
|
} // namespace
|
||
|
|
|
||
|
|
TEST(PerfettoAndroidIntegrationTest, TestKmemActivity) {
|
||
|
|
using C = protos::gen::VmstatCounters;
|
||
|
|
|
||
|
|
base::TestTaskRunner task_runner;
|
||
|
|
|
||
|
|
TestHelper helper(&task_runner);
|
||
|
|
|
||
|
|
helper.StartServiceIfRequired();
|
||
|
|
|
||
|
|
#if PERFETTO_BUILDFLAG(PERFETTO_START_DAEMONS)
|
||
|
|
ProbesProducerThread probes(GetTestProducerSockName());
|
||
|
|
probes.Connect();
|
||
|
|
#endif
|
||
|
|
|
||
|
|
auto* producer = helper.ConnectFakeProducer();
|
||
|
|
helper.ConnectConsumer();
|
||
|
|
helper.WaitForConsumerConnect();
|
||
|
|
helper.WaitForDataSourceConnected("linux.ftrace");
|
||
|
|
|
||
|
|
TraceConfig trace_config;
|
||
|
|
trace_config.add_buffers()->set_size_kb(1024);
|
||
|
|
trace_config.set_unique_session_name("kmem_activity_test");
|
||
|
|
|
||
|
|
auto* ftrace_ds_config = trace_config.add_data_sources()->mutable_config();
|
||
|
|
ftrace_ds_config->set_name("linux.ftrace");
|
||
|
|
protos::gen::FtraceConfig ftrace_config = CreateFtraceConfig({
|
||
|
|
"vmscan/mm_vmscan_kswapd_wake",
|
||
|
|
"vmscan/mm_vmscan_kswapd_sleep",
|
||
|
|
"vmscan/mm_vmscan_direct_reclaim_begin",
|
||
|
|
"vmscan/mm_vmscan_direct_reclaim_end",
|
||
|
|
"compaction/mm_compaction_begin",
|
||
|
|
"compaction/mm_compaction_end",
|
||
|
|
});
|
||
|
|
ftrace_ds_config->set_ftrace_config_raw(ftrace_config.SerializeAsString());
|
||
|
|
|
||
|
|
auto* sys_stats_ds_config = trace_config.add_data_sources()->mutable_config();
|
||
|
|
sys_stats_ds_config->set_name("linux.sys_stats");
|
||
|
|
protos::gen::SysStatsConfig sys_stats_config;
|
||
|
|
sys_stats_config.set_vmstat_period_ms(50);
|
||
|
|
std::vector<C> vmstat_counters = {
|
||
|
|
C::VMSTAT_NR_FREE_PAGES,
|
||
|
|
C::VMSTAT_NR_SLAB_RECLAIMABLE,
|
||
|
|
C::VMSTAT_NR_SLAB_UNRECLAIMABLE,
|
||
|
|
C::VMSTAT_NR_ACTIVE_FILE,
|
||
|
|
C::VMSTAT_NR_INACTIVE_FILE,
|
||
|
|
C::VMSTAT_NR_ACTIVE_ANON,
|
||
|
|
C::VMSTAT_NR_INACTIVE_ANON,
|
||
|
|
C::VMSTAT_WORKINGSET_REFAULT,
|
||
|
|
C::VMSTAT_WORKINGSET_ACTIVATE,
|
||
|
|
C::VMSTAT_NR_FILE_PAGES,
|
||
|
|
C::VMSTAT_PGPGIN,
|
||
|
|
C::VMSTAT_PGPGOUT,
|
||
|
|
C::VMSTAT_PSWPIN,
|
||
|
|
C::VMSTAT_PSWPOUT,
|
||
|
|
C::VMSTAT_PGSTEAL_KSWAPD_DMA,
|
||
|
|
C::VMSTAT_PGSTEAL_KSWAPD_NORMAL,
|
||
|
|
C::VMSTAT_PGSTEAL_KSWAPD_MOVABLE,
|
||
|
|
C::VMSTAT_PGSTEAL_DIRECT_DMA,
|
||
|
|
C::VMSTAT_PGSTEAL_DIRECT_NORMAL,
|
||
|
|
C::VMSTAT_PGSTEAL_DIRECT_MOVABLE,
|
||
|
|
C::VMSTAT_PGSCAN_KSWAPD_DMA,
|
||
|
|
C::VMSTAT_PGSCAN_KSWAPD_NORMAL,
|
||
|
|
C::VMSTAT_PGSCAN_KSWAPD_MOVABLE,
|
||
|
|
C::VMSTAT_PGSCAN_DIRECT_DMA,
|
||
|
|
C::VMSTAT_PGSCAN_DIRECT_NORMAL,
|
||
|
|
C::VMSTAT_PGSCAN_DIRECT_MOVABLE,
|
||
|
|
C::VMSTAT_COMPACT_MIGRATE_SCANNED,
|
||
|
|
C::VMSTAT_COMPACT_FREE_SCANNED,
|
||
|
|
};
|
||
|
|
for (const auto& counter : vmstat_counters) {
|
||
|
|
sys_stats_config.add_vmstat_counters(counter);
|
||
|
|
}
|
||
|
|
sys_stats_ds_config->set_sys_stats_config_raw(
|
||
|
|
sys_stats_config.SerializeAsString());
|
||
|
|
|
||
|
|
auto* trigger_cfg = trace_config.mutable_trigger_config();
|
||
|
|
trigger_cfg->set_trigger_mode(
|
||
|
|
protos::gen::TraceConfig::TriggerConfig::START_TRACING);
|
||
|
|
trigger_cfg->set_trigger_timeout_ms(15000);
|
||
|
|
auto* trigger = trigger_cfg->add_triggers();
|
||
|
|
trigger->set_name("kmem_activity");
|
||
|
|
// |stop_delay_ms| must be long enough that we can write the packets in
|
||
|
|
// before the trace finishes.
|
||
|
|
trigger->set_stop_delay_ms(1000);
|
||
|
|
|
||
|
|
helper.StartTracing(trace_config);
|
||
|
|
|
||
|
|
// Linearize with StartTracing. This ensures that the service has seen the
|
||
|
|
// StartTracing IPC and has armed the triggers.
|
||
|
|
helper.FlushAndWait(kDefaultTestTimeoutMs);
|
||
|
|
|
||
|
|
// Generating synthetic memory pressure to trigger kmem activity is
|
||
|
|
// inherently flaky on different devices. The same goes for writing
|
||
|
|
// /proc/sys/vm/compact_memory to trigger compaction, since compaction is
|
||
|
|
// only started if needed (even if explicitly triggered from proc).
|
||
|
|
// Trigger kmem activity using perfetto trigger.
|
||
|
|
producer->ActivateTrigger("kmem_activity");
|
||
|
|
|
||
|
|
helper.WaitForTracingDisabled();
|
||
|
|
|
||
|
|
helper.ReadData();
|
||
|
|
helper.WaitForReadData();
|
||
|
|
|
||
|
|
const auto& packets = helper.trace();
|
||
|
|
ASSERT_GT(packets.size(), 0u);
|
||
|
|
|
||
|
|
bool sys_stats_captured = false;
|
||
|
|
for (const auto& packet : packets) {
|
||
|
|
for (int ev = 0; ev < packet.ftrace_events().event_size(); ev++) {
|
||
|
|
auto ftrace_event =
|
||
|
|
packet.ftrace_events().event()[static_cast<size_t>(ev)];
|
||
|
|
ASSERT_TRUE(ftrace_event.has_mm_vmscan_kswapd_wake() ||
|
||
|
|
ftrace_event.has_mm_vmscan_kswapd_sleep() ||
|
||
|
|
ftrace_event.has_mm_vmscan_direct_reclaim_begin() ||
|
||
|
|
ftrace_event.has_mm_vmscan_direct_reclaim_end() ||
|
||
|
|
ftrace_event.has_mm_compaction_begin() ||
|
||
|
|
ftrace_event.has_mm_compaction_end());
|
||
|
|
}
|
||
|
|
|
||
|
|
if (packet.has_sys_stats()) {
|
||
|
|
sys_stats_captured = true;
|
||
|
|
const auto& sys_stats = packet.sys_stats();
|
||
|
|
const auto& vmstat = sys_stats.vmstat();
|
||
|
|
ASSERT_GT(vmstat.size(), 0u);
|
||
|
|
for (const auto& vmstat_value : vmstat) {
|
||
|
|
ASSERT_NE(std::find(vmstat_counters.begin(), vmstat_counters.end(),
|
||
|
|
vmstat_value.key()),
|
||
|
|
vmstat_counters.end());
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
// Don't explicitly check that ftrace events were captured, since this test
|
||
|
|
// doesn't rely on memory pressure.
|
||
|
|
ASSERT_TRUE(sys_stats_captured);
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST(PerfettoAndroidIntegrationTest, TestBatteryTracing) {
|
||
|
|
base::TestTaskRunner task_runner;
|
||
|
|
|
||
|
|
TestHelper helper(&task_runner);
|
||
|
|
helper.StartServiceIfRequired();
|
||
|
|
|
||
|
|
#if PERFETTO_BUILDFLAG(PERFETTO_START_DAEMONS)
|
||
|
|
ProbesProducerThread probes(GetTestProducerSockName());
|
||
|
|
probes.Connect();
|
||
|
|
#endif
|
||
|
|
|
||
|
|
helper.ConnectConsumer();
|
||
|
|
helper.WaitForConsumerConnect();
|
||
|
|
|
||
|
|
TraceConfig trace_config;
|
||
|
|
trace_config.add_buffers()->set_size_kb(128);
|
||
|
|
trace_config.set_duration_ms(3000);
|
||
|
|
|
||
|
|
auto* ds_config = trace_config.add_data_sources()->mutable_config();
|
||
|
|
ds_config->set_name("android.power");
|
||
|
|
ds_config->set_target_buffer(0);
|
||
|
|
|
||
|
|
using protos::pbzero::AndroidPowerConfig;
|
||
|
|
protozero::HeapBuffered<AndroidPowerConfig> power_config;
|
||
|
|
power_config->set_battery_poll_ms(250);
|
||
|
|
power_config->add_battery_counters(
|
||
|
|
AndroidPowerConfig::BATTERY_COUNTER_CHARGE);
|
||
|
|
power_config->add_battery_counters(
|
||
|
|
AndroidPowerConfig::BATTERY_COUNTER_CAPACITY_PERCENT);
|
||
|
|
ds_config->set_android_power_config_raw(power_config.SerializeAsString());
|
||
|
|
|
||
|
|
helper.StartTracing(trace_config);
|
||
|
|
helper.WaitForTracingDisabled();
|
||
|
|
|
||
|
|
helper.ReadData();
|
||
|
|
helper.WaitForReadData();
|
||
|
|
|
||
|
|
const auto& packets = helper.trace();
|
||
|
|
ASSERT_GT(packets.size(), 0u);
|
||
|
|
|
||
|
|
bool has_battery_packet = false;
|
||
|
|
for (const auto& packet : packets) {
|
||
|
|
if (!packet.has_battery())
|
||
|
|
continue;
|
||
|
|
has_battery_packet = true;
|
||
|
|
// Unfortunately we cannot make any assertions on the charge counter.
|
||
|
|
// On some devices it can reach negative values (b/64685329).
|
||
|
|
EXPECT_GE(packet.battery().capacity_percent(), 0.f);
|
||
|
|
EXPECT_LE(packet.battery().capacity_percent(), 100.f);
|
||
|
|
}
|
||
|
|
|
||
|
|
ASSERT_TRUE(has_battery_packet);
|
||
|
|
}
|
||
|
|
|
||
|
|
} // namespace perfetto
|
||
|
|
|
||
|
|
#endif // PERFETTO_OS_ANDROID
|