[PATCH 2/4] perf/arm-cmn: Add CMN-650 support

Robin Murphy robin.murphy at arm.com
Thu Apr 21 02:46:05 PDT 2022


On 2022-04-21 08:25, Ilkka Koskinen wrote:
> 
> I still have a couple tiny comments. Otherwise the patch set looks good 
> to me.
> 
> On Mon, 18 Apr 2022, Robin Murphy wrote:
>> Add the identifiers and events for CMN-650, which slots into its
>> evolutionary position between CMN-600 and the 700-series products.
>> Imagine CMN-600 made bigger, and with most of the rough edges smoothed
>> off, but that then balanced out by some bonkers PMU functionality for
>> the new HN-P enhancement in CMN-650r2.
>>
>> Most of the CXG events are actually common to newer revisions of CMN-600
>> too, so they're arguably a little late; oh well.
>>
>> Signed-off-by: Robin Murphy <robin.murphy at arm.com>
>> ---
>> drivers/perf/arm-cmn.c | 222 ++++++++++++++++++++++++++++++++---------
>> 1 file changed, 176 insertions(+), 46 deletions(-)
>>
>> diff --git a/drivers/perf/arm-cmn.c b/drivers/perf/arm-cmn.c
>> index 9c1d82be7a2f..cce8516d465c 100644
>> --- a/drivers/perf/arm-cmn.c
>> +++ b/drivers/perf/arm-cmn.c
>> @@ -39,7 +39,7 @@
>> #define CMN_CHILD_NODE_ADDR        GENMASK(27, 0)
>> #define CMN_CHILD_NODE_EXTERNAL        BIT(31)
>>
>> -#define CMN_MAX_DIMENSION        8
>> +#define CMN_MAX_DIMENSION        12
>> #define CMN_MAX_XPS            (CMN_MAX_DIMENSION * CMN_MAX_DIMENSION)
>> #define CMN_MAX_DTMS            (CMN_MAX_XPS + (CMN_MAX_DIMENSION - 1) 
>> * 4)
>>
>> @@ -65,7 +65,9 @@
>>
>> /* For most nodes, this is all there is */
>> #define CMN_PMU_EVENT_SEL        0x000
>> -#define CMN_PMU_EVENTn_ID_SHIFT(n)    ((n) * 8)
>> +
>> +/* HN-Ps are weird... */
>> +#define CMN_HNP_PMU_EVENT_SEL        0x008
>>
>> /* DTMs live in the PMU space of XP registers */
>> #define CMN_DTM_WPn(n)            (0x1A0 + (n) * 0x18)
>> @@ -177,9 +179,12 @@
>>
>>
>> enum cmn_model {
>> -    CMN_ANY = -1,
>>     CMN600 = 1,
>> -    CI700 = 2,
>> +    CMN650 = 2,
>> +    CI700 = 8,
>> +    /* ...and then we can use bitmap tricks for commonality */
>> +    CMN_ANY = -1,
>> +    NOT_CMN600 = -2,
> 
> Matter of taste, but I'd probably prefer NOT_CMN600 = ~CMN600

Sure, I had it written that way at one point during development when it 
was a separate macro, so even I'm not entirely certain why it ended up 
like this - I must have been feeling particularly whimsical that day.

>> };
>>
>> /* CMN-600 r0px shouldn't exist in silicon, thankfully */
>> @@ -191,6 +196,11 @@ enum cmn_revision {
>>     CMN600_R2P0,
>>     CMN600_R3P0,
>>     CMN600_R3P1,
>> +    CMN650_R0P0 = 0,
>> +    CMN650_R1P0,
>> +    CMN650_R1P1,
>> +    CMN650_R2P0,
>> +    CMN650_R1P2,
>>     CI700_R0P0 = 0,
>>     CI700_R1P0,
>>     CI700_R2P0,
>> @@ -211,6 +221,7 @@ enum cmn_node_type {
>>     CMN_TYPE_RND = 0xd,
>>     CMN_TYPE_RNSAM = 0xf,
>>     CMN_TYPE_MTSX,
>> +    CMN_TYPE_HNP,
>>     CMN_TYPE_CXRA = 0x100,
>>     CMN_TYPE_CXHA = 0x101,
>>     CMN_TYPE_CXLA = 0x102,
>> @@ -307,9 +318,7 @@ struct arm_cmn_nodeid {
> 
> <snip>
> 
>> @@ -580,20 +592,25 @@ static umode_t 
>> arm_cmn_event_attr_is_visible(struct kobject *kobj,
>>     struct device *dev = kobj_to_dev(kobj);
>>     struct arm_cmn *cmn = to_cmn(dev_get_drvdata(dev));
>>     struct arm_cmn_event_attr *eattr;
>> +    enum cmn_node_type type;
>> +    u16 eventid;
>>
>>     eattr = container_of(attr, typeof(*eattr), attr.attr);
>>
>>     if (!(eattr->model & cmn->model))
>>         return 0;
>>
>> +    type = eattr->type;
>> +    eventid = eattr->eventid;
>> +
>>     /* Watchpoints aren't nodes, so avoid confusion */
>> -    if (eattr->type == CMN_TYPE_WP)
>> +    if (type == CMN_TYPE_WP)
>>         return attr->mode;
>>
>>     /* Hide XP events for unused interfaces/channels */
>> -    if (eattr->type == CMN_TYPE_XP) {
>> -        unsigned int intf = (eattr->eventid >> 2) & 7;
>> -        unsigned int chan = eattr->eventid >> 5;
>> +    if (type == CMN_TYPE_XP) {
>> +        unsigned int intf = (eventid >> 2) & 7;
>> +        unsigned int chan = eventid >> 5;
>>
>>         if ((intf & 4) && !(cmn->ports_used & BIT(intf & 3)))
>>             return 0;
>> @@ -607,12 +624,29 @@ static umode_t 
>> arm_cmn_event_attr_is_visible(struct kobject *kobj,
>>     }
>>
>>     /* Revision-specific differences */
>> -    if (cmn->model == CMN600 && cmn->rev < CMN600_R1P2) {
>> -        if (eattr->type == CMN_TYPE_HNF && eattr->eventid == 0x1b)
>> -            return 0;
>> +    if (cmn->model == CMN600) {
>> +        if (cmn->rev < CMN600_R1P3) {
>> +            if (type == CMN_TYPE_CXRA && eventid > 0x10)
>> +                return 0;
>> +        }
>> +        if (cmn->rev < CMN600_R1P2) {
>> +            if (type == CMN_TYPE_HNF && eventid == 0x1b)
>> +                return 0;
>> +            if (type == CMN_TYPE_CXRA || type == CMN_TYPE_CXHA)
>> +                return 0;
>> +        }
>> +    } else if (cmn->model == CMN650) {
>> +        if (cmn->rev < CMN650_R2P0 || cmn->rev == CMN650_R1P2) {
>> +            if (type == CMN_TYPE_HNF && eventid > 0x22)
>> +                return 0;
>> +            if (type == CMN_TYPE_SBSX && eventid == 0x17)
>> +                return 0;
>> +            if (type == CMN_TYPE_RNI && eventid > 0x10)
>> +                return 0;
>> +        }
> 
> What's the plan with cmn-650 r2p0 event settings? There seem to be a few 
> extra ones made visible now. I'm fine with updating this patch or taking 
> care of them in a separate one, which ever makes more sense.

arm_cmn_event_attrs *should* contain all the events supported by r2p0, 
with the ones not present on earlier revisions then being filtered out 
here. Have I missed anything? Note that HN-Ps don't need per-event 
filtering since that node type simply doesn't exist prior to CMN-650 r2, 
so is already filtered by arm_cmn_node().

Thanks,
Robin.



More information about the linux-arm-kernel mailing list