162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * ARM CoreSight Architecture PMU driver.
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * This driver adds support for uncore PMU based on ARM CoreSight Performance
662306a36Sopenharmony_ci * Monitoring Unit Architecture. The PMU is accessible via MMIO registers and
762306a36Sopenharmony_ci * like other uncore PMUs, it does not support process specific events and
862306a36Sopenharmony_ci * cannot be used in sampling mode.
962306a36Sopenharmony_ci *
1062306a36Sopenharmony_ci * This code is based on other uncore PMUs like ARM DSU PMU. It provides a
1162306a36Sopenharmony_ci * generic implementation to operate the PMU according to CoreSight PMU
1262306a36Sopenharmony_ci * architecture and ACPI ARM PMU table (APMT) documents below:
1362306a36Sopenharmony_ci *   - ARM CoreSight PMU architecture document number: ARM IHI 0091 A.a-00bet0.
1462306a36Sopenharmony_ci *   - APMT document number: ARM DEN0117.
1562306a36Sopenharmony_ci *
1662306a36Sopenharmony_ci * The user should refer to the vendor technical documentation to get details
1762306a36Sopenharmony_ci * about the supported events.
1862306a36Sopenharmony_ci *
1962306a36Sopenharmony_ci * Copyright (c) 2022, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
2062306a36Sopenharmony_ci *
2162306a36Sopenharmony_ci */
2262306a36Sopenharmony_ci
2362306a36Sopenharmony_ci#include <linux/acpi.h>
2462306a36Sopenharmony_ci#include <linux/cacheinfo.h>
2562306a36Sopenharmony_ci#include <linux/ctype.h>
2662306a36Sopenharmony_ci#include <linux/interrupt.h>
2762306a36Sopenharmony_ci#include <linux/io-64-nonatomic-lo-hi.h>
2862306a36Sopenharmony_ci#include <linux/module.h>
2962306a36Sopenharmony_ci#include <linux/perf_event.h>
3062306a36Sopenharmony_ci#include <linux/platform_device.h>
3162306a36Sopenharmony_ci
3262306a36Sopenharmony_ci#include "arm_cspmu.h"
3362306a36Sopenharmony_ci#include "nvidia_cspmu.h"
3462306a36Sopenharmony_ci
3562306a36Sopenharmony_ci#define PMUNAME "arm_cspmu"
3662306a36Sopenharmony_ci#define DRVNAME "arm-cs-arch-pmu"
3762306a36Sopenharmony_ci
3862306a36Sopenharmony_ci#define ARM_CSPMU_CPUMASK_ATTR(_name, _config)			\
3962306a36Sopenharmony_ci	ARM_CSPMU_EXT_ATTR(_name, arm_cspmu_cpumask_show,	\
4062306a36Sopenharmony_ci				(unsigned long)_config)
4162306a36Sopenharmony_ci
4262306a36Sopenharmony_ci/*
4362306a36Sopenharmony_ci * CoreSight PMU Arch register offsets.
4462306a36Sopenharmony_ci */
4562306a36Sopenharmony_ci#define PMEVCNTR_LO					0x0
4662306a36Sopenharmony_ci#define PMEVCNTR_HI					0x4
4762306a36Sopenharmony_ci#define PMEVTYPER					0x400
4862306a36Sopenharmony_ci#define PMCCFILTR					0x47C
4962306a36Sopenharmony_ci#define PMEVFILTR					0xA00
5062306a36Sopenharmony_ci#define PMCNTENSET					0xC00
5162306a36Sopenharmony_ci#define PMCNTENCLR					0xC20
5262306a36Sopenharmony_ci#define PMINTENSET					0xC40
5362306a36Sopenharmony_ci#define PMINTENCLR					0xC60
5462306a36Sopenharmony_ci#define PMOVSCLR					0xC80
5562306a36Sopenharmony_ci#define PMOVSSET					0xCC0
5662306a36Sopenharmony_ci#define PMCFGR						0xE00
5762306a36Sopenharmony_ci#define PMCR						0xE04
5862306a36Sopenharmony_ci#define PMIIDR						0xE08
5962306a36Sopenharmony_ci
6062306a36Sopenharmony_ci/* PMCFGR register field */
6162306a36Sopenharmony_ci#define PMCFGR_NCG					GENMASK(31, 28)
6262306a36Sopenharmony_ci#define PMCFGR_HDBG					BIT(24)
6362306a36Sopenharmony_ci#define PMCFGR_TRO					BIT(23)
6462306a36Sopenharmony_ci#define PMCFGR_SS					BIT(22)
6562306a36Sopenharmony_ci#define PMCFGR_FZO					BIT(21)
6662306a36Sopenharmony_ci#define PMCFGR_MSI					BIT(20)
6762306a36Sopenharmony_ci#define PMCFGR_UEN					BIT(19)
6862306a36Sopenharmony_ci#define PMCFGR_NA					BIT(17)
6962306a36Sopenharmony_ci#define PMCFGR_EX					BIT(16)
7062306a36Sopenharmony_ci#define PMCFGR_CCD					BIT(15)
7162306a36Sopenharmony_ci#define PMCFGR_CC					BIT(14)
7262306a36Sopenharmony_ci#define PMCFGR_SIZE					GENMASK(13, 8)
7362306a36Sopenharmony_ci#define PMCFGR_N					GENMASK(7, 0)
7462306a36Sopenharmony_ci
7562306a36Sopenharmony_ci/* PMCR register field */
7662306a36Sopenharmony_ci#define PMCR_TRO					BIT(11)
7762306a36Sopenharmony_ci#define PMCR_HDBG					BIT(10)
7862306a36Sopenharmony_ci#define PMCR_FZO					BIT(9)
7962306a36Sopenharmony_ci#define PMCR_NA						BIT(8)
8062306a36Sopenharmony_ci#define PMCR_DP						BIT(5)
8162306a36Sopenharmony_ci#define PMCR_X						BIT(4)
8262306a36Sopenharmony_ci#define PMCR_D						BIT(3)
8362306a36Sopenharmony_ci#define PMCR_C						BIT(2)
8462306a36Sopenharmony_ci#define PMCR_P						BIT(1)
8562306a36Sopenharmony_ci#define PMCR_E						BIT(0)
8662306a36Sopenharmony_ci
8762306a36Sopenharmony_ci/* Each SET/CLR register supports up to 32 counters. */
8862306a36Sopenharmony_ci#define ARM_CSPMU_SET_CLR_COUNTER_SHIFT		5
8962306a36Sopenharmony_ci#define ARM_CSPMU_SET_CLR_COUNTER_NUM		\
9062306a36Sopenharmony_ci	(1 << ARM_CSPMU_SET_CLR_COUNTER_SHIFT)
9162306a36Sopenharmony_ci
9262306a36Sopenharmony_ci/* Convert counter idx into SET/CLR register number. */
9362306a36Sopenharmony_ci#define COUNTER_TO_SET_CLR_ID(idx)			\
9462306a36Sopenharmony_ci	(idx >> ARM_CSPMU_SET_CLR_COUNTER_SHIFT)
9562306a36Sopenharmony_ci
9662306a36Sopenharmony_ci/* Convert counter idx into SET/CLR register bit. */
9762306a36Sopenharmony_ci#define COUNTER_TO_SET_CLR_BIT(idx)			\
9862306a36Sopenharmony_ci	(idx & (ARM_CSPMU_SET_CLR_COUNTER_NUM - 1))
9962306a36Sopenharmony_ci
10062306a36Sopenharmony_ci#define ARM_CSPMU_ACTIVE_CPU_MASK		0x0
10162306a36Sopenharmony_ci#define ARM_CSPMU_ASSOCIATED_CPU_MASK		0x1
10262306a36Sopenharmony_ci
10362306a36Sopenharmony_ci/* Check and use default if implementer doesn't provide attribute callback */
10462306a36Sopenharmony_ci#define CHECK_DEFAULT_IMPL_OPS(ops, callback)			\
10562306a36Sopenharmony_ci	do {							\
10662306a36Sopenharmony_ci		if (!ops->callback)				\
10762306a36Sopenharmony_ci			ops->callback = arm_cspmu_ ## callback;	\
10862306a36Sopenharmony_ci	} while (0)
10962306a36Sopenharmony_ci
11062306a36Sopenharmony_ci/*
11162306a36Sopenharmony_ci * Maximum poll count for reading counter value using high-low-high sequence.
11262306a36Sopenharmony_ci */
11362306a36Sopenharmony_ci#define HILOHI_MAX_POLL	1000
11462306a36Sopenharmony_ci
11562306a36Sopenharmony_ci/* JEDEC-assigned JEP106 identification code */
11662306a36Sopenharmony_ci#define ARM_CSPMU_IMPL_ID_NVIDIA		0x36B
11762306a36Sopenharmony_ci
11862306a36Sopenharmony_cistatic unsigned long arm_cspmu_cpuhp_state;
11962306a36Sopenharmony_ci
12062306a36Sopenharmony_cistatic struct acpi_apmt_node *arm_cspmu_apmt_node(struct device *dev)
12162306a36Sopenharmony_ci{
12262306a36Sopenharmony_ci	return *(struct acpi_apmt_node **)dev_get_platdata(dev);
12362306a36Sopenharmony_ci}
12462306a36Sopenharmony_ci
12562306a36Sopenharmony_ci/*
12662306a36Sopenharmony_ci * In CoreSight PMU architecture, all of the MMIO registers are 32-bit except
12762306a36Sopenharmony_ci * counter register. The counter register can be implemented as 32-bit or 64-bit
12862306a36Sopenharmony_ci * register depending on the value of PMCFGR.SIZE field. For 64-bit access,
12962306a36Sopenharmony_ci * single-copy 64-bit atomic support is implementation defined. APMT node flag
13062306a36Sopenharmony_ci * is used to identify if the PMU supports 64-bit single copy atomic. If 64-bit
13162306a36Sopenharmony_ci * single copy atomic is not supported, the driver treats the register as a pair
13262306a36Sopenharmony_ci * of 32-bit register.
13362306a36Sopenharmony_ci */
13462306a36Sopenharmony_ci
13562306a36Sopenharmony_ci/*
13662306a36Sopenharmony_ci * Read 64-bit register as a pair of 32-bit registers using hi-lo-hi sequence.
13762306a36Sopenharmony_ci */
13862306a36Sopenharmony_cistatic u64 read_reg64_hilohi(const void __iomem *addr, u32 max_poll_count)
13962306a36Sopenharmony_ci{
14062306a36Sopenharmony_ci	u32 val_lo, val_hi;
14162306a36Sopenharmony_ci	u64 val;
14262306a36Sopenharmony_ci
14362306a36Sopenharmony_ci	/* Use high-low-high sequence to avoid tearing */
14462306a36Sopenharmony_ci	do {
14562306a36Sopenharmony_ci		if (max_poll_count-- == 0) {
14662306a36Sopenharmony_ci			pr_err("ARM CSPMU: timeout hi-low-high sequence\n");
14762306a36Sopenharmony_ci			return 0;
14862306a36Sopenharmony_ci		}
14962306a36Sopenharmony_ci
15062306a36Sopenharmony_ci		val_hi = readl(addr + 4);
15162306a36Sopenharmony_ci		val_lo = readl(addr);
15262306a36Sopenharmony_ci	} while (val_hi != readl(addr + 4));
15362306a36Sopenharmony_ci
15462306a36Sopenharmony_ci	val = (((u64)val_hi << 32) | val_lo);
15562306a36Sopenharmony_ci
15662306a36Sopenharmony_ci	return val;
15762306a36Sopenharmony_ci}
15862306a36Sopenharmony_ci
15962306a36Sopenharmony_ci/* Check if cycle counter is supported. */
16062306a36Sopenharmony_cistatic inline bool supports_cycle_counter(const struct arm_cspmu *cspmu)
16162306a36Sopenharmony_ci{
16262306a36Sopenharmony_ci	return (cspmu->pmcfgr & PMCFGR_CC);
16362306a36Sopenharmony_ci}
16462306a36Sopenharmony_ci
16562306a36Sopenharmony_ci/* Get counter size, which is (PMCFGR_SIZE + 1). */
16662306a36Sopenharmony_cistatic inline u32 counter_size(const struct arm_cspmu *cspmu)
16762306a36Sopenharmony_ci{
16862306a36Sopenharmony_ci	return FIELD_GET(PMCFGR_SIZE, cspmu->pmcfgr) + 1;
16962306a36Sopenharmony_ci}
17062306a36Sopenharmony_ci
17162306a36Sopenharmony_ci/* Get counter mask. */
17262306a36Sopenharmony_cistatic inline u64 counter_mask(const struct arm_cspmu *cspmu)
17362306a36Sopenharmony_ci{
17462306a36Sopenharmony_ci	return GENMASK_ULL(counter_size(cspmu) - 1, 0);
17562306a36Sopenharmony_ci}
17662306a36Sopenharmony_ci
17762306a36Sopenharmony_ci/* Check if counter is implemented as 64-bit register. */
17862306a36Sopenharmony_cistatic inline bool use_64b_counter_reg(const struct arm_cspmu *cspmu)
17962306a36Sopenharmony_ci{
18062306a36Sopenharmony_ci	return (counter_size(cspmu) > 32);
18162306a36Sopenharmony_ci}
18262306a36Sopenharmony_ci
18362306a36Sopenharmony_cissize_t arm_cspmu_sysfs_event_show(struct device *dev,
18462306a36Sopenharmony_ci				struct device_attribute *attr, char *buf)
18562306a36Sopenharmony_ci{
18662306a36Sopenharmony_ci	struct perf_pmu_events_attr *pmu_attr;
18762306a36Sopenharmony_ci
18862306a36Sopenharmony_ci	pmu_attr = container_of(attr, typeof(*pmu_attr), attr);
18962306a36Sopenharmony_ci	return sysfs_emit(buf, "event=0x%llx\n", pmu_attr->id);
19062306a36Sopenharmony_ci}
19162306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(arm_cspmu_sysfs_event_show);
19262306a36Sopenharmony_ci
19362306a36Sopenharmony_ci/* Default event list. */
19462306a36Sopenharmony_cistatic struct attribute *arm_cspmu_event_attrs[] = {
19562306a36Sopenharmony_ci	ARM_CSPMU_EVENT_ATTR(cycles, ARM_CSPMU_EVT_CYCLES_DEFAULT),
19662306a36Sopenharmony_ci	NULL,
19762306a36Sopenharmony_ci};
19862306a36Sopenharmony_ci
19962306a36Sopenharmony_cistatic struct attribute **
20062306a36Sopenharmony_ciarm_cspmu_get_event_attrs(const struct arm_cspmu *cspmu)
20162306a36Sopenharmony_ci{
20262306a36Sopenharmony_ci	struct attribute **attrs;
20362306a36Sopenharmony_ci
20462306a36Sopenharmony_ci	attrs = devm_kmemdup(cspmu->dev, arm_cspmu_event_attrs,
20562306a36Sopenharmony_ci		sizeof(arm_cspmu_event_attrs), GFP_KERNEL);
20662306a36Sopenharmony_ci
20762306a36Sopenharmony_ci	return attrs;
20862306a36Sopenharmony_ci}
20962306a36Sopenharmony_ci
21062306a36Sopenharmony_cistatic umode_t
21162306a36Sopenharmony_ciarm_cspmu_event_attr_is_visible(struct kobject *kobj,
21262306a36Sopenharmony_ci				struct attribute *attr, int unused)
21362306a36Sopenharmony_ci{
21462306a36Sopenharmony_ci	struct device *dev = kobj_to_dev(kobj);
21562306a36Sopenharmony_ci	struct arm_cspmu *cspmu = to_arm_cspmu(dev_get_drvdata(dev));
21662306a36Sopenharmony_ci	struct perf_pmu_events_attr *eattr;
21762306a36Sopenharmony_ci
21862306a36Sopenharmony_ci	eattr = container_of(attr, typeof(*eattr), attr.attr);
21962306a36Sopenharmony_ci
22062306a36Sopenharmony_ci	/* Hide cycle event if not supported */
22162306a36Sopenharmony_ci	if (!supports_cycle_counter(cspmu) &&
22262306a36Sopenharmony_ci	    eattr->id == ARM_CSPMU_EVT_CYCLES_DEFAULT)
22362306a36Sopenharmony_ci		return 0;
22462306a36Sopenharmony_ci
22562306a36Sopenharmony_ci	return attr->mode;
22662306a36Sopenharmony_ci}
22762306a36Sopenharmony_ci
22862306a36Sopenharmony_cissize_t arm_cspmu_sysfs_format_show(struct device *dev,
22962306a36Sopenharmony_ci				struct device_attribute *attr,
23062306a36Sopenharmony_ci				char *buf)
23162306a36Sopenharmony_ci{
23262306a36Sopenharmony_ci	struct dev_ext_attribute *eattr =
23362306a36Sopenharmony_ci		container_of(attr, struct dev_ext_attribute, attr);
23462306a36Sopenharmony_ci	return sysfs_emit(buf, "%s\n", (char *)eattr->var);
23562306a36Sopenharmony_ci}
23662306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(arm_cspmu_sysfs_format_show);
23762306a36Sopenharmony_ci
23862306a36Sopenharmony_cistatic struct attribute *arm_cspmu_format_attrs[] = {
23962306a36Sopenharmony_ci	ARM_CSPMU_FORMAT_EVENT_ATTR,
24062306a36Sopenharmony_ci	ARM_CSPMU_FORMAT_FILTER_ATTR,
24162306a36Sopenharmony_ci	NULL,
24262306a36Sopenharmony_ci};
24362306a36Sopenharmony_ci
24462306a36Sopenharmony_cistatic struct attribute **
24562306a36Sopenharmony_ciarm_cspmu_get_format_attrs(const struct arm_cspmu *cspmu)
24662306a36Sopenharmony_ci{
24762306a36Sopenharmony_ci	struct attribute **attrs;
24862306a36Sopenharmony_ci
24962306a36Sopenharmony_ci	attrs = devm_kmemdup(cspmu->dev, arm_cspmu_format_attrs,
25062306a36Sopenharmony_ci		sizeof(arm_cspmu_format_attrs), GFP_KERNEL);
25162306a36Sopenharmony_ci
25262306a36Sopenharmony_ci	return attrs;
25362306a36Sopenharmony_ci}
25462306a36Sopenharmony_ci
25562306a36Sopenharmony_cistatic u32 arm_cspmu_event_type(const struct perf_event *event)
25662306a36Sopenharmony_ci{
25762306a36Sopenharmony_ci	return event->attr.config & ARM_CSPMU_EVENT_MASK;
25862306a36Sopenharmony_ci}
25962306a36Sopenharmony_ci
26062306a36Sopenharmony_cistatic bool arm_cspmu_is_cycle_counter_event(const struct perf_event *event)
26162306a36Sopenharmony_ci{
26262306a36Sopenharmony_ci	return (event->attr.config == ARM_CSPMU_EVT_CYCLES_DEFAULT);
26362306a36Sopenharmony_ci}
26462306a36Sopenharmony_ci
26562306a36Sopenharmony_cistatic u32 arm_cspmu_event_filter(const struct perf_event *event)
26662306a36Sopenharmony_ci{
26762306a36Sopenharmony_ci	return event->attr.config1 & ARM_CSPMU_FILTER_MASK;
26862306a36Sopenharmony_ci}
26962306a36Sopenharmony_ci
27062306a36Sopenharmony_cistatic ssize_t arm_cspmu_identifier_show(struct device *dev,
27162306a36Sopenharmony_ci					 struct device_attribute *attr,
27262306a36Sopenharmony_ci					 char *page)
27362306a36Sopenharmony_ci{
27462306a36Sopenharmony_ci	struct arm_cspmu *cspmu = to_arm_cspmu(dev_get_drvdata(dev));
27562306a36Sopenharmony_ci
27662306a36Sopenharmony_ci	return sysfs_emit(page, "%s\n", cspmu->identifier);
27762306a36Sopenharmony_ci}
27862306a36Sopenharmony_ci
27962306a36Sopenharmony_cistatic struct device_attribute arm_cspmu_identifier_attr =
28062306a36Sopenharmony_ci	__ATTR(identifier, 0444, arm_cspmu_identifier_show, NULL);
28162306a36Sopenharmony_ci
28262306a36Sopenharmony_cistatic struct attribute *arm_cspmu_identifier_attrs[] = {
28362306a36Sopenharmony_ci	&arm_cspmu_identifier_attr.attr,
28462306a36Sopenharmony_ci	NULL,
28562306a36Sopenharmony_ci};
28662306a36Sopenharmony_ci
28762306a36Sopenharmony_cistatic struct attribute_group arm_cspmu_identifier_attr_group = {
28862306a36Sopenharmony_ci	.attrs = arm_cspmu_identifier_attrs,
28962306a36Sopenharmony_ci};
29062306a36Sopenharmony_ci
29162306a36Sopenharmony_cistatic const char *arm_cspmu_get_identifier(const struct arm_cspmu *cspmu)
29262306a36Sopenharmony_ci{
29362306a36Sopenharmony_ci	const char *identifier =
29462306a36Sopenharmony_ci		devm_kasprintf(cspmu->dev, GFP_KERNEL, "%x",
29562306a36Sopenharmony_ci			       cspmu->impl.pmiidr);
29662306a36Sopenharmony_ci	return identifier;
29762306a36Sopenharmony_ci}
29862306a36Sopenharmony_ci
29962306a36Sopenharmony_cistatic const char *arm_cspmu_type_str[ACPI_APMT_NODE_TYPE_COUNT] = {
30062306a36Sopenharmony_ci	"mc",
30162306a36Sopenharmony_ci	"smmu",
30262306a36Sopenharmony_ci	"pcie",
30362306a36Sopenharmony_ci	"acpi",
30462306a36Sopenharmony_ci	"cache",
30562306a36Sopenharmony_ci};
30662306a36Sopenharmony_ci
30762306a36Sopenharmony_cistatic const char *arm_cspmu_get_name(const struct arm_cspmu *cspmu)
30862306a36Sopenharmony_ci{
30962306a36Sopenharmony_ci	struct device *dev;
31062306a36Sopenharmony_ci	struct acpi_apmt_node *apmt_node;
31162306a36Sopenharmony_ci	u8 pmu_type;
31262306a36Sopenharmony_ci	char *name;
31362306a36Sopenharmony_ci	char acpi_hid_string[ACPI_ID_LEN] = { 0 };
31462306a36Sopenharmony_ci	static atomic_t pmu_idx[ACPI_APMT_NODE_TYPE_COUNT] = { 0 };
31562306a36Sopenharmony_ci
31662306a36Sopenharmony_ci	dev = cspmu->dev;
31762306a36Sopenharmony_ci	apmt_node = arm_cspmu_apmt_node(dev);
31862306a36Sopenharmony_ci	pmu_type = apmt_node->type;
31962306a36Sopenharmony_ci
32062306a36Sopenharmony_ci	if (pmu_type >= ACPI_APMT_NODE_TYPE_COUNT) {
32162306a36Sopenharmony_ci		dev_err(dev, "unsupported PMU type-%u\n", pmu_type);
32262306a36Sopenharmony_ci		return NULL;
32362306a36Sopenharmony_ci	}
32462306a36Sopenharmony_ci
32562306a36Sopenharmony_ci	if (pmu_type == ACPI_APMT_NODE_TYPE_ACPI) {
32662306a36Sopenharmony_ci		memcpy(acpi_hid_string,
32762306a36Sopenharmony_ci			&apmt_node->inst_primary,
32862306a36Sopenharmony_ci			sizeof(apmt_node->inst_primary));
32962306a36Sopenharmony_ci		name = devm_kasprintf(dev, GFP_KERNEL, "%s_%s_%s_%u", PMUNAME,
33062306a36Sopenharmony_ci				      arm_cspmu_type_str[pmu_type],
33162306a36Sopenharmony_ci				      acpi_hid_string,
33262306a36Sopenharmony_ci				      apmt_node->inst_secondary);
33362306a36Sopenharmony_ci	} else {
33462306a36Sopenharmony_ci		name = devm_kasprintf(dev, GFP_KERNEL, "%s_%s_%d", PMUNAME,
33562306a36Sopenharmony_ci				      arm_cspmu_type_str[pmu_type],
33662306a36Sopenharmony_ci				      atomic_fetch_inc(&pmu_idx[pmu_type]));
33762306a36Sopenharmony_ci	}
33862306a36Sopenharmony_ci
33962306a36Sopenharmony_ci	return name;
34062306a36Sopenharmony_ci}
34162306a36Sopenharmony_ci
34262306a36Sopenharmony_cistatic ssize_t arm_cspmu_cpumask_show(struct device *dev,
34362306a36Sopenharmony_ci				      struct device_attribute *attr,
34462306a36Sopenharmony_ci				      char *buf)
34562306a36Sopenharmony_ci{
34662306a36Sopenharmony_ci	struct pmu *pmu = dev_get_drvdata(dev);
34762306a36Sopenharmony_ci	struct arm_cspmu *cspmu = to_arm_cspmu(pmu);
34862306a36Sopenharmony_ci	struct dev_ext_attribute *eattr =
34962306a36Sopenharmony_ci		container_of(attr, struct dev_ext_attribute, attr);
35062306a36Sopenharmony_ci	unsigned long mask_id = (unsigned long)eattr->var;
35162306a36Sopenharmony_ci	const cpumask_t *cpumask;
35262306a36Sopenharmony_ci
35362306a36Sopenharmony_ci	switch (mask_id) {
35462306a36Sopenharmony_ci	case ARM_CSPMU_ACTIVE_CPU_MASK:
35562306a36Sopenharmony_ci		cpumask = &cspmu->active_cpu;
35662306a36Sopenharmony_ci		break;
35762306a36Sopenharmony_ci	case ARM_CSPMU_ASSOCIATED_CPU_MASK:
35862306a36Sopenharmony_ci		cpumask = &cspmu->associated_cpus;
35962306a36Sopenharmony_ci		break;
36062306a36Sopenharmony_ci	default:
36162306a36Sopenharmony_ci		return 0;
36262306a36Sopenharmony_ci	}
36362306a36Sopenharmony_ci	return cpumap_print_to_pagebuf(true, buf, cpumask);
36462306a36Sopenharmony_ci}
36562306a36Sopenharmony_ci
36662306a36Sopenharmony_cistatic struct attribute *arm_cspmu_cpumask_attrs[] = {
36762306a36Sopenharmony_ci	ARM_CSPMU_CPUMASK_ATTR(cpumask, ARM_CSPMU_ACTIVE_CPU_MASK),
36862306a36Sopenharmony_ci	ARM_CSPMU_CPUMASK_ATTR(associated_cpus, ARM_CSPMU_ASSOCIATED_CPU_MASK),
36962306a36Sopenharmony_ci	NULL,
37062306a36Sopenharmony_ci};
37162306a36Sopenharmony_ci
37262306a36Sopenharmony_cistatic struct attribute_group arm_cspmu_cpumask_attr_group = {
37362306a36Sopenharmony_ci	.attrs = arm_cspmu_cpumask_attrs,
37462306a36Sopenharmony_ci};
37562306a36Sopenharmony_ci
37662306a36Sopenharmony_cistruct impl_match {
37762306a36Sopenharmony_ci	u32 pmiidr;
37862306a36Sopenharmony_ci	u32 mask;
37962306a36Sopenharmony_ci	int (*impl_init_ops)(struct arm_cspmu *cspmu);
38062306a36Sopenharmony_ci};
38162306a36Sopenharmony_ci
38262306a36Sopenharmony_cistatic const struct impl_match impl_match[] = {
38362306a36Sopenharmony_ci	{
38462306a36Sopenharmony_ci	  .pmiidr = ARM_CSPMU_IMPL_ID_NVIDIA,
38562306a36Sopenharmony_ci	  .mask = ARM_CSPMU_PMIIDR_IMPLEMENTER,
38662306a36Sopenharmony_ci	  .impl_init_ops = nv_cspmu_init_ops
38762306a36Sopenharmony_ci	},
38862306a36Sopenharmony_ci	{}
38962306a36Sopenharmony_ci};
39062306a36Sopenharmony_ci
39162306a36Sopenharmony_cistatic int arm_cspmu_init_impl_ops(struct arm_cspmu *cspmu)
39262306a36Sopenharmony_ci{
39362306a36Sopenharmony_ci	int ret;
39462306a36Sopenharmony_ci	struct arm_cspmu_impl_ops *impl_ops = &cspmu->impl.ops;
39562306a36Sopenharmony_ci	struct acpi_apmt_node *apmt_node = arm_cspmu_apmt_node(cspmu->dev);
39662306a36Sopenharmony_ci	const struct impl_match *match = impl_match;
39762306a36Sopenharmony_ci
39862306a36Sopenharmony_ci	/*
39962306a36Sopenharmony_ci	 * Get PMU implementer and product id from APMT node.
40062306a36Sopenharmony_ci	 * If APMT node doesn't have implementer/product id, try get it
40162306a36Sopenharmony_ci	 * from PMIIDR.
40262306a36Sopenharmony_ci	 */
40362306a36Sopenharmony_ci	cspmu->impl.pmiidr =
40462306a36Sopenharmony_ci		(apmt_node->impl_id) ? apmt_node->impl_id :
40562306a36Sopenharmony_ci				       readl(cspmu->base0 + PMIIDR);
40662306a36Sopenharmony_ci
40762306a36Sopenharmony_ci	/* Find implementer specific attribute ops. */
40862306a36Sopenharmony_ci	for (; match->pmiidr; match++) {
40962306a36Sopenharmony_ci		const u32 mask = match->mask;
41062306a36Sopenharmony_ci
41162306a36Sopenharmony_ci		if ((match->pmiidr & mask) == (cspmu->impl.pmiidr & mask)) {
41262306a36Sopenharmony_ci			ret = match->impl_init_ops(cspmu);
41362306a36Sopenharmony_ci			if (ret)
41462306a36Sopenharmony_ci				return ret;
41562306a36Sopenharmony_ci
41662306a36Sopenharmony_ci			break;
41762306a36Sopenharmony_ci		}
41862306a36Sopenharmony_ci	}
41962306a36Sopenharmony_ci
42062306a36Sopenharmony_ci	/* Use default callbacks if implementer doesn't provide one. */
42162306a36Sopenharmony_ci	CHECK_DEFAULT_IMPL_OPS(impl_ops, get_event_attrs);
42262306a36Sopenharmony_ci	CHECK_DEFAULT_IMPL_OPS(impl_ops, get_format_attrs);
42362306a36Sopenharmony_ci	CHECK_DEFAULT_IMPL_OPS(impl_ops, get_identifier);
42462306a36Sopenharmony_ci	CHECK_DEFAULT_IMPL_OPS(impl_ops, get_name);
42562306a36Sopenharmony_ci	CHECK_DEFAULT_IMPL_OPS(impl_ops, is_cycle_counter_event);
42662306a36Sopenharmony_ci	CHECK_DEFAULT_IMPL_OPS(impl_ops, event_type);
42762306a36Sopenharmony_ci	CHECK_DEFAULT_IMPL_OPS(impl_ops, event_filter);
42862306a36Sopenharmony_ci	CHECK_DEFAULT_IMPL_OPS(impl_ops, event_attr_is_visible);
42962306a36Sopenharmony_ci
43062306a36Sopenharmony_ci	return 0;
43162306a36Sopenharmony_ci}
43262306a36Sopenharmony_ci
43362306a36Sopenharmony_cistatic struct attribute_group *
43462306a36Sopenharmony_ciarm_cspmu_alloc_event_attr_group(struct arm_cspmu *cspmu)
43562306a36Sopenharmony_ci{
43662306a36Sopenharmony_ci	struct attribute_group *event_group;
43762306a36Sopenharmony_ci	struct device *dev = cspmu->dev;
43862306a36Sopenharmony_ci	const struct arm_cspmu_impl_ops *impl_ops = &cspmu->impl.ops;
43962306a36Sopenharmony_ci
44062306a36Sopenharmony_ci	event_group =
44162306a36Sopenharmony_ci		devm_kzalloc(dev, sizeof(struct attribute_group), GFP_KERNEL);
44262306a36Sopenharmony_ci	if (!event_group)
44362306a36Sopenharmony_ci		return NULL;
44462306a36Sopenharmony_ci
44562306a36Sopenharmony_ci	event_group->name = "events";
44662306a36Sopenharmony_ci	event_group->is_visible = impl_ops->event_attr_is_visible;
44762306a36Sopenharmony_ci	event_group->attrs = impl_ops->get_event_attrs(cspmu);
44862306a36Sopenharmony_ci
44962306a36Sopenharmony_ci	if (!event_group->attrs)
45062306a36Sopenharmony_ci		return NULL;
45162306a36Sopenharmony_ci
45262306a36Sopenharmony_ci	return event_group;
45362306a36Sopenharmony_ci}
45462306a36Sopenharmony_ci
45562306a36Sopenharmony_cistatic struct attribute_group *
45662306a36Sopenharmony_ciarm_cspmu_alloc_format_attr_group(struct arm_cspmu *cspmu)
45762306a36Sopenharmony_ci{
45862306a36Sopenharmony_ci	struct attribute_group *format_group;
45962306a36Sopenharmony_ci	struct device *dev = cspmu->dev;
46062306a36Sopenharmony_ci
46162306a36Sopenharmony_ci	format_group =
46262306a36Sopenharmony_ci		devm_kzalloc(dev, sizeof(struct attribute_group), GFP_KERNEL);
46362306a36Sopenharmony_ci	if (!format_group)
46462306a36Sopenharmony_ci		return NULL;
46562306a36Sopenharmony_ci
46662306a36Sopenharmony_ci	format_group->name = "format";
46762306a36Sopenharmony_ci	format_group->attrs = cspmu->impl.ops.get_format_attrs(cspmu);
46862306a36Sopenharmony_ci
46962306a36Sopenharmony_ci	if (!format_group->attrs)
47062306a36Sopenharmony_ci		return NULL;
47162306a36Sopenharmony_ci
47262306a36Sopenharmony_ci	return format_group;
47362306a36Sopenharmony_ci}
47462306a36Sopenharmony_ci
47562306a36Sopenharmony_cistatic struct attribute_group **
47662306a36Sopenharmony_ciarm_cspmu_alloc_attr_group(struct arm_cspmu *cspmu)
47762306a36Sopenharmony_ci{
47862306a36Sopenharmony_ci	struct attribute_group **attr_groups = NULL;
47962306a36Sopenharmony_ci	struct device *dev = cspmu->dev;
48062306a36Sopenharmony_ci	const struct arm_cspmu_impl_ops *impl_ops = &cspmu->impl.ops;
48162306a36Sopenharmony_ci	int ret;
48262306a36Sopenharmony_ci
48362306a36Sopenharmony_ci	ret = arm_cspmu_init_impl_ops(cspmu);
48462306a36Sopenharmony_ci	if (ret)
48562306a36Sopenharmony_ci		return NULL;
48662306a36Sopenharmony_ci
48762306a36Sopenharmony_ci	cspmu->identifier = impl_ops->get_identifier(cspmu);
48862306a36Sopenharmony_ci	cspmu->name = impl_ops->get_name(cspmu);
48962306a36Sopenharmony_ci
49062306a36Sopenharmony_ci	if (!cspmu->identifier || !cspmu->name)
49162306a36Sopenharmony_ci		return NULL;
49262306a36Sopenharmony_ci
49362306a36Sopenharmony_ci	attr_groups = devm_kcalloc(dev, 5, sizeof(struct attribute_group *),
49462306a36Sopenharmony_ci				   GFP_KERNEL);
49562306a36Sopenharmony_ci	if (!attr_groups)
49662306a36Sopenharmony_ci		return NULL;
49762306a36Sopenharmony_ci
49862306a36Sopenharmony_ci	attr_groups[0] = arm_cspmu_alloc_event_attr_group(cspmu);
49962306a36Sopenharmony_ci	attr_groups[1] = arm_cspmu_alloc_format_attr_group(cspmu);
50062306a36Sopenharmony_ci	attr_groups[2] = &arm_cspmu_identifier_attr_group;
50162306a36Sopenharmony_ci	attr_groups[3] = &arm_cspmu_cpumask_attr_group;
50262306a36Sopenharmony_ci
50362306a36Sopenharmony_ci	if (!attr_groups[0] || !attr_groups[1])
50462306a36Sopenharmony_ci		return NULL;
50562306a36Sopenharmony_ci
50662306a36Sopenharmony_ci	return attr_groups;
50762306a36Sopenharmony_ci}
50862306a36Sopenharmony_ci
50962306a36Sopenharmony_cistatic inline void arm_cspmu_reset_counters(struct arm_cspmu *cspmu)
51062306a36Sopenharmony_ci{
51162306a36Sopenharmony_ci	u32 pmcr = 0;
51262306a36Sopenharmony_ci
51362306a36Sopenharmony_ci	pmcr |= PMCR_P;
51462306a36Sopenharmony_ci	pmcr |= PMCR_C;
51562306a36Sopenharmony_ci	writel(pmcr, cspmu->base0 + PMCR);
51662306a36Sopenharmony_ci}
51762306a36Sopenharmony_ci
51862306a36Sopenharmony_cistatic inline void arm_cspmu_start_counters(struct arm_cspmu *cspmu)
51962306a36Sopenharmony_ci{
52062306a36Sopenharmony_ci	writel(PMCR_E, cspmu->base0 + PMCR);
52162306a36Sopenharmony_ci}
52262306a36Sopenharmony_ci
52362306a36Sopenharmony_cistatic inline void arm_cspmu_stop_counters(struct arm_cspmu *cspmu)
52462306a36Sopenharmony_ci{
52562306a36Sopenharmony_ci	writel(0, cspmu->base0 + PMCR);
52662306a36Sopenharmony_ci}
52762306a36Sopenharmony_ci
52862306a36Sopenharmony_cistatic void arm_cspmu_enable(struct pmu *pmu)
52962306a36Sopenharmony_ci{
53062306a36Sopenharmony_ci	bool disabled;
53162306a36Sopenharmony_ci	struct arm_cspmu *cspmu = to_arm_cspmu(pmu);
53262306a36Sopenharmony_ci
53362306a36Sopenharmony_ci	disabled = bitmap_empty(cspmu->hw_events.used_ctrs,
53462306a36Sopenharmony_ci				cspmu->num_logical_ctrs);
53562306a36Sopenharmony_ci
53662306a36Sopenharmony_ci	if (disabled)
53762306a36Sopenharmony_ci		return;
53862306a36Sopenharmony_ci
53962306a36Sopenharmony_ci	arm_cspmu_start_counters(cspmu);
54062306a36Sopenharmony_ci}
54162306a36Sopenharmony_ci
54262306a36Sopenharmony_cistatic void arm_cspmu_disable(struct pmu *pmu)
54362306a36Sopenharmony_ci{
54462306a36Sopenharmony_ci	struct arm_cspmu *cspmu = to_arm_cspmu(pmu);
54562306a36Sopenharmony_ci
54662306a36Sopenharmony_ci	arm_cspmu_stop_counters(cspmu);
54762306a36Sopenharmony_ci}
54862306a36Sopenharmony_ci
54962306a36Sopenharmony_cistatic int arm_cspmu_get_event_idx(struct arm_cspmu_hw_events *hw_events,
55062306a36Sopenharmony_ci				struct perf_event *event)
55162306a36Sopenharmony_ci{
55262306a36Sopenharmony_ci	int idx;
55362306a36Sopenharmony_ci	struct arm_cspmu *cspmu = to_arm_cspmu(event->pmu);
55462306a36Sopenharmony_ci
55562306a36Sopenharmony_ci	if (supports_cycle_counter(cspmu)) {
55662306a36Sopenharmony_ci		if (cspmu->impl.ops.is_cycle_counter_event(event)) {
55762306a36Sopenharmony_ci			/* Search for available cycle counter. */
55862306a36Sopenharmony_ci			if (test_and_set_bit(cspmu->cycle_counter_logical_idx,
55962306a36Sopenharmony_ci					     hw_events->used_ctrs))
56062306a36Sopenharmony_ci				return -EAGAIN;
56162306a36Sopenharmony_ci
56262306a36Sopenharmony_ci			return cspmu->cycle_counter_logical_idx;
56362306a36Sopenharmony_ci		}
56462306a36Sopenharmony_ci
56562306a36Sopenharmony_ci		/*
56662306a36Sopenharmony_ci		 * Search a regular counter from the used counter bitmap.
56762306a36Sopenharmony_ci		 * The cycle counter divides the bitmap into two parts. Search
56862306a36Sopenharmony_ci		 * the first then second half to exclude the cycle counter bit.
56962306a36Sopenharmony_ci		 */
57062306a36Sopenharmony_ci		idx = find_first_zero_bit(hw_events->used_ctrs,
57162306a36Sopenharmony_ci					  cspmu->cycle_counter_logical_idx);
57262306a36Sopenharmony_ci		if (idx >= cspmu->cycle_counter_logical_idx) {
57362306a36Sopenharmony_ci			idx = find_next_zero_bit(
57462306a36Sopenharmony_ci				hw_events->used_ctrs,
57562306a36Sopenharmony_ci				cspmu->num_logical_ctrs,
57662306a36Sopenharmony_ci				cspmu->cycle_counter_logical_idx + 1);
57762306a36Sopenharmony_ci		}
57862306a36Sopenharmony_ci	} else {
57962306a36Sopenharmony_ci		idx = find_first_zero_bit(hw_events->used_ctrs,
58062306a36Sopenharmony_ci					  cspmu->num_logical_ctrs);
58162306a36Sopenharmony_ci	}
58262306a36Sopenharmony_ci
58362306a36Sopenharmony_ci	if (idx >= cspmu->num_logical_ctrs)
58462306a36Sopenharmony_ci		return -EAGAIN;
58562306a36Sopenharmony_ci
58662306a36Sopenharmony_ci	set_bit(idx, hw_events->used_ctrs);
58762306a36Sopenharmony_ci
58862306a36Sopenharmony_ci	return idx;
58962306a36Sopenharmony_ci}
59062306a36Sopenharmony_ci
59162306a36Sopenharmony_cistatic bool arm_cspmu_validate_event(struct pmu *pmu,
59262306a36Sopenharmony_ci				 struct arm_cspmu_hw_events *hw_events,
59362306a36Sopenharmony_ci				 struct perf_event *event)
59462306a36Sopenharmony_ci{
59562306a36Sopenharmony_ci	if (is_software_event(event))
59662306a36Sopenharmony_ci		return true;
59762306a36Sopenharmony_ci
59862306a36Sopenharmony_ci	/* Reject groups spanning multiple HW PMUs. */
59962306a36Sopenharmony_ci	if (event->pmu != pmu)
60062306a36Sopenharmony_ci		return false;
60162306a36Sopenharmony_ci
60262306a36Sopenharmony_ci	return (arm_cspmu_get_event_idx(hw_events, event) >= 0);
60362306a36Sopenharmony_ci}
60462306a36Sopenharmony_ci
60562306a36Sopenharmony_ci/*
60662306a36Sopenharmony_ci * Make sure the group of events can be scheduled at once
60762306a36Sopenharmony_ci * on the PMU.
60862306a36Sopenharmony_ci */
60962306a36Sopenharmony_cistatic bool arm_cspmu_validate_group(struct perf_event *event)
61062306a36Sopenharmony_ci{
61162306a36Sopenharmony_ci	struct perf_event *sibling, *leader = event->group_leader;
61262306a36Sopenharmony_ci	struct arm_cspmu_hw_events fake_hw_events;
61362306a36Sopenharmony_ci
61462306a36Sopenharmony_ci	if (event->group_leader == event)
61562306a36Sopenharmony_ci		return true;
61662306a36Sopenharmony_ci
61762306a36Sopenharmony_ci	memset(&fake_hw_events, 0, sizeof(fake_hw_events));
61862306a36Sopenharmony_ci
61962306a36Sopenharmony_ci	if (!arm_cspmu_validate_event(event->pmu, &fake_hw_events, leader))
62062306a36Sopenharmony_ci		return false;
62162306a36Sopenharmony_ci
62262306a36Sopenharmony_ci	for_each_sibling_event(sibling, leader) {
62362306a36Sopenharmony_ci		if (!arm_cspmu_validate_event(event->pmu, &fake_hw_events,
62462306a36Sopenharmony_ci						  sibling))
62562306a36Sopenharmony_ci			return false;
62662306a36Sopenharmony_ci	}
62762306a36Sopenharmony_ci
62862306a36Sopenharmony_ci	return arm_cspmu_validate_event(event->pmu, &fake_hw_events, event);
62962306a36Sopenharmony_ci}
63062306a36Sopenharmony_ci
63162306a36Sopenharmony_cistatic int arm_cspmu_event_init(struct perf_event *event)
63262306a36Sopenharmony_ci{
63362306a36Sopenharmony_ci	struct arm_cspmu *cspmu;
63462306a36Sopenharmony_ci	struct hw_perf_event *hwc = &event->hw;
63562306a36Sopenharmony_ci
63662306a36Sopenharmony_ci	cspmu = to_arm_cspmu(event->pmu);
63762306a36Sopenharmony_ci
63862306a36Sopenharmony_ci	if (event->attr.type != event->pmu->type)
63962306a36Sopenharmony_ci		return -ENOENT;
64062306a36Sopenharmony_ci
64162306a36Sopenharmony_ci	/*
64262306a36Sopenharmony_ci	 * Following other "uncore" PMUs, we do not support sampling mode or
64362306a36Sopenharmony_ci	 * attach to a task (per-process mode).
64462306a36Sopenharmony_ci	 */
64562306a36Sopenharmony_ci	if (is_sampling_event(event)) {
64662306a36Sopenharmony_ci		dev_dbg(cspmu->pmu.dev,
64762306a36Sopenharmony_ci			"Can't support sampling events\n");
64862306a36Sopenharmony_ci		return -EOPNOTSUPP;
64962306a36Sopenharmony_ci	}
65062306a36Sopenharmony_ci
65162306a36Sopenharmony_ci	if (event->cpu < 0 || event->attach_state & PERF_ATTACH_TASK) {
65262306a36Sopenharmony_ci		dev_dbg(cspmu->pmu.dev,
65362306a36Sopenharmony_ci			"Can't support per-task counters\n");
65462306a36Sopenharmony_ci		return -EINVAL;
65562306a36Sopenharmony_ci	}
65662306a36Sopenharmony_ci
65762306a36Sopenharmony_ci	/*
65862306a36Sopenharmony_ci	 * Make sure the CPU assignment is on one of the CPUs associated with
65962306a36Sopenharmony_ci	 * this PMU.
66062306a36Sopenharmony_ci	 */
66162306a36Sopenharmony_ci	if (!cpumask_test_cpu(event->cpu, &cspmu->associated_cpus)) {
66262306a36Sopenharmony_ci		dev_dbg(cspmu->pmu.dev,
66362306a36Sopenharmony_ci			"Requested cpu is not associated with the PMU\n");
66462306a36Sopenharmony_ci		return -EINVAL;
66562306a36Sopenharmony_ci	}
66662306a36Sopenharmony_ci
66762306a36Sopenharmony_ci	/* Enforce the current active CPU to handle the events in this PMU. */
66862306a36Sopenharmony_ci	event->cpu = cpumask_first(&cspmu->active_cpu);
66962306a36Sopenharmony_ci	if (event->cpu >= nr_cpu_ids)
67062306a36Sopenharmony_ci		return -EINVAL;
67162306a36Sopenharmony_ci
67262306a36Sopenharmony_ci	if (!arm_cspmu_validate_group(event))
67362306a36Sopenharmony_ci		return -EINVAL;
67462306a36Sopenharmony_ci
67562306a36Sopenharmony_ci	/*
67662306a36Sopenharmony_ci	 * The logical counter id is tracked with hw_perf_event.extra_reg.idx.
67762306a36Sopenharmony_ci	 * The physical counter id is tracked with hw_perf_event.idx.
67862306a36Sopenharmony_ci	 * We don't assign an index until we actually place the event onto
67962306a36Sopenharmony_ci	 * hardware. Use -1 to signify that we haven't decided where to put it
68062306a36Sopenharmony_ci	 * yet.
68162306a36Sopenharmony_ci	 */
68262306a36Sopenharmony_ci	hwc->idx = -1;
68362306a36Sopenharmony_ci	hwc->extra_reg.idx = -1;
68462306a36Sopenharmony_ci	hwc->config = cspmu->impl.ops.event_type(event);
68562306a36Sopenharmony_ci
68662306a36Sopenharmony_ci	return 0;
68762306a36Sopenharmony_ci}
68862306a36Sopenharmony_ci
68962306a36Sopenharmony_cistatic inline u32 counter_offset(u32 reg_sz, u32 ctr_idx)
69062306a36Sopenharmony_ci{
69162306a36Sopenharmony_ci	return (PMEVCNTR_LO + (reg_sz * ctr_idx));
69262306a36Sopenharmony_ci}
69362306a36Sopenharmony_ci
69462306a36Sopenharmony_cistatic void arm_cspmu_write_counter(struct perf_event *event, u64 val)
69562306a36Sopenharmony_ci{
69662306a36Sopenharmony_ci	u32 offset;
69762306a36Sopenharmony_ci	struct arm_cspmu *cspmu = to_arm_cspmu(event->pmu);
69862306a36Sopenharmony_ci
69962306a36Sopenharmony_ci	if (use_64b_counter_reg(cspmu)) {
70062306a36Sopenharmony_ci		offset = counter_offset(sizeof(u64), event->hw.idx);
70162306a36Sopenharmony_ci
70262306a36Sopenharmony_ci		writeq(val, cspmu->base1 + offset);
70362306a36Sopenharmony_ci	} else {
70462306a36Sopenharmony_ci		offset = counter_offset(sizeof(u32), event->hw.idx);
70562306a36Sopenharmony_ci
70662306a36Sopenharmony_ci		writel(lower_32_bits(val), cspmu->base1 + offset);
70762306a36Sopenharmony_ci	}
70862306a36Sopenharmony_ci}
70962306a36Sopenharmony_ci
71062306a36Sopenharmony_cistatic u64 arm_cspmu_read_counter(struct perf_event *event)
71162306a36Sopenharmony_ci{
71262306a36Sopenharmony_ci	u32 offset;
71362306a36Sopenharmony_ci	const void __iomem *counter_addr;
71462306a36Sopenharmony_ci	struct arm_cspmu *cspmu = to_arm_cspmu(event->pmu);
71562306a36Sopenharmony_ci
71662306a36Sopenharmony_ci	if (use_64b_counter_reg(cspmu)) {
71762306a36Sopenharmony_ci		offset = counter_offset(sizeof(u64), event->hw.idx);
71862306a36Sopenharmony_ci		counter_addr = cspmu->base1 + offset;
71962306a36Sopenharmony_ci
72062306a36Sopenharmony_ci		return cspmu->has_atomic_dword ?
72162306a36Sopenharmony_ci			       readq(counter_addr) :
72262306a36Sopenharmony_ci			       read_reg64_hilohi(counter_addr, HILOHI_MAX_POLL);
72362306a36Sopenharmony_ci	}
72462306a36Sopenharmony_ci
72562306a36Sopenharmony_ci	offset = counter_offset(sizeof(u32), event->hw.idx);
72662306a36Sopenharmony_ci	return readl(cspmu->base1 + offset);
72762306a36Sopenharmony_ci}
72862306a36Sopenharmony_ci
72962306a36Sopenharmony_ci/*
73062306a36Sopenharmony_ci * arm_cspmu_set_event_period: Set the period for the counter.
73162306a36Sopenharmony_ci *
73262306a36Sopenharmony_ci * To handle cases of extreme interrupt latency, we program
73362306a36Sopenharmony_ci * the counter with half of the max count for the counters.
73462306a36Sopenharmony_ci */
73562306a36Sopenharmony_cistatic void arm_cspmu_set_event_period(struct perf_event *event)
73662306a36Sopenharmony_ci{
73762306a36Sopenharmony_ci	struct arm_cspmu *cspmu = to_arm_cspmu(event->pmu);
73862306a36Sopenharmony_ci	u64 val = counter_mask(cspmu) >> 1ULL;
73962306a36Sopenharmony_ci
74062306a36Sopenharmony_ci	local64_set(&event->hw.prev_count, val);
74162306a36Sopenharmony_ci	arm_cspmu_write_counter(event, val);
74262306a36Sopenharmony_ci}
74362306a36Sopenharmony_ci
74462306a36Sopenharmony_cistatic void arm_cspmu_enable_counter(struct arm_cspmu *cspmu, int idx)
74562306a36Sopenharmony_ci{
74662306a36Sopenharmony_ci	u32 reg_id, reg_bit, inten_off, cnten_off;
74762306a36Sopenharmony_ci
74862306a36Sopenharmony_ci	reg_id = COUNTER_TO_SET_CLR_ID(idx);
74962306a36Sopenharmony_ci	reg_bit = COUNTER_TO_SET_CLR_BIT(idx);
75062306a36Sopenharmony_ci
75162306a36Sopenharmony_ci	inten_off = PMINTENSET + (4 * reg_id);
75262306a36Sopenharmony_ci	cnten_off = PMCNTENSET + (4 * reg_id);
75362306a36Sopenharmony_ci
75462306a36Sopenharmony_ci	writel(BIT(reg_bit), cspmu->base0 + inten_off);
75562306a36Sopenharmony_ci	writel(BIT(reg_bit), cspmu->base0 + cnten_off);
75662306a36Sopenharmony_ci}
75762306a36Sopenharmony_ci
75862306a36Sopenharmony_cistatic void arm_cspmu_disable_counter(struct arm_cspmu *cspmu, int idx)
75962306a36Sopenharmony_ci{
76062306a36Sopenharmony_ci	u32 reg_id, reg_bit, inten_off, cnten_off;
76162306a36Sopenharmony_ci
76262306a36Sopenharmony_ci	reg_id = COUNTER_TO_SET_CLR_ID(idx);
76362306a36Sopenharmony_ci	reg_bit = COUNTER_TO_SET_CLR_BIT(idx);
76462306a36Sopenharmony_ci
76562306a36Sopenharmony_ci	inten_off = PMINTENCLR + (4 * reg_id);
76662306a36Sopenharmony_ci	cnten_off = PMCNTENCLR + (4 * reg_id);
76762306a36Sopenharmony_ci
76862306a36Sopenharmony_ci	writel(BIT(reg_bit), cspmu->base0 + cnten_off);
76962306a36Sopenharmony_ci	writel(BIT(reg_bit), cspmu->base0 + inten_off);
77062306a36Sopenharmony_ci}
77162306a36Sopenharmony_ci
77262306a36Sopenharmony_cistatic void arm_cspmu_event_update(struct perf_event *event)
77362306a36Sopenharmony_ci{
77462306a36Sopenharmony_ci	struct arm_cspmu *cspmu = to_arm_cspmu(event->pmu);
77562306a36Sopenharmony_ci	struct hw_perf_event *hwc = &event->hw;
77662306a36Sopenharmony_ci	u64 delta, prev, now;
77762306a36Sopenharmony_ci
77862306a36Sopenharmony_ci	do {
77962306a36Sopenharmony_ci		prev = local64_read(&hwc->prev_count);
78062306a36Sopenharmony_ci		now = arm_cspmu_read_counter(event);
78162306a36Sopenharmony_ci	} while (local64_cmpxchg(&hwc->prev_count, prev, now) != prev);
78262306a36Sopenharmony_ci
78362306a36Sopenharmony_ci	delta = (now - prev) & counter_mask(cspmu);
78462306a36Sopenharmony_ci	local64_add(delta, &event->count);
78562306a36Sopenharmony_ci}
78662306a36Sopenharmony_ci
78762306a36Sopenharmony_cistatic inline void arm_cspmu_set_event(struct arm_cspmu *cspmu,
78862306a36Sopenharmony_ci					struct hw_perf_event *hwc)
78962306a36Sopenharmony_ci{
79062306a36Sopenharmony_ci	u32 offset = PMEVTYPER + (4 * hwc->idx);
79162306a36Sopenharmony_ci
79262306a36Sopenharmony_ci	writel(hwc->config, cspmu->base0 + offset);
79362306a36Sopenharmony_ci}
79462306a36Sopenharmony_ci
79562306a36Sopenharmony_cistatic inline void arm_cspmu_set_ev_filter(struct arm_cspmu *cspmu,
79662306a36Sopenharmony_ci					   struct hw_perf_event *hwc,
79762306a36Sopenharmony_ci					   u32 filter)
79862306a36Sopenharmony_ci{
79962306a36Sopenharmony_ci	u32 offset = PMEVFILTR + (4 * hwc->idx);
80062306a36Sopenharmony_ci
80162306a36Sopenharmony_ci	writel(filter, cspmu->base0 + offset);
80262306a36Sopenharmony_ci}
80362306a36Sopenharmony_ci
80462306a36Sopenharmony_cistatic inline void arm_cspmu_set_cc_filter(struct arm_cspmu *cspmu, u32 filter)
80562306a36Sopenharmony_ci{
80662306a36Sopenharmony_ci	u32 offset = PMCCFILTR;
80762306a36Sopenharmony_ci
80862306a36Sopenharmony_ci	writel(filter, cspmu->base0 + offset);
80962306a36Sopenharmony_ci}
81062306a36Sopenharmony_ci
81162306a36Sopenharmony_cistatic void arm_cspmu_start(struct perf_event *event, int pmu_flags)
81262306a36Sopenharmony_ci{
81362306a36Sopenharmony_ci	struct arm_cspmu *cspmu = to_arm_cspmu(event->pmu);
81462306a36Sopenharmony_ci	struct hw_perf_event *hwc = &event->hw;
81562306a36Sopenharmony_ci	u32 filter;
81662306a36Sopenharmony_ci
81762306a36Sopenharmony_ci	/* We always reprogram the counter */
81862306a36Sopenharmony_ci	if (pmu_flags & PERF_EF_RELOAD)
81962306a36Sopenharmony_ci		WARN_ON(!(hwc->state & PERF_HES_UPTODATE));
82062306a36Sopenharmony_ci
82162306a36Sopenharmony_ci	arm_cspmu_set_event_period(event);
82262306a36Sopenharmony_ci
82362306a36Sopenharmony_ci	filter = cspmu->impl.ops.event_filter(event);
82462306a36Sopenharmony_ci
82562306a36Sopenharmony_ci	if (event->hw.extra_reg.idx == cspmu->cycle_counter_logical_idx) {
82662306a36Sopenharmony_ci		arm_cspmu_set_cc_filter(cspmu, filter);
82762306a36Sopenharmony_ci	} else {
82862306a36Sopenharmony_ci		arm_cspmu_set_event(cspmu, hwc);
82962306a36Sopenharmony_ci		arm_cspmu_set_ev_filter(cspmu, hwc, filter);
83062306a36Sopenharmony_ci	}
83162306a36Sopenharmony_ci
83262306a36Sopenharmony_ci	hwc->state = 0;
83362306a36Sopenharmony_ci
83462306a36Sopenharmony_ci	arm_cspmu_enable_counter(cspmu, hwc->idx);
83562306a36Sopenharmony_ci}
83662306a36Sopenharmony_ci
83762306a36Sopenharmony_cistatic void arm_cspmu_stop(struct perf_event *event, int pmu_flags)
83862306a36Sopenharmony_ci{
83962306a36Sopenharmony_ci	struct arm_cspmu *cspmu = to_arm_cspmu(event->pmu);
84062306a36Sopenharmony_ci	struct hw_perf_event *hwc = &event->hw;
84162306a36Sopenharmony_ci
84262306a36Sopenharmony_ci	if (hwc->state & PERF_HES_STOPPED)
84362306a36Sopenharmony_ci		return;
84462306a36Sopenharmony_ci
84562306a36Sopenharmony_ci	arm_cspmu_disable_counter(cspmu, hwc->idx);
84662306a36Sopenharmony_ci	arm_cspmu_event_update(event);
84762306a36Sopenharmony_ci
84862306a36Sopenharmony_ci	hwc->state |= PERF_HES_STOPPED | PERF_HES_UPTODATE;
84962306a36Sopenharmony_ci}
85062306a36Sopenharmony_ci
85162306a36Sopenharmony_cistatic inline u32 to_phys_idx(struct arm_cspmu *cspmu, u32 idx)
85262306a36Sopenharmony_ci{
85362306a36Sopenharmony_ci	return (idx == cspmu->cycle_counter_logical_idx) ?
85462306a36Sopenharmony_ci		ARM_CSPMU_CYCLE_CNTR_IDX : idx;
85562306a36Sopenharmony_ci}
85662306a36Sopenharmony_ci
85762306a36Sopenharmony_cistatic int arm_cspmu_add(struct perf_event *event, int flags)
85862306a36Sopenharmony_ci{
85962306a36Sopenharmony_ci	struct arm_cspmu *cspmu = to_arm_cspmu(event->pmu);
86062306a36Sopenharmony_ci	struct arm_cspmu_hw_events *hw_events = &cspmu->hw_events;
86162306a36Sopenharmony_ci	struct hw_perf_event *hwc = &event->hw;
86262306a36Sopenharmony_ci	int idx;
86362306a36Sopenharmony_ci
86462306a36Sopenharmony_ci	if (WARN_ON_ONCE(!cpumask_test_cpu(smp_processor_id(),
86562306a36Sopenharmony_ci					   &cspmu->associated_cpus)))
86662306a36Sopenharmony_ci		return -ENOENT;
86762306a36Sopenharmony_ci
86862306a36Sopenharmony_ci	idx = arm_cspmu_get_event_idx(hw_events, event);
86962306a36Sopenharmony_ci	if (idx < 0)
87062306a36Sopenharmony_ci		return idx;
87162306a36Sopenharmony_ci
87262306a36Sopenharmony_ci	hw_events->events[idx] = event;
87362306a36Sopenharmony_ci	hwc->idx = to_phys_idx(cspmu, idx);
87462306a36Sopenharmony_ci	hwc->extra_reg.idx = idx;
87562306a36Sopenharmony_ci	hwc->state = PERF_HES_STOPPED | PERF_HES_UPTODATE;
87662306a36Sopenharmony_ci
87762306a36Sopenharmony_ci	if (flags & PERF_EF_START)
87862306a36Sopenharmony_ci		arm_cspmu_start(event, PERF_EF_RELOAD);
87962306a36Sopenharmony_ci
88062306a36Sopenharmony_ci	/* Propagate changes to the userspace mapping. */
88162306a36Sopenharmony_ci	perf_event_update_userpage(event);
88262306a36Sopenharmony_ci
88362306a36Sopenharmony_ci	return 0;
88462306a36Sopenharmony_ci}
88562306a36Sopenharmony_ci
88662306a36Sopenharmony_cistatic void arm_cspmu_del(struct perf_event *event, int flags)
88762306a36Sopenharmony_ci{
88862306a36Sopenharmony_ci	struct arm_cspmu *cspmu = to_arm_cspmu(event->pmu);
88962306a36Sopenharmony_ci	struct arm_cspmu_hw_events *hw_events = &cspmu->hw_events;
89062306a36Sopenharmony_ci	struct hw_perf_event *hwc = &event->hw;
89162306a36Sopenharmony_ci	int idx = hwc->extra_reg.idx;
89262306a36Sopenharmony_ci
89362306a36Sopenharmony_ci	arm_cspmu_stop(event, PERF_EF_UPDATE);
89462306a36Sopenharmony_ci
89562306a36Sopenharmony_ci	hw_events->events[idx] = NULL;
89662306a36Sopenharmony_ci
89762306a36Sopenharmony_ci	clear_bit(idx, hw_events->used_ctrs);
89862306a36Sopenharmony_ci
89962306a36Sopenharmony_ci	perf_event_update_userpage(event);
90062306a36Sopenharmony_ci}
90162306a36Sopenharmony_ci
90262306a36Sopenharmony_cistatic void arm_cspmu_read(struct perf_event *event)
90362306a36Sopenharmony_ci{
90462306a36Sopenharmony_ci	arm_cspmu_event_update(event);
90562306a36Sopenharmony_ci}
90662306a36Sopenharmony_ci
90762306a36Sopenharmony_cistatic struct arm_cspmu *arm_cspmu_alloc(struct platform_device *pdev)
90862306a36Sopenharmony_ci{
90962306a36Sopenharmony_ci	struct acpi_apmt_node *apmt_node;
91062306a36Sopenharmony_ci	struct arm_cspmu *cspmu;
91162306a36Sopenharmony_ci	struct device *dev = &pdev->dev;
91262306a36Sopenharmony_ci
91362306a36Sopenharmony_ci	cspmu = devm_kzalloc(dev, sizeof(*cspmu), GFP_KERNEL);
91462306a36Sopenharmony_ci	if (!cspmu)
91562306a36Sopenharmony_ci		return NULL;
91662306a36Sopenharmony_ci
91762306a36Sopenharmony_ci	cspmu->dev = dev;
91862306a36Sopenharmony_ci	platform_set_drvdata(pdev, cspmu);
91962306a36Sopenharmony_ci
92062306a36Sopenharmony_ci	apmt_node = arm_cspmu_apmt_node(dev);
92162306a36Sopenharmony_ci	cspmu->has_atomic_dword = apmt_node->flags & ACPI_APMT_FLAGS_ATOMIC;
92262306a36Sopenharmony_ci
92362306a36Sopenharmony_ci	return cspmu;
92462306a36Sopenharmony_ci}
92562306a36Sopenharmony_ci
92662306a36Sopenharmony_cistatic int arm_cspmu_init_mmio(struct arm_cspmu *cspmu)
92762306a36Sopenharmony_ci{
92862306a36Sopenharmony_ci	struct device *dev;
92962306a36Sopenharmony_ci	struct platform_device *pdev;
93062306a36Sopenharmony_ci
93162306a36Sopenharmony_ci	dev = cspmu->dev;
93262306a36Sopenharmony_ci	pdev = to_platform_device(dev);
93362306a36Sopenharmony_ci
93462306a36Sopenharmony_ci	/* Base address for page 0. */
93562306a36Sopenharmony_ci	cspmu->base0 = devm_platform_ioremap_resource(pdev, 0);
93662306a36Sopenharmony_ci	if (IS_ERR(cspmu->base0)) {
93762306a36Sopenharmony_ci		dev_err(dev, "ioremap failed for page-0 resource\n");
93862306a36Sopenharmony_ci		return PTR_ERR(cspmu->base0);
93962306a36Sopenharmony_ci	}
94062306a36Sopenharmony_ci
94162306a36Sopenharmony_ci	/* Base address for page 1 if supported. Otherwise point to page 0. */
94262306a36Sopenharmony_ci	cspmu->base1 = cspmu->base0;
94362306a36Sopenharmony_ci	if (platform_get_resource(pdev, IORESOURCE_MEM, 1)) {
94462306a36Sopenharmony_ci		cspmu->base1 = devm_platform_ioremap_resource(pdev, 1);
94562306a36Sopenharmony_ci		if (IS_ERR(cspmu->base1)) {
94662306a36Sopenharmony_ci			dev_err(dev, "ioremap failed for page-1 resource\n");
94762306a36Sopenharmony_ci			return PTR_ERR(cspmu->base1);
94862306a36Sopenharmony_ci		}
94962306a36Sopenharmony_ci	}
95062306a36Sopenharmony_ci
95162306a36Sopenharmony_ci	cspmu->pmcfgr = readl(cspmu->base0 + PMCFGR);
95262306a36Sopenharmony_ci
95362306a36Sopenharmony_ci	cspmu->num_logical_ctrs = FIELD_GET(PMCFGR_N, cspmu->pmcfgr) + 1;
95462306a36Sopenharmony_ci
95562306a36Sopenharmony_ci	cspmu->cycle_counter_logical_idx = ARM_CSPMU_MAX_HW_CNTRS;
95662306a36Sopenharmony_ci
95762306a36Sopenharmony_ci	if (supports_cycle_counter(cspmu)) {
95862306a36Sopenharmony_ci		/*
95962306a36Sopenharmony_ci		 * The last logical counter is mapped to cycle counter if
96062306a36Sopenharmony_ci		 * there is a gap between regular and cycle counter. Otherwise,
96162306a36Sopenharmony_ci		 * logical and physical have 1-to-1 mapping.
96262306a36Sopenharmony_ci		 */
96362306a36Sopenharmony_ci		cspmu->cycle_counter_logical_idx =
96462306a36Sopenharmony_ci			(cspmu->num_logical_ctrs <= ARM_CSPMU_CYCLE_CNTR_IDX) ?
96562306a36Sopenharmony_ci				cspmu->num_logical_ctrs - 1 :
96662306a36Sopenharmony_ci				ARM_CSPMU_CYCLE_CNTR_IDX;
96762306a36Sopenharmony_ci	}
96862306a36Sopenharmony_ci
96962306a36Sopenharmony_ci	cspmu->num_set_clr_reg =
97062306a36Sopenharmony_ci		DIV_ROUND_UP(cspmu->num_logical_ctrs,
97162306a36Sopenharmony_ci				ARM_CSPMU_SET_CLR_COUNTER_NUM);
97262306a36Sopenharmony_ci
97362306a36Sopenharmony_ci	cspmu->hw_events.events =
97462306a36Sopenharmony_ci		devm_kcalloc(dev, cspmu->num_logical_ctrs,
97562306a36Sopenharmony_ci			     sizeof(*cspmu->hw_events.events), GFP_KERNEL);
97662306a36Sopenharmony_ci
97762306a36Sopenharmony_ci	if (!cspmu->hw_events.events)
97862306a36Sopenharmony_ci		return -ENOMEM;
97962306a36Sopenharmony_ci
98062306a36Sopenharmony_ci	return 0;
98162306a36Sopenharmony_ci}
98262306a36Sopenharmony_ci
98362306a36Sopenharmony_cistatic inline int arm_cspmu_get_reset_overflow(struct arm_cspmu *cspmu,
98462306a36Sopenharmony_ci					       u32 *pmovs)
98562306a36Sopenharmony_ci{
98662306a36Sopenharmony_ci	int i;
98762306a36Sopenharmony_ci	u32 pmovclr_offset = PMOVSCLR;
98862306a36Sopenharmony_ci	u32 has_overflowed = 0;
98962306a36Sopenharmony_ci
99062306a36Sopenharmony_ci	for (i = 0; i < cspmu->num_set_clr_reg; ++i) {
99162306a36Sopenharmony_ci		pmovs[i] = readl(cspmu->base1 + pmovclr_offset);
99262306a36Sopenharmony_ci		has_overflowed |= pmovs[i];
99362306a36Sopenharmony_ci		writel(pmovs[i], cspmu->base1 + pmovclr_offset);
99462306a36Sopenharmony_ci		pmovclr_offset += sizeof(u32);
99562306a36Sopenharmony_ci	}
99662306a36Sopenharmony_ci
99762306a36Sopenharmony_ci	return has_overflowed != 0;
99862306a36Sopenharmony_ci}
99962306a36Sopenharmony_ci
100062306a36Sopenharmony_cistatic irqreturn_t arm_cspmu_handle_irq(int irq_num, void *dev)
100162306a36Sopenharmony_ci{
100262306a36Sopenharmony_ci	int idx, has_overflowed;
100362306a36Sopenharmony_ci	struct perf_event *event;
100462306a36Sopenharmony_ci	struct arm_cspmu *cspmu = dev;
100562306a36Sopenharmony_ci	DECLARE_BITMAP(pmovs, ARM_CSPMU_MAX_HW_CNTRS);
100662306a36Sopenharmony_ci	bool handled = false;
100762306a36Sopenharmony_ci
100862306a36Sopenharmony_ci	arm_cspmu_stop_counters(cspmu);
100962306a36Sopenharmony_ci
101062306a36Sopenharmony_ci	has_overflowed = arm_cspmu_get_reset_overflow(cspmu, (u32 *)pmovs);
101162306a36Sopenharmony_ci	if (!has_overflowed)
101262306a36Sopenharmony_ci		goto done;
101362306a36Sopenharmony_ci
101462306a36Sopenharmony_ci	for_each_set_bit(idx, cspmu->hw_events.used_ctrs,
101562306a36Sopenharmony_ci			cspmu->num_logical_ctrs) {
101662306a36Sopenharmony_ci		event = cspmu->hw_events.events[idx];
101762306a36Sopenharmony_ci
101862306a36Sopenharmony_ci		if (!event)
101962306a36Sopenharmony_ci			continue;
102062306a36Sopenharmony_ci
102162306a36Sopenharmony_ci		if (!test_bit(event->hw.idx, pmovs))
102262306a36Sopenharmony_ci			continue;
102362306a36Sopenharmony_ci
102462306a36Sopenharmony_ci		arm_cspmu_event_update(event);
102562306a36Sopenharmony_ci		arm_cspmu_set_event_period(event);
102662306a36Sopenharmony_ci
102762306a36Sopenharmony_ci		handled = true;
102862306a36Sopenharmony_ci	}
102962306a36Sopenharmony_ci
103062306a36Sopenharmony_cidone:
103162306a36Sopenharmony_ci	arm_cspmu_start_counters(cspmu);
103262306a36Sopenharmony_ci	return IRQ_RETVAL(handled);
103362306a36Sopenharmony_ci}
103462306a36Sopenharmony_ci
103562306a36Sopenharmony_cistatic int arm_cspmu_request_irq(struct arm_cspmu *cspmu)
103662306a36Sopenharmony_ci{
103762306a36Sopenharmony_ci	int irq, ret;
103862306a36Sopenharmony_ci	struct device *dev;
103962306a36Sopenharmony_ci	struct platform_device *pdev;
104062306a36Sopenharmony_ci
104162306a36Sopenharmony_ci	dev = cspmu->dev;
104262306a36Sopenharmony_ci	pdev = to_platform_device(dev);
104362306a36Sopenharmony_ci
104462306a36Sopenharmony_ci	/* Skip IRQ request if the PMU does not support overflow interrupt. */
104562306a36Sopenharmony_ci	irq = platform_get_irq_optional(pdev, 0);
104662306a36Sopenharmony_ci	if (irq < 0)
104762306a36Sopenharmony_ci		return irq == -ENXIO ? 0 : irq;
104862306a36Sopenharmony_ci
104962306a36Sopenharmony_ci	ret = devm_request_irq(dev, irq, arm_cspmu_handle_irq,
105062306a36Sopenharmony_ci			       IRQF_NOBALANCING | IRQF_NO_THREAD, dev_name(dev),
105162306a36Sopenharmony_ci			       cspmu);
105262306a36Sopenharmony_ci	if (ret) {
105362306a36Sopenharmony_ci		dev_err(dev, "Could not request IRQ %d\n", irq);
105462306a36Sopenharmony_ci		return ret;
105562306a36Sopenharmony_ci	}
105662306a36Sopenharmony_ci
105762306a36Sopenharmony_ci	cspmu->irq = irq;
105862306a36Sopenharmony_ci
105962306a36Sopenharmony_ci	return 0;
106062306a36Sopenharmony_ci}
106162306a36Sopenharmony_ci
106262306a36Sopenharmony_ci#if defined(CONFIG_ACPI) && defined(CONFIG_ARM64)
106362306a36Sopenharmony_ci#include <acpi/processor.h>
106462306a36Sopenharmony_ci
106562306a36Sopenharmony_cistatic inline int arm_cspmu_find_cpu_container(int cpu, u32 container_uid)
106662306a36Sopenharmony_ci{
106762306a36Sopenharmony_ci	u32 acpi_uid;
106862306a36Sopenharmony_ci	struct device *cpu_dev;
106962306a36Sopenharmony_ci	struct acpi_device *acpi_dev;
107062306a36Sopenharmony_ci
107162306a36Sopenharmony_ci	cpu_dev = get_cpu_device(cpu);
107262306a36Sopenharmony_ci	if (!cpu_dev)
107362306a36Sopenharmony_ci		return -ENODEV;
107462306a36Sopenharmony_ci
107562306a36Sopenharmony_ci	acpi_dev = ACPI_COMPANION(cpu_dev);
107662306a36Sopenharmony_ci	while (acpi_dev) {
107762306a36Sopenharmony_ci		if (!strcmp(acpi_device_hid(acpi_dev),
107862306a36Sopenharmony_ci			    ACPI_PROCESSOR_CONTAINER_HID) &&
107962306a36Sopenharmony_ci		    !kstrtouint(acpi_device_uid(acpi_dev), 0, &acpi_uid) &&
108062306a36Sopenharmony_ci		    acpi_uid == container_uid)
108162306a36Sopenharmony_ci			return 0;
108262306a36Sopenharmony_ci
108362306a36Sopenharmony_ci		acpi_dev = acpi_dev_parent(acpi_dev);
108462306a36Sopenharmony_ci	}
108562306a36Sopenharmony_ci
108662306a36Sopenharmony_ci	return -ENODEV;
108762306a36Sopenharmony_ci}
108862306a36Sopenharmony_ci
108962306a36Sopenharmony_cistatic int arm_cspmu_acpi_get_cpus(struct arm_cspmu *cspmu)
109062306a36Sopenharmony_ci{
109162306a36Sopenharmony_ci	struct acpi_apmt_node *apmt_node;
109262306a36Sopenharmony_ci	int affinity_flag;
109362306a36Sopenharmony_ci	int cpu;
109462306a36Sopenharmony_ci
109562306a36Sopenharmony_ci	apmt_node = arm_cspmu_apmt_node(cspmu->dev);
109662306a36Sopenharmony_ci	affinity_flag = apmt_node->flags & ACPI_APMT_FLAGS_AFFINITY;
109762306a36Sopenharmony_ci
109862306a36Sopenharmony_ci	if (affinity_flag == ACPI_APMT_FLAGS_AFFINITY_PROC) {
109962306a36Sopenharmony_ci		for_each_possible_cpu(cpu) {
110062306a36Sopenharmony_ci			if (apmt_node->proc_affinity ==
110162306a36Sopenharmony_ci			    get_acpi_id_for_cpu(cpu)) {
110262306a36Sopenharmony_ci				cpumask_set_cpu(cpu, &cspmu->associated_cpus);
110362306a36Sopenharmony_ci				break;
110462306a36Sopenharmony_ci			}
110562306a36Sopenharmony_ci		}
110662306a36Sopenharmony_ci	} else {
110762306a36Sopenharmony_ci		for_each_possible_cpu(cpu) {
110862306a36Sopenharmony_ci			if (arm_cspmu_find_cpu_container(
110962306a36Sopenharmony_ci				    cpu, apmt_node->proc_affinity))
111062306a36Sopenharmony_ci				continue;
111162306a36Sopenharmony_ci
111262306a36Sopenharmony_ci			cpumask_set_cpu(cpu, &cspmu->associated_cpus);
111362306a36Sopenharmony_ci		}
111462306a36Sopenharmony_ci	}
111562306a36Sopenharmony_ci
111662306a36Sopenharmony_ci	if (cpumask_empty(&cspmu->associated_cpus)) {
111762306a36Sopenharmony_ci		dev_dbg(cspmu->dev, "No cpu associated with the PMU\n");
111862306a36Sopenharmony_ci		return -ENODEV;
111962306a36Sopenharmony_ci	}
112062306a36Sopenharmony_ci
112162306a36Sopenharmony_ci	return 0;
112262306a36Sopenharmony_ci}
112362306a36Sopenharmony_ci#else
112462306a36Sopenharmony_cistatic int arm_cspmu_acpi_get_cpus(struct arm_cspmu *cspmu)
112562306a36Sopenharmony_ci{
112662306a36Sopenharmony_ci	return -ENODEV;
112762306a36Sopenharmony_ci}
112862306a36Sopenharmony_ci#endif
112962306a36Sopenharmony_ci
113062306a36Sopenharmony_cistatic int arm_cspmu_get_cpus(struct arm_cspmu *cspmu)
113162306a36Sopenharmony_ci{
113262306a36Sopenharmony_ci	return arm_cspmu_acpi_get_cpus(cspmu);
113362306a36Sopenharmony_ci}
113462306a36Sopenharmony_ci
113562306a36Sopenharmony_cistatic int arm_cspmu_register_pmu(struct arm_cspmu *cspmu)
113662306a36Sopenharmony_ci{
113762306a36Sopenharmony_ci	int ret, capabilities;
113862306a36Sopenharmony_ci	struct attribute_group **attr_groups;
113962306a36Sopenharmony_ci
114062306a36Sopenharmony_ci	attr_groups = arm_cspmu_alloc_attr_group(cspmu);
114162306a36Sopenharmony_ci	if (!attr_groups)
114262306a36Sopenharmony_ci		return -ENOMEM;
114362306a36Sopenharmony_ci
114462306a36Sopenharmony_ci	ret = cpuhp_state_add_instance(arm_cspmu_cpuhp_state,
114562306a36Sopenharmony_ci				       &cspmu->cpuhp_node);
114662306a36Sopenharmony_ci	if (ret)
114762306a36Sopenharmony_ci		return ret;
114862306a36Sopenharmony_ci
114962306a36Sopenharmony_ci	capabilities = PERF_PMU_CAP_NO_EXCLUDE;
115062306a36Sopenharmony_ci	if (cspmu->irq == 0)
115162306a36Sopenharmony_ci		capabilities |= PERF_PMU_CAP_NO_INTERRUPT;
115262306a36Sopenharmony_ci
115362306a36Sopenharmony_ci	cspmu->pmu = (struct pmu){
115462306a36Sopenharmony_ci		.task_ctx_nr	= perf_invalid_context,
115562306a36Sopenharmony_ci		.module		= THIS_MODULE,
115662306a36Sopenharmony_ci		.pmu_enable	= arm_cspmu_enable,
115762306a36Sopenharmony_ci		.pmu_disable	= arm_cspmu_disable,
115862306a36Sopenharmony_ci		.event_init	= arm_cspmu_event_init,
115962306a36Sopenharmony_ci		.add		= arm_cspmu_add,
116062306a36Sopenharmony_ci		.del		= arm_cspmu_del,
116162306a36Sopenharmony_ci		.start		= arm_cspmu_start,
116262306a36Sopenharmony_ci		.stop		= arm_cspmu_stop,
116362306a36Sopenharmony_ci		.read		= arm_cspmu_read,
116462306a36Sopenharmony_ci		.attr_groups	= (const struct attribute_group **)attr_groups,
116562306a36Sopenharmony_ci		.capabilities	= capabilities,
116662306a36Sopenharmony_ci	};
116762306a36Sopenharmony_ci
116862306a36Sopenharmony_ci	/* Hardware counter init */
116962306a36Sopenharmony_ci	arm_cspmu_stop_counters(cspmu);
117062306a36Sopenharmony_ci	arm_cspmu_reset_counters(cspmu);
117162306a36Sopenharmony_ci
117262306a36Sopenharmony_ci	ret = perf_pmu_register(&cspmu->pmu, cspmu->name, -1);
117362306a36Sopenharmony_ci	if (ret) {
117462306a36Sopenharmony_ci		cpuhp_state_remove_instance(arm_cspmu_cpuhp_state,
117562306a36Sopenharmony_ci					    &cspmu->cpuhp_node);
117662306a36Sopenharmony_ci	}
117762306a36Sopenharmony_ci
117862306a36Sopenharmony_ci	return ret;
117962306a36Sopenharmony_ci}
118062306a36Sopenharmony_ci
118162306a36Sopenharmony_cistatic int arm_cspmu_device_probe(struct platform_device *pdev)
118262306a36Sopenharmony_ci{
118362306a36Sopenharmony_ci	int ret;
118462306a36Sopenharmony_ci	struct arm_cspmu *cspmu;
118562306a36Sopenharmony_ci
118662306a36Sopenharmony_ci	cspmu = arm_cspmu_alloc(pdev);
118762306a36Sopenharmony_ci	if (!cspmu)
118862306a36Sopenharmony_ci		return -ENOMEM;
118962306a36Sopenharmony_ci
119062306a36Sopenharmony_ci	ret = arm_cspmu_init_mmio(cspmu);
119162306a36Sopenharmony_ci	if (ret)
119262306a36Sopenharmony_ci		return ret;
119362306a36Sopenharmony_ci
119462306a36Sopenharmony_ci	ret = arm_cspmu_request_irq(cspmu);
119562306a36Sopenharmony_ci	if (ret)
119662306a36Sopenharmony_ci		return ret;
119762306a36Sopenharmony_ci
119862306a36Sopenharmony_ci	ret = arm_cspmu_get_cpus(cspmu);
119962306a36Sopenharmony_ci	if (ret)
120062306a36Sopenharmony_ci		return ret;
120162306a36Sopenharmony_ci
120262306a36Sopenharmony_ci	ret = arm_cspmu_register_pmu(cspmu);
120362306a36Sopenharmony_ci	if (ret)
120462306a36Sopenharmony_ci		return ret;
120562306a36Sopenharmony_ci
120662306a36Sopenharmony_ci	return 0;
120762306a36Sopenharmony_ci}
120862306a36Sopenharmony_ci
120962306a36Sopenharmony_cistatic int arm_cspmu_device_remove(struct platform_device *pdev)
121062306a36Sopenharmony_ci{
121162306a36Sopenharmony_ci	struct arm_cspmu *cspmu = platform_get_drvdata(pdev);
121262306a36Sopenharmony_ci
121362306a36Sopenharmony_ci	perf_pmu_unregister(&cspmu->pmu);
121462306a36Sopenharmony_ci	cpuhp_state_remove_instance(arm_cspmu_cpuhp_state, &cspmu->cpuhp_node);
121562306a36Sopenharmony_ci
121662306a36Sopenharmony_ci	return 0;
121762306a36Sopenharmony_ci}
121862306a36Sopenharmony_ci
121962306a36Sopenharmony_cistatic const struct platform_device_id arm_cspmu_id[] = {
122062306a36Sopenharmony_ci	{DRVNAME, 0},
122162306a36Sopenharmony_ci	{ },
122262306a36Sopenharmony_ci};
122362306a36Sopenharmony_ciMODULE_DEVICE_TABLE(platform, arm_cspmu_id);
122462306a36Sopenharmony_ci
122562306a36Sopenharmony_cistatic struct platform_driver arm_cspmu_driver = {
122662306a36Sopenharmony_ci	.driver = {
122762306a36Sopenharmony_ci			.name = DRVNAME,
122862306a36Sopenharmony_ci			.suppress_bind_attrs = true,
122962306a36Sopenharmony_ci		},
123062306a36Sopenharmony_ci	.probe = arm_cspmu_device_probe,
123162306a36Sopenharmony_ci	.remove = arm_cspmu_device_remove,
123262306a36Sopenharmony_ci	.id_table = arm_cspmu_id,
123362306a36Sopenharmony_ci};
123462306a36Sopenharmony_ci
123562306a36Sopenharmony_cistatic void arm_cspmu_set_active_cpu(int cpu, struct arm_cspmu *cspmu)
123662306a36Sopenharmony_ci{
123762306a36Sopenharmony_ci	cpumask_set_cpu(cpu, &cspmu->active_cpu);
123862306a36Sopenharmony_ci	if (cspmu->irq)
123962306a36Sopenharmony_ci		WARN_ON(irq_set_affinity(cspmu->irq, &cspmu->active_cpu));
124062306a36Sopenharmony_ci}
124162306a36Sopenharmony_ci
124262306a36Sopenharmony_cistatic int arm_cspmu_cpu_online(unsigned int cpu, struct hlist_node *node)
124362306a36Sopenharmony_ci{
124462306a36Sopenharmony_ci	struct arm_cspmu *cspmu =
124562306a36Sopenharmony_ci		hlist_entry_safe(node, struct arm_cspmu, cpuhp_node);
124662306a36Sopenharmony_ci
124762306a36Sopenharmony_ci	if (!cpumask_test_cpu(cpu, &cspmu->associated_cpus))
124862306a36Sopenharmony_ci		return 0;
124962306a36Sopenharmony_ci
125062306a36Sopenharmony_ci	/* If the PMU is already managed, there is nothing to do */
125162306a36Sopenharmony_ci	if (!cpumask_empty(&cspmu->active_cpu))
125262306a36Sopenharmony_ci		return 0;
125362306a36Sopenharmony_ci
125462306a36Sopenharmony_ci	/* Use this CPU for event counting */
125562306a36Sopenharmony_ci	arm_cspmu_set_active_cpu(cpu, cspmu);
125662306a36Sopenharmony_ci
125762306a36Sopenharmony_ci	return 0;
125862306a36Sopenharmony_ci}
125962306a36Sopenharmony_ci
126062306a36Sopenharmony_cistatic int arm_cspmu_cpu_teardown(unsigned int cpu, struct hlist_node *node)
126162306a36Sopenharmony_ci{
126262306a36Sopenharmony_ci	int dst;
126362306a36Sopenharmony_ci	struct cpumask online_supported;
126462306a36Sopenharmony_ci
126562306a36Sopenharmony_ci	struct arm_cspmu *cspmu =
126662306a36Sopenharmony_ci		hlist_entry_safe(node, struct arm_cspmu, cpuhp_node);
126762306a36Sopenharmony_ci
126862306a36Sopenharmony_ci	/* Nothing to do if this CPU doesn't own the PMU */
126962306a36Sopenharmony_ci	if (!cpumask_test_and_clear_cpu(cpu, &cspmu->active_cpu))
127062306a36Sopenharmony_ci		return 0;
127162306a36Sopenharmony_ci
127262306a36Sopenharmony_ci	/* Choose a new CPU to migrate ownership of the PMU to */
127362306a36Sopenharmony_ci	cpumask_and(&online_supported, &cspmu->associated_cpus,
127462306a36Sopenharmony_ci		    cpu_online_mask);
127562306a36Sopenharmony_ci	dst = cpumask_any_but(&online_supported, cpu);
127662306a36Sopenharmony_ci	if (dst >= nr_cpu_ids)
127762306a36Sopenharmony_ci		return 0;
127862306a36Sopenharmony_ci
127962306a36Sopenharmony_ci	/* Use this CPU for event counting */
128062306a36Sopenharmony_ci	perf_pmu_migrate_context(&cspmu->pmu, cpu, dst);
128162306a36Sopenharmony_ci	arm_cspmu_set_active_cpu(dst, cspmu);
128262306a36Sopenharmony_ci
128362306a36Sopenharmony_ci	return 0;
128462306a36Sopenharmony_ci}
128562306a36Sopenharmony_ci
128662306a36Sopenharmony_cistatic int __init arm_cspmu_init(void)
128762306a36Sopenharmony_ci{
128862306a36Sopenharmony_ci	int ret;
128962306a36Sopenharmony_ci
129062306a36Sopenharmony_ci	ret = cpuhp_setup_state_multi(CPUHP_AP_ONLINE_DYN,
129162306a36Sopenharmony_ci					"perf/arm/cspmu:online",
129262306a36Sopenharmony_ci					arm_cspmu_cpu_online,
129362306a36Sopenharmony_ci					arm_cspmu_cpu_teardown);
129462306a36Sopenharmony_ci	if (ret < 0)
129562306a36Sopenharmony_ci		return ret;
129662306a36Sopenharmony_ci	arm_cspmu_cpuhp_state = ret;
129762306a36Sopenharmony_ci	return platform_driver_register(&arm_cspmu_driver);
129862306a36Sopenharmony_ci}
129962306a36Sopenharmony_ci
130062306a36Sopenharmony_cistatic void __exit arm_cspmu_exit(void)
130162306a36Sopenharmony_ci{
130262306a36Sopenharmony_ci	platform_driver_unregister(&arm_cspmu_driver);
130362306a36Sopenharmony_ci	cpuhp_remove_multi_state(arm_cspmu_cpuhp_state);
130462306a36Sopenharmony_ci}
130562306a36Sopenharmony_ci
130662306a36Sopenharmony_cimodule_init(arm_cspmu_init);
130762306a36Sopenharmony_cimodule_exit(arm_cspmu_exit);
130862306a36Sopenharmony_ci
130962306a36Sopenharmony_ciMODULE_LICENSE("GPL v2");
1310