169 lines
4.3 KiB
C
169 lines
4.3 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* RISC-V performance counter support.
|
|
*
|
|
* Copyright (C) 2021 Western Digital Corporation or its affiliates.
|
|
*
|
|
* This implementation is based on old RISC-V perf and ARM perf event code
|
|
* which are in turn based on sparc64 and x86 code.
|
|
*/
|
|
|
|
#include <linux/mod_devicetable.h>
|
|
#include <linux/perf/riscv_pmu.h>
|
|
#include <linux/platform_device.h>
|
|
|
|
#define RISCV_PMU_LEGACY_CYCLE 0
|
|
#define RISCV_PMU_LEGACY_INSTRET 2
|
|
|
|
static bool pmu_init_done;
|
|
|
|
static int pmu_legacy_ctr_get_idx(struct perf_event *event)
|
|
{
|
|
struct perf_event_attr *attr = &event->attr;
|
|
|
|
if (event->attr.type != PERF_TYPE_HARDWARE)
|
|
return -EOPNOTSUPP;
|
|
if (attr->config == PERF_COUNT_HW_CPU_CYCLES)
|
|
return RISCV_PMU_LEGACY_CYCLE;
|
|
else if (attr->config == PERF_COUNT_HW_INSTRUCTIONS)
|
|
return RISCV_PMU_LEGACY_INSTRET;
|
|
else
|
|
return -EOPNOTSUPP;
|
|
}
|
|
|
|
/* For legacy config & counter index are same */
|
|
static int pmu_legacy_event_map(struct perf_event *event, u64 *config)
|
|
{
|
|
return pmu_legacy_ctr_get_idx(event);
|
|
}
|
|
|
|
static u64 pmu_legacy_read_ctr(struct perf_event *event)
|
|
{
|
|
struct hw_perf_event *hwc = &event->hw;
|
|
int idx = hwc->idx;
|
|
u64 val;
|
|
|
|
if (idx == RISCV_PMU_LEGACY_CYCLE) {
|
|
val = riscv_pmu_ctr_read_csr(CSR_CYCLE);
|
|
if (IS_ENABLED(CONFIG_32BIT))
|
|
val = (u64)riscv_pmu_ctr_read_csr(CSR_CYCLEH) << 32 | val;
|
|
} else if (idx == RISCV_PMU_LEGACY_INSTRET) {
|
|
val = riscv_pmu_ctr_read_csr(CSR_INSTRET);
|
|
if (IS_ENABLED(CONFIG_32BIT))
|
|
val = ((u64)riscv_pmu_ctr_read_csr(CSR_INSTRETH)) << 32 | val;
|
|
} else
|
|
return 0;
|
|
|
|
return val;
|
|
}
|
|
|
|
static void pmu_legacy_ctr_start(struct perf_event *event, u64 ival)
|
|
{
|
|
struct hw_perf_event *hwc = &event->hw;
|
|
u64 initial_val = pmu_legacy_read_ctr(event);
|
|
|
|
/**
|
|
* The legacy method doesn't really have a start/stop method.
|
|
* It also can not update the counter with a initial value.
|
|
* But we still need to set the prev_count so that read() can compute
|
|
* the delta. Just use the current counter value to set the prev_count.
|
|
*/
|
|
local64_set(&hwc->prev_count, initial_val);
|
|
}
|
|
|
|
static uint8_t pmu_legacy_csr_index(struct perf_event *event)
|
|
{
|
|
return event->hw.idx;
|
|
}
|
|
|
|
static void pmu_legacy_event_mapped(struct perf_event *event, struct mm_struct *mm)
|
|
{
|
|
if (event->attr.config != PERF_COUNT_HW_CPU_CYCLES &&
|
|
event->attr.config != PERF_COUNT_HW_INSTRUCTIONS)
|
|
return;
|
|
|
|
event->hw.flags |= PERF_EVENT_FLAG_USER_READ_CNT;
|
|
}
|
|
|
|
static void pmu_legacy_event_unmapped(struct perf_event *event, struct mm_struct *mm)
|
|
{
|
|
if (event->attr.config != PERF_COUNT_HW_CPU_CYCLES &&
|
|
event->attr.config != PERF_COUNT_HW_INSTRUCTIONS)
|
|
return;
|
|
|
|
event->hw.flags &= ~PERF_EVENT_FLAG_USER_READ_CNT;
|
|
}
|
|
|
|
/*
|
|
* This is just a simple implementation to allow legacy implementations
|
|
* compatible with new RISC-V PMU driver framework.
|
|
* This driver only allows reading two counters i.e CYCLE & INSTRET.
|
|
* However, it can not start or stop the counter. Thus, it is not very useful
|
|
* will be removed in future.
|
|
*/
|
|
static void pmu_legacy_init(struct riscv_pmu *pmu)
|
|
{
|
|
pr_info("Legacy PMU implementation is available\n");
|
|
|
|
pmu->cmask = BIT(RISCV_PMU_LEGACY_CYCLE) |
|
|
BIT(RISCV_PMU_LEGACY_INSTRET);
|
|
pmu->ctr_start = pmu_legacy_ctr_start;
|
|
pmu->ctr_stop = NULL;
|
|
pmu->event_map = pmu_legacy_event_map;
|
|
pmu->ctr_get_idx = pmu_legacy_ctr_get_idx;
|
|
pmu->ctr_get_width = NULL;
|
|
pmu->ctr_clear_idx = NULL;
|
|
pmu->ctr_read = pmu_legacy_read_ctr;
|
|
pmu->event_mapped = pmu_legacy_event_mapped;
|
|
pmu->event_unmapped = pmu_legacy_event_unmapped;
|
|
pmu->csr_index = pmu_legacy_csr_index;
|
|
|
|
perf_pmu_register(&pmu->pmu, "cpu", PERF_TYPE_RAW);
|
|
}
|
|
|
|
static int pmu_legacy_device_probe(struct platform_device *pdev)
|
|
{
|
|
struct riscv_pmu *pmu = NULL;
|
|
|
|
pmu = riscv_pmu_alloc();
|
|
if (!pmu)
|
|
return -ENOMEM;
|
|
pmu_legacy_init(pmu);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct platform_driver pmu_legacy_driver = {
|
|
.probe = pmu_legacy_device_probe,
|
|
.driver = {
|
|
.name = RISCV_PMU_LEGACY_PDEV_NAME,
|
|
},
|
|
};
|
|
|
|
static int __init riscv_pmu_legacy_devinit(void)
|
|
{
|
|
int ret;
|
|
struct platform_device *pdev;
|
|
|
|
if (likely(pmu_init_done))
|
|
return 0;
|
|
|
|
ret = platform_driver_register(&pmu_legacy_driver);
|
|
if (ret)
|
|
return ret;
|
|
|
|
pdev = platform_device_register_simple(RISCV_PMU_LEGACY_PDEV_NAME, -1, NULL, 0);
|
|
if (IS_ERR(pdev)) {
|
|
platform_driver_unregister(&pmu_legacy_driver);
|
|
return PTR_ERR(pdev);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
late_initcall(riscv_pmu_legacy_devinit);
|
|
|
|
void riscv_pmu_legacy_skip_init(void)
|
|
{
|
|
pmu_init_done = true;
|
|
}
|