diff options
| -rw-r--r-- | Documentation/arm64/silicon-errata.txt | 1 | ||||
| -rw-r--r-- | drivers/acpi/arm64/iort.c | 16 | ||||
| -rw-r--r-- | drivers/perf/arm_smmuv3_pmu.c | 48 | ||||
| -rw-r--r-- | include/linux/acpi_iort.h | 1 | 
4 files changed, 58 insertions, 8 deletions
diff --git a/Documentation/arm64/silicon-errata.txt b/Documentation/arm64/silicon-errata.txt index d1e2bb801e1b..c00efb639e46 100644 --- a/Documentation/arm64/silicon-errata.txt +++ b/Documentation/arm64/silicon-errata.txt @@ -77,6 +77,7 @@ stable kernels.  | Hisilicon      | Hip0{5,6,7}     | #161010101      | HISILICON_ERRATUM_161010101 |  | Hisilicon      | Hip0{6,7}       | #161010701      | N/A                         |  | Hisilicon      | Hip07           | #161600802      | HISILICON_ERRATUM_161600802 | +| Hisilicon      | Hip08 SMMU PMCG | #162001800      | N/A                         |  |                |                 |                 |                             |  | Qualcomm Tech. | Kryo/Falkor v1  | E1003           | QCOM_FALKOR_ERRATUM_1003    |  | Qualcomm Tech. | Falkor v1       | E1009           | QCOM_FALKOR_ERRATUM_1009    | diff --git a/drivers/acpi/arm64/iort.c b/drivers/acpi/arm64/iort.c index e2c9b26bbee6..2d70b349bd6c 100644 --- a/drivers/acpi/arm64/iort.c +++ b/drivers/acpi/arm64/iort.c @@ -1366,9 +1366,23 @@ static void __init arm_smmu_v3_pmcg_init_resources(struct resource *res,  				       ACPI_EDGE_SENSITIVE, &res[2]);  } +static struct acpi_platform_list pmcg_plat_info[] __initdata = { +	/* HiSilicon Hip08 Platform */ +	{"HISI  ", "HIP08   ", 0, ACPI_SIG_IORT, greater_than_or_equal, +	 "Erratum #162001800", IORT_SMMU_V3_PMCG_HISI_HIP08}, +	{ } +}; +  static int __init arm_smmu_v3_pmcg_add_platdata(struct platform_device *pdev)  { -	u32 model = IORT_SMMU_V3_PMCG_GENERIC; +	u32 model; +	int idx; + +	idx = acpi_match_platform_list(pmcg_plat_info); +	if (idx >= 0) +		model = pmcg_plat_info[idx].data; +	else +		model = IORT_SMMU_V3_PMCG_GENERIC;  	return platform_device_add_data(pdev, &model, sizeof(model));  } diff --git a/drivers/perf/arm_smmuv3_pmu.c b/drivers/perf/arm_smmuv3_pmu.c index a4f4b488a2de..da71c741cb46 100644 --- a/drivers/perf/arm_smmuv3_pmu.c +++ b/drivers/perf/arm_smmuv3_pmu.c @@ -35,6 +35,7 @@   */  #include <linux/acpi.h> +#include <linux/acpi_iort.h>  #include <linux/bitfield.h>  #include <linux/bitops.h>  #include <linux/cpuhotplug.h> @@ -93,6 +94,8 @@  #define SMMU_PMCG_PA_SHIFT              12 +#define SMMU_PMCG_EVCNTR_RDONLY         BIT(0) +  static int cpuhp_state_num;  struct smmu_pmu { @@ -108,6 +111,7 @@ struct smmu_pmu {  	void __iomem *reg_base;  	void __iomem *reloc_base;  	u64 counter_mask; +	u32 options;  	bool global_filter;  	u32 global_filter_span;  	u32 global_filter_sid; @@ -222,15 +226,27 @@ static void smmu_pmu_set_period(struct smmu_pmu *smmu_pmu,  	u32 idx = hwc->idx;  	u64 new; -	/* -	 * We limit the max period to half the max counter value of the counter -	 * size, so that even in the case of extreme interrupt latency the -	 * counter will (hopefully) not wrap past its initial value. -	 */ -	new = smmu_pmu->counter_mask >> 1; +	if (smmu_pmu->options & SMMU_PMCG_EVCNTR_RDONLY) { +		/* +		 * On platforms that require this quirk, if the counter starts +		 * at < half_counter value and wraps, the current logic of +		 * handling the overflow may not work. It is expected that, +		 * those platforms will have full 64 counter bits implemented +		 * so that such a possibility is remote(eg: HiSilicon HIP08). +		 */ +		new = smmu_pmu_counter_get_value(smmu_pmu, idx); +	} else { +		/* +		 * We limit the max period to half the max counter value +		 * of the counter size, so that even in the case of extreme +		 * interrupt latency the counter will (hopefully) not wrap +		 * past its initial value. +		 */ +		new = smmu_pmu->counter_mask >> 1; +		smmu_pmu_counter_set_value(smmu_pmu, idx, new); +	}  	local64_set(&hwc->prev_count, new); -	smmu_pmu_counter_set_value(smmu_pmu, idx, new);  }  static void smmu_pmu_set_event_filter(struct perf_event *event, @@ -665,6 +681,22 @@ static void smmu_pmu_reset(struct smmu_pmu *smmu_pmu)  		       smmu_pmu->reloc_base + SMMU_PMCG_OVSCLR0);  } +static void smmu_pmu_get_acpi_options(struct smmu_pmu *smmu_pmu) +{ +	u32 model; + +	model = *(u32 *)dev_get_platdata(smmu_pmu->dev); + +	switch (model) { +	case IORT_SMMU_V3_PMCG_HISI_HIP08: +		/* HiSilicon Erratum 162001800 */ +		smmu_pmu->options |= SMMU_PMCG_EVCNTR_RDONLY; +		break; +	} + +	dev_notice(smmu_pmu->dev, "option mask 0x%x\n", smmu_pmu->options); +} +  static int smmu_pmu_probe(struct platform_device *pdev)  {  	struct smmu_pmu *smmu_pmu; @@ -744,6 +776,8 @@ static int smmu_pmu_probe(struct platform_device *pdev)  		return -EINVAL;  	} +	smmu_pmu_get_acpi_options(smmu_pmu); +  	/* Pick one CPU to be the preferred one to use */  	smmu_pmu->on_cpu = raw_smp_processor_id();  	WARN_ON(irq_set_affinity_hint(smmu_pmu->irq, diff --git a/include/linux/acpi_iort.h b/include/linux/acpi_iort.h index 052ef7b9f985..723e4dfa1c14 100644 --- a/include/linux/acpi_iort.h +++ b/include/linux/acpi_iort.h @@ -32,6 +32,7 @@   * do with hardware or with IORT specification.   */  #define IORT_SMMU_V3_PMCG_GENERIC        0x00000000 /* Generic SMMUv3 PMCG */ +#define IORT_SMMU_V3_PMCG_HISI_HIP08     0x00000001 /* HiSilicon HIP08 PMCG */  int iort_register_domain_token(int trans_id, phys_addr_t base,  			       struct fwnode_handle *fw_node);  | 

