blob: 9d4e5437c091c948858bd270f18a65f54c1e247f [file] [log] [blame]
/*
* 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