blob: c0897463bb8826cb35ae23d9fcfa294058780bee [file] [log] [blame]
Sandrine Bailleux3cd87d72018-10-09 11:12:55 +02001/*
Boyan Karatotev4e282422024-10-25 14:34:13 +01002 * Copyright (c) 2018-2024, Arm Limited. All rights reserved.
Sandrine Bailleux3cd87d72018-10-09 11:12:55 +02003 *
4 * SPDX-License-Identifier: BSD-3-Clause
5 */
6
7#include <amu.h>
8#include <amu_private.h>
9#include <arch.h>
Boyan Karatotev4e282422024-10-25 14:34:13 +010010#include <arch_features.h>
Sandrine Bailleux3cd87d72018-10-09 11:12:55 +020011#include <arch_helpers.h>
12#include <assert.h>
13
Sandrine Bailleux3cd87d72018-10-09 11:12:55 +020014
johpow01b7d752a2020-10-08 17:29:11 -050015/* Check if group 1 counters is implemented */
16int amu_group1_supported(void)
17{
18 uint64_t features = read_amcfgr_el0() >> AMCFGR_EL0_NCG_SHIFT;
19
20 return (features & AMCFGR_EL0_NCG_MASK) == 1U;
Sandrine Bailleux3cd87d72018-10-09 11:12:55 +020021}
22
23/* Read the group 0 counter identified by the given `idx`. */
johpow01b7d752a2020-10-08 17:29:11 -050024uint64_t amu_group0_cnt_read(unsigned int idx)
Sandrine Bailleux3cd87d72018-10-09 11:12:55 +020025{
johpow01b7d752a2020-10-08 17:29:11 -050026 assert(amu_get_version() != ID_AA64PFR0_AMU_NOT_SUPPORTED);
27 assert(idx < AMU_GROUP0_NR_COUNTERS);
Sandrine Bailleux3cd87d72018-10-09 11:12:55 +020028
29 return amu_group0_cnt_read_internal(idx);
30}
31
johpow01b7d752a2020-10-08 17:29:11 -050032/*
33 * Read the group 0 offset register for a given index. Index must be 0, 2, or
34 * 3, the register for 1 does not exist.
35 *
36 * Using this function requires v8.6 FEAT_AMUv1p1 support.
37 */
38uint64_t amu_group0_voffset_read(unsigned int idx)
Sandrine Bailleux3cd87d72018-10-09 11:12:55 +020039{
johpow01b7d752a2020-10-08 17:29:11 -050040 assert(amu_get_version() >= ID_AA64PFR0_AMU_V1P1);
41 assert(idx < AMU_GROUP0_NR_COUNTERS);
42 assert(idx != 1U);
43
44 return amu_group0_voffset_read_internal(idx);
45}
46
47/*
48 * Write the group 0 offset register for a given index. Index must be 0, 2, or
49 * 3, the register for 1 does not exist.
50 *
51 * Using this function requires v8.6 FEAT_AMUv1p1 support.
52 */
53void amu_group0_voffset_write(unsigned int idx, uint64_t val)
54{
55 assert(amu_get_version() >= ID_AA64PFR0_AMU_V1P1);
56 assert(idx < AMU_GROUP0_NR_COUNTERS);
57 assert(idx != 1U);
58
59 amu_group0_voffset_write_internal(idx, val);
60 isb();
61}
62
63/* Read the group 1 counter identified by the given `idx`. */
64uint64_t amu_group1_cnt_read(unsigned int idx)
65{
66 assert(amu_get_version() != ID_AA64PFR0_AMU_NOT_SUPPORTED);
Juan Pablo Condec3cf2da2024-04-01 13:57:19 -050067 assert(amu_group1_supported());
68 assert(idx < amu_group1_num_counters());
Sandrine Bailleux3cd87d72018-10-09 11:12:55 +020069
70 return amu_group1_cnt_read_internal(idx);
71}
johpow01b7d752a2020-10-08 17:29:11 -050072
Juan Pablo Condec3cf2da2024-04-01 13:57:19 -050073/* Return the number of counters available for group 1 */
74uint64_t amu_group1_num_counters(void)
75{
76 assert(amu_get_version() != ID_AA64PFR0_AMU_NOT_SUPPORTED);
77 assert(amu_group1_supported());
78
79 uint64_t num_counters = amu_group1_num_counters_internal();
80 if (num_counters < AMU_GROUP1_NR_COUNTERS) {
81 return num_counters;
82 }
83 return AMU_GROUP1_NR_COUNTERS;
84}
85
86/* Return the type for group 1 counter with index `idx`. */
87uint64_t amu_group1_evtype_read(unsigned int idx)
88{
89 assert(amu_get_version() != ID_AA64PFR0_AMU_NOT_SUPPORTED);
90 assert(amu_group1_supported());
91 assert(idx < amu_group1_num_counters());
92
93 return amu_group1_evtype_read_internal(idx);
94}
95
96/* Set the type for group 1 counter with index `idx`. */
97void amu_group1_evtype_write(unsigned int idx, uint64_t val)
98{
99 assert(amu_get_version() != ID_AA64PFR0_AMU_NOT_SUPPORTED);
100 assert(amu_group1_supported());
101 assert(idx < amu_group1_num_counters());
102
103 amu_group1_evtype_write_internal(idx, val);
104}
105
106/*
107 * Return whether group 1 counter at index `idx` is implemented.
108 *
109 * Using this function requires v8.6 FEAT_AMUv1p1 support.
110 */
111uint64_t amu_group1_is_counter_implemented(unsigned int idx)
112{
113 assert(amu_get_version() >= ID_AA64PFR0_AMU_V1P1);
114 assert(amu_group1_supported());
115
116 return amu_group1_is_cnt_impl_internal(idx);
117}
118
johpow01b7d752a2020-10-08 17:29:11 -0500119/*
120 * Read the group 1 offset register for a given index.
121 *
122 * Using this function requires v8.6 FEAT_AMUv1p1 support.
123 */
124uint64_t amu_group1_voffset_read(unsigned int idx)
125{
126 assert(amu_get_version() >= ID_AA64PFR0_AMU_V1P1);
127 assert(amu_group1_supported());
128 assert(idx < AMU_GROUP1_NR_COUNTERS);
129 assert(((read_amcg1idr_el0() >> AMCG1IDR_VOFF_SHIFT) &
130 (1U << idx)) != 0U);
131
132 return amu_group1_voffset_read_internal(idx);
133}
134
135/*
136 * Write the group 1 offset register for a given index.
137 *
138 * Using this function requires v8.6 FEAT_AMUv1p1 support.
139 */
140void amu_group1_voffset_write(unsigned int idx, uint64_t val)
141{
142 assert(amu_get_version() >= ID_AA64PFR0_AMU_V1P1);
143 assert(amu_group1_supported());
144 assert(idx < AMU_GROUP1_NR_COUNTERS);
145 assert(((read_amcg1idr_el0() >> AMCG1IDR_VOFF_SHIFT) &
146 (1U << idx)) != 0U);
147
148 amu_group1_voffset_write_internal(idx, val);
149 isb();
150}