[PATCH/RFC] arm64: pmu: add Qualcomm Technologies extensions
Leeder, Neil
nleeder at codeaurora.org
Wed Mar 1 13:36:07 PST 2017
Hi Mark,
Thanks for the quick response.
On 3/1/2017 1:10 PM, Mark Rutland wrote:
> Hi Neil,
>
> On Wed, Mar 01, 2017 at 11:18:05AM -0500, Neil Leeder wrote:
>> Adds CPU PMU perf events support for Qualcomm Technologies' Falkor CPU.
>>
>> The Qualcomm Technologies CPU PMU is named qcom_pmuv3 and provides
>> extensions to the architected PMU events.
>
> Is this is a strict superset of PMUv3 (that could validly be treated as
> just PMUv3), or do those IMP DEF parts need to be poked to use this at
> all?
>
> What is reported by ID_AA64DFR0_EL1.PMUVer on these CPUs?
It's a strict superset. If you don't use any of the extensions than it
behaves as a PMUv3 for architected events. ID_AA64DFR0_EL1.PMUVer = 1.
> Quite frankly, I'm less than thrilled about the prospect of
> IMPLEMENTATION DEFINED CPU PMUs that fall outside of the architected PMU
> model, especially for ACPI systems where the raison d'être is standards
> and uniformity, and where we have no sensible mechanism to provide
> information regarding IMPLEMENTATION DEFINED functionality.
>
> This has knock-on effects for other things, like userspace PMU access
> and/or virtualization, and this multiplies the support effort.
>
> KVM already has (architected) PMU support, and without a corresponding
> KVM patch this is at best insufficient. I don't imagine the KVM folk
> will be too thrilled about the prospect of emulating an IMPLEMENTATION
> DEFINED CPU feature like this.
Does KVM handle ARMv7 PMU implementations? If so, do you know what it
does for the scorpion_* and krait_* implementations in
arch/arm/kernel/perf_events_v7.c? These extensions in ARMv8 are very
similar to the krait extensions, with some 64-bit tweaks, so could be
handled by KVM the same way it handles the ARMv7 cases.
I'm not sure about userspace changes - these extensions use a different
config format to specify an event, but otherwise are transparent to
userspace.
[...]
>> The Qualcomm Technologies CPU PMU extensions have an additional set of registers
>> which need to be programmed when configuring an event. These are the PMRESRs,
>> which are similar to the krait & scorpion registers in armv7, and the L2
>> variants in the Qualcomm Technologies L2 PMU driver.
>
> If these *must* be programmed, it sounds like this isn't PMUv3
> compatible.
Sorry for the imprecise wording. They only have to be programmed when
using an event in these extensions, not for architected events.
>> There are additional constraints on qc events. The armv7 implementation checks
>> these in get_event_idx, but during L2 PMU reviews it was thought better to do
>> these during init processing where possible. I added these in the map_event
>> callback because its the only callback from within armpmu_event_init(). I'm not
>> sure if that would be considered stretching the acceptable use of that interface,
>> so I'm open to other suggestions.
>
> As a general rule for PMU drivers:
>
> * pmu::event_init() should check whether the entire group the event is
> in (i.e. the parent and all siblings) can potentially be allocated
> into counters simultaneously. If it is always impossible for the whole
> group to go on, pmu::event_init should fail, since the group is
> impossible.
>
> * pmu::add() needs to handle inter-group and intra-group conflicts that
> can arise dynamically dependent on how (other) events are scheduled.
> This also needs to fail in the event of a conflict.
>
Checking whether there is a conflict within the group is what
qc_verify_event() does, as it's called from the map_event callback.
[...]
>> This requires Jeremy Linton's patch sequence to add arm64 CPU PMU ACPI support:
>> https://patchwork.kernel.org/patch/9533677/
>
> As a heads-up, I'm currently working on an alternative approach that you
> can find at:
>
> git://git.kernel.org/pub/scm/linux/kernel/git/mark/linux.git arm/perf/acpi
>
> That's a work-in-progress, and there are a few issues (notably IRQ
> management) that I'm currently fixing up. I also intend to add a PMUv3
> check to the PMUv3 probe.
Thanks - I'll check this out and I'll be interested to see the final
version.
>> +static bool qc_pmu;
>
> Sorry, but a global boolean to describe whether a (single ?) PMU
> instance is a particular implementation is not going to fly.
>
Yeah, I wasn't too happy about that, so I was looking for alternatives.
Duplicate the enable/disable_event() functions and add qc-specific
processing there? Add additional callbacks to be called from within
armv8pmu_enable/disable_event and only populate them for the qc case?
Something else?
[...]
>
>> +static void qc_pmu_reset(void *info)
>> +{
>> + qc_clear_resrs();
>> + armv8pmu_reset(info);
>> +}
>
> Is the QC-specific reset required only if we use the QC-specific events,
> or is that required for the standard PMUv3 features to be usable?
>
> Are standard PMUv3 events guaranteed to work if we didn't call
> qc_clear_resrs()?
>
> If this is not required for the standard PMUv3 features, and/or any IMP
> DEF reset is performed by FW, it looks like we *may* be able to treat
> this as PMUv3.
This reset is only required for the QC extensions.
>> +static void qc_pmu_enable_event(struct perf_event *event,
>> + struct hw_perf_event *hwc, int idx)
>> +{
>> + unsigned int reg, code, group;
>> +
>> + if (QC_EVT_PFX(hwc->config_base) != QC_EVT_PREFIX) {
>> + armv8pmu_write_evtype(idx, hwc->config_base);
>> + return;
>> + }
>
> This really shows that this is not a workable structure. It's hideous to
> hook the PMUv3 code to call this, then try to duplicate what the PMUv3
> code would have done in this case.
I can rework this once there's an acceptable way to handle the
qc-specific enable/disables.
>> static const struct of_device_id armv8_pmu_of_device_ids[] = {
>> {.compatible = "arm,armv8-pmuv3", .data = armv8_pmuv3_init},
>> {.compatible = "arm,cortex-a53-pmu", .data = armv8_a53_pmu_init},
>> @@ -1087,6 +1421,8 @@ static int armv8_vulcan_pmu_init(struct arm_pmu *cpu_pmu)
>> * aren't supported by the current PMU are disabled.
>> */
>> static const struct pmu_probe_info armv8_pmu_probe_table[] = {
>> + PMU_PROBE(QCOM_CPU_PART_FALKOR_V1 << MIDR_PARTNUM_SHIFT,
>> + MIDR_PARTNUM_MASK, armv8_falkor_pmu_init),
>> PMU_PROBE(0, 0, armv8_pmuv3_init), /* enable all defined counters */
>> { /* sentinel value */ }
>
> We don't special-case other PMUs here, and the plan for ACPI was to
> rely solely on the architectural information, i.e. the architected ID
> registers associated with PMUv3.
>
> I don't think we should special-case implementations like this.
> My series removes this MIDR matching.
So would there be equivalent ACPI support for the various 3rd-party
implementations (vulcan, thunder... and then qc) as there is with device
tree?
> Thanks,
> Mark.
>
Thanks,
Neil
--
Qualcomm Datacenter Technologies, Inc. as an affiliate of Qualcomm
Technologies Inc.
Qualcomm Technologies, Inc. is a member of the Code Aurora Forum,
a Linux Foundation Collaborative Project.
More information about the linux-arm-kernel
mailing list