/* * Copyright © 2019-2021 Collabora, Ltd. * Author: Antonio Caggiano * Author: Rohan Garg * Author: Robert Beckett * * SPDX-License-Identifier: MIT */ #include "pan_pps_driver.h" #include #include #include #include #include #include #include #include namespace pps { PanfrostDriver::PanfrostDriver() { } PanfrostDriver::~PanfrostDriver() { } uint64_t PanfrostDriver::get_min_sampling_period_ns() { return 1000000; } uint32_t find_id_within_group(uint32_t counter_id, const struct panfrost_perf_config *cfg) { for (uint32_t cat_id = 0; cat_id < cfg->n_categories; ++cat_id) { const struct panfrost_perf_category *cat = &cfg->categories[cat_id]; if (counter_id < cat->n_counters) { break; } counter_id -= cat->n_counters; } return counter_id; } std::pair, std::vector> PanfrostDriver::create_available_counters(const PanfrostPerf &perf) { std::pair, std::vector> ret; auto &[groups, counters] = ret; size_t cid = 0; for (uint32_t gid = 0; gid < perf.perf->cfg->n_categories; ++gid) { const auto &category = perf.perf->cfg->categories[gid]; CounterGroup group = {}; group.id = gid; group.name = category.name; for (; cid < category.n_counters; ++cid) { Counter counter = {}; counter.id = cid; counter.group = gid; uint32_t id_within_group = find_id_within_group(cid, perf.perf->cfg); counter.name = category.counters[id_within_group].name; counter.set_getter([](const Counter &c, const Driver &d) { auto &pan_driver = PanfrostDriver::into(d); struct panfrost_perf *perf = pan_driver.perf->perf; uint32_t id_within_group = find_id_within_group(c.id, perf->cfg); const auto counter = &perf->cfg->categories[c.group].counters[id_within_group]; return int64_t(panfrost_perf_counter_read(counter, perf)); }); group.counters.push_back(cid); counters.emplace_back(counter); } groups.push_back(group); } return ret; } bool PanfrostDriver::init_perfcnt() { if (!dev) { dev = std::make_unique(drm_device.fd); } if (!perf) { perf = std::make_unique(*dev); } if (groups.empty() && counters.empty()) { std::tie(groups, counters) = create_available_counters(*perf); } return true; } void PanfrostDriver::enable_counter(const uint32_t counter_id) { enabled_counters.push_back(counters[counter_id]); } void PanfrostDriver::enable_all_counters() { enabled_counters.resize(counters.size()); for (size_t i = 0; i < counters.size(); ++i) { enabled_counters[i] = counters[i]; } } void PanfrostDriver::enable_perfcnt(const uint64_t /* sampling_period_ns */) { auto res = perf->enable(); if (!check(res, "Failed to enable performance counters")) { if (res == -ENOSYS) { PERFETTO_FATAL("Please enable unstable ioctls with: modprobe panfrost unstable_ioctls=1"); } PERFETTO_FATAL("Please verify graphics card"); } } bool PanfrostDriver::dump_perfcnt() { last_dump_ts = perfetto::base::GetBootTimeNs().count(); // Dump performance counters to buffer if (!check(perf->dump(), "Failed to dump performance counters")) { PERFETTO_ELOG("Skipping sample"); return false; } return true; } uint64_t PanfrostDriver::next() { auto ret = last_dump_ts; last_dump_ts = 0; return ret; } void PanfrostDriver::disable_perfcnt() { perf->disable(); perf.reset(); dev.reset(); groups.clear(); counters.clear(); enabled_counters.clear(); } uint32_t PanfrostDriver::gpu_clock_id() const { return perfetto::protos::pbzero::BUILTIN_CLOCK_BOOTTIME; } uint64_t PanfrostDriver::gpu_timestamp() const { return perfetto::base::GetBootTimeNs().count(); } } // namespace pps