// 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; }