1 /*
2 * Copyright (c) 2022-2023 HPMicro
3 *
4 * SPDX-License-Identifier: BSD-3-Clause
5 *
6 */
7
8 #include "hpm_sysctl_drv.h"
9 #include "hpm_soc_feature.h"
10
11 #define SYSCTL_RESOURCE_GROUP0 0
12
13 #define SYSCTL_CPU_RELEASE_KEY(cpu) (0xC0BEF1A9UL | (((cpu) & 1) << 24))
14
sysctl_valid_cpu_index(uint8_t cpu)15 static inline bool sysctl_valid_cpu_index(uint8_t cpu)
16 {
17 if (cpu != SYSCTL_CPU_CPU0) {
18 return false;
19 }
20 return true;
21 }
22
sysctl_get_cpu_gpr(SYSCTL_Type * ptr,uint8_t cpu,uint32_t * data,uint32_t size)23 hpm_stat_t sysctl_get_cpu_gpr(SYSCTL_Type *ptr, uint8_t cpu, uint32_t *data, uint32_t size)
24 {
25 uint32_t i;
26 if ((!sysctl_valid_cpu_index(cpu)) || (size > ARRAY_SIZE(ptr->CPU[cpu].GPR))) {
27 return status_invalid_argument;
28 }
29 for (i = 0; i < size; i++) {
30 *(data + i) = ptr->CPU[cpu].GPR[i];
31 }
32 return status_success;
33 }
34
_sysctl_cpu_get_gpr(SYSCTL_Type * ptr,uint8_t cpu,uint8_t start,uint8_t count,uint32_t * data)35 static hpm_stat_t _sysctl_cpu_get_gpr(SYSCTL_Type *ptr, uint8_t cpu, uint8_t start, uint8_t count, uint32_t *data)
36 {
37 uint8_t i, size = ARRAY_SIZE(ptr->CPU[cpu].GPR);
38 if (!sysctl_valid_cpu_index(cpu) || (data == NULL) || !count || start > size || count > size ||
39 (start + count) > size) {
40 return status_invalid_argument;
41 }
42 for (i = 0; i < count; i++) {
43 *(data + i) = ptr->CPU[cpu].GPR[start + i];
44 }
45 return status_success;
46 }
47
sysctl_cpu0_get_gpr(SYSCTL_Type * ptr,uint8_t start,uint8_t count,uint32_t * data)48 hpm_stat_t sysctl_cpu0_get_gpr(SYSCTL_Type *ptr, uint8_t start, uint8_t count, uint32_t *data)
49 {
50 return _sysctl_cpu_get_gpr(ptr, 0, start, count, data);
51 }
52
sysctl_cpu1_get_gpr(SYSCTL_Type * ptr,uint8_t start,uint8_t count,uint32_t * data)53 hpm_stat_t sysctl_cpu1_get_gpr(SYSCTL_Type *ptr, uint8_t start, uint8_t count, uint32_t *data)
54 {
55 return _sysctl_cpu_get_gpr(ptr, 1, start, count, data);
56 }
57
_sysctl_cpu_set_gpr(SYSCTL_Type * ptr,uint8_t cpu,uint8_t start,uint8_t count,const uint32_t * data)58 static hpm_stat_t _sysctl_cpu_set_gpr(SYSCTL_Type *ptr, uint8_t cpu, uint8_t start, uint8_t count, const uint32_t *data)
59 {
60 uint8_t i, size = ARRAY_SIZE(ptr->CPU[cpu].GPR);
61 if (!sysctl_valid_cpu_index(cpu) || (data == NULL) || !count || start > size || count > size ||
62 (start + count) > size) {
63 return status_invalid_argument;
64 }
65 for (i = 0; i < count; i++) {
66 ptr->CPU[cpu].GPR[start + i] = *(data + i);
67 }
68 return status_success;
69 }
70
sysctl_cpu0_set_gpr(SYSCTL_Type * ptr,uint8_t start,uint8_t count,uint32_t * data,bool lock)71 hpm_stat_t sysctl_cpu0_set_gpr(SYSCTL_Type *ptr, uint8_t start, uint8_t count, uint32_t *data, bool lock)
72 {
73 hpm_stat_t stat = status_success;
74 uint16_t gpr_mask;
75 stat = _sysctl_cpu_set_gpr(ptr, 0, start, count, data);
76 if (stat != status_success) {
77 return stat;
78 }
79 if (lock) {
80 gpr_mask = ((1 << count) - 1) << start;
81 sysctl_cpu0_lock_gpr_with_mask(ptr, gpr_mask);
82 }
83 return stat;
84 }
85
sysctl_monitor_get_default_config(SYSCTL_Type * ptr,monitor_config_t * config)86 void sysctl_monitor_get_default_config(SYSCTL_Type *ptr, monitor_config_t *config)
87 {
88 (void) ptr;
89 config->mode = monitor_work_mode_record;
90 config->accuracy = monitor_accuracy_1khz;
91 config->reference = monitor_reference_24mhz;
92 config->divide_by = 1;
93 config->high_limit = 0;
94 config->low_limit = 0;
95 config->start_measure = true;
96 config->enable_output = false;
97 config->target = monitor_target_clk_top_cpu0;
98 }
99
sysctl_monitor_init(SYSCTL_Type * ptr,uint8_t monitor_index,monitor_config_t * config)100 void sysctl_monitor_init(SYSCTL_Type *ptr, uint8_t monitor_index, monitor_config_t *config)
101 {
102 ptr->MONITOR[monitor_index].CONTROL &= ~(SYSCTL_MONITOR_CONTROL_START_MASK | SYSCTL_MONITOR_CONTROL_OUTEN_MASK);
103
104 if (config->mode == monitor_work_mode_compare) {
105 ptr->MONITOR[monitor_index].HIGH_LIMIT = SYSCTL_MONITOR_HIGH_LIMIT_FREQUENCY_SET(config->high_limit);
106 ptr->MONITOR[monitor_index].LOW_LIMIT = SYSCTL_MONITOR_LOW_LIMIT_FREQUENCY_SET(config->low_limit);
107 }
108
109 ptr->MONITOR[monitor_index].CONTROL = (ptr->MONITOR[monitor_index].CONTROL &
110 ~(SYSCTL_MONITOR_CONTROL_DIV_MASK | SYSCTL_MONITOR_CONTROL_MODE_MASK | SYSCTL_MONITOR_CONTROL_ACCURACY_MASK |
111 SYSCTL_MONITOR_CONTROL_REFERENCE_MASK | SYSCTL_MONITOR_CONTROL_SELECTION_MASK)) |
112 (SYSCTL_MONITOR_CONTROL_DIV_SET(config->divide_by - 1) | SYSCTL_MONITOR_CONTROL_MODE_SET(config->mode) |
113 SYSCTL_MONITOR_CONTROL_ACCURACY_SET(config->accuracy) |
114 SYSCTL_MONITOR_CONTROL_REFERENCE_SET(config->reference) |
115 SYSCTL_MONITOR_CONTROL_START_SET(config->start_measure) |
116 SYSCTL_MONITOR_CONTROL_OUTEN_SET(config->enable_output) |
117 SYSCTL_MONITOR_CONTROL_SELECTION_SET(config->target));
118 }
119
sysctl_monitor_measure_frequency(SYSCTL_Type * ptr,uint8_t monitor_index,monitor_target_t target,bool enable_output)120 uint32_t sysctl_monitor_measure_frequency(SYSCTL_Type *ptr,
121 uint8_t monitor_index,
122 monitor_target_t target,
123 bool enable_output)
124 {
125 uint32_t frequency = 0;
126 monitor_config_t monitor = { 0 };
127 sysctl_monitor_get_default_config(ptr, &monitor);
128 monitor.target = target;
129 monitor.enable_output = enable_output;
130 sysctl_monitor_init(ptr, monitor_index, &monitor);
131 if (monitor_index < SYSCTL_SOC_MONITOR_SLICE_COUNT) {
132 frequency = sysctl_monitor_get_current_result(ptr, monitor_index);
133 }
134 return frequency;
135 }
136
sysctl_set_cpu_entry(SYSCTL_Type * ptr,uint8_t cpu,uint32_t entry)137 hpm_stat_t sysctl_set_cpu_entry(SYSCTL_Type *ptr, uint8_t cpu, uint32_t entry)
138 {
139 if (!sysctl_valid_cpu_index(cpu)) {
140 return status_invalid_argument;
141 }
142 ptr->CPU[cpu].GPR[0] = entry;
143 ptr->CPU[cpu].GPR[1] = SYSCTL_CPU_RELEASE_KEY(cpu);
144 return status_success;
145 }
146
sysctl_set_cpu0_wakeup_entry(SYSCTL_Type * ptr,uint32_t entry)147 hpm_stat_t sysctl_set_cpu0_wakeup_entry(SYSCTL_Type *ptr, uint32_t entry)
148 {
149 return sysctl_set_cpu_entry(ptr, 0, entry);
150 }
151
sysctl_enable_group_resource(SYSCTL_Type * ptr,uint8_t group,sysctl_resource_t resource,bool enable)152 hpm_stat_t sysctl_enable_group_resource(SYSCTL_Type *ptr,
153 uint8_t group,
154 sysctl_resource_t resource,
155 bool enable)
156 {
157 uint32_t index, offset;
158 if (resource < sysctl_resource_linkable_start) {
159 return status_invalid_argument;
160 }
161
162 index = (resource - sysctl_resource_linkable_start) / 32;
163 offset = (resource - sysctl_resource_linkable_start) % 32;
164 switch (group) {
165 case SYSCTL_RESOURCE_GROUP0:
166 ptr->GROUP0[index].VALUE = (ptr->GROUP0[index].VALUE & ~(1UL << offset)) | (enable ? (1UL << offset) : 0);
167 if (enable) {
168 while (sysctl_resource_target_is_busy(ptr, resource)) {
169 ;
170 }
171 }
172 break;
173 default:
174 return status_invalid_argument;
175 }
176
177 return status_success;
178 }
179
sysctl_check_group_resource_enable(SYSCTL_Type * ptr,uint8_t group,sysctl_resource_t resource)180 bool sysctl_check_group_resource_enable(SYSCTL_Type *ptr,
181 uint8_t group,
182 sysctl_resource_t resource)
183 {
184 uint32_t index, offset;
185 bool enable;
186
187 index = (resource - sysctl_resource_linkable_start) / 32;
188 offset = (resource - sysctl_resource_linkable_start) % 32;
189 switch (group) {
190 case SYSCTL_RESOURCE_GROUP0:
191 enable = ((ptr->GROUP0[index].VALUE & (1UL << offset)) != 0) ? true : false;
192 break;
193 default:
194 enable = false;
195 break;
196 }
197
198 return enable;
199 }
200
sysctl_get_group_resource_value(SYSCTL_Type * ptr,uint8_t group,uint8_t index)201 uint32_t sysctl_get_group_resource_value(SYSCTL_Type *ptr, uint8_t group, uint8_t index)
202 {
203 uint32_t value;
204 switch (group) {
205 case SYSCTL_RESOURCE_GROUP0:
206 value = ptr->GROUP0[index].VALUE;
207 break;
208 default:
209 value = 0;
210 break;
211 }
212 return value;
213 }
214
sysctl_add_resource_to_cpu0(SYSCTL_Type * ptr,sysctl_resource_t resource)215 hpm_stat_t sysctl_add_resource_to_cpu0(SYSCTL_Type *ptr, sysctl_resource_t resource)
216 {
217 return sysctl_enable_group_resource(ptr, SYSCTL_RESOURCE_GROUP0, resource, true);
218 }
219
sysctl_remove_resource_from_cpu0(SYSCTL_Type * ptr,sysctl_resource_t resource)220 hpm_stat_t sysctl_remove_resource_from_cpu0(SYSCTL_Type *ptr, sysctl_resource_t resource)
221 {
222 return sysctl_enable_group_resource(ptr, SYSCTL_RESOURCE_GROUP0, resource, false);
223 }
224
sysctl_update_divider(SYSCTL_Type * ptr,clock_node_t node,uint32_t divide_by)225 hpm_stat_t sysctl_update_divider(SYSCTL_Type *ptr, clock_node_t node, uint32_t divide_by)
226 {
227 if (node >= clock_node_adc_start) {
228 return status_invalid_argument;
229 }
230
231 ptr->CLOCK[node] = (ptr->CLOCK[node] & ~(SYSCTL_CLOCK_DIV_MASK)) | SYSCTL_CLOCK_DIV_SET(divide_by - 1);
232 while (sysctl_clock_target_is_busy(ptr, node)) {
233 }
234 return status_success;
235 }
236
sysctl_config_clock(SYSCTL_Type * ptr,clock_node_t node,clock_source_t source,uint32_t divide_by)237 hpm_stat_t sysctl_config_clock(SYSCTL_Type *ptr, clock_node_t node, clock_source_t source, uint32_t divide_by)
238 {
239 if (node >= clock_node_adc_start) {
240 return status_invalid_argument;
241 }
242
243 if (source >= clock_source_general_source_end) {
244 return status_invalid_argument;
245 }
246 ptr->CLOCK[node] = (ptr->CLOCK[node] & ~(SYSCTL_CLOCK_MUX_MASK | SYSCTL_CLOCK_DIV_MASK)) |
247 (SYSCTL_CLOCK_MUX_SET(source) | SYSCTL_CLOCK_DIV_SET(divide_by - 1));
248 while (sysctl_clock_target_is_busy(ptr, node)) {
249 }
250 return status_success;
251 }
252
sysctl_config_cpu0_domain_clock(SYSCTL_Type * ptr,clock_source_t source,uint32_t cpu_div,uint32_t ahb_sub_div)253 hpm_stat_t sysctl_config_cpu0_domain_clock(SYSCTL_Type *ptr,
254 clock_source_t source,
255 uint32_t cpu_div,
256 uint32_t ahb_sub_div)
257 {
258 if (source >= clock_source_general_source_end) {
259 return status_invalid_argument;
260 }
261
262 uint32_t origin_cpu_div = SYSCTL_CLOCK_CPU_DIV_GET(ptr->CLOCK_CPU[0]) + 1U;
263 if (origin_cpu_div == cpu_div) {
264 ptr->CLOCK_CPU[0] = SYSCTL_CLOCK_CPU_MUX_SET(source) | SYSCTL_CLOCK_CPU_DIV_SET(cpu_div) | SYSCTL_CLOCK_CPU_SUB0_DIV_SET(ahb_sub_div - 1);
265 while (sysctl_cpu_clock_any_is_busy(ptr)) {
266 }
267 }
268 ptr->CLOCK_CPU[0] = SYSCTL_CLOCK_CPU_MUX_SET(source) | SYSCTL_CLOCK_CPU_DIV_SET(cpu_div - 1) | SYSCTL_CLOCK_CPU_SUB0_DIV_SET(ahb_sub_div - 1);
269
270 while (sysctl_cpu_clock_any_is_busy(ptr)) {
271 }
272
273 return status_success;
274 }
275
sysctl_set_adc_clock_mux(SYSCTL_Type * ptr,clock_node_t node,clock_source_adc_t source)276 hpm_stat_t sysctl_set_adc_clock_mux(SYSCTL_Type *ptr, clock_node_t node, clock_source_adc_t source)
277 {
278 if (source >= clock_source_adc_clk_end) {
279 return status_invalid_argument;
280 }
281 uint32_t adc_index = (uint32_t) (node - clock_node_adc_start);
282 if (adc_index >= ARRAY_SIZE(ptr->ADCCLK)) {
283 return status_invalid_argument;
284 }
285
286 ptr->ADCCLK[adc_index] = (ptr->ADCCLK[adc_index] & ~SYSCTL_ADCCLK_MUX_MASK) | SYSCTL_ADCCLK_MUX_SET(source);
287
288 return status_success;
289 }
290
sysctl_set_dac_clock_mux(SYSCTL_Type * ptr,clock_node_t node,clock_source_dac_t source)291 hpm_stat_t sysctl_set_dac_clock_mux(SYSCTL_Type *ptr, clock_node_t node, clock_source_dac_t source)
292 {
293 if (source >= clock_source_dac_clk_end) {
294 return status_invalid_argument;
295 }
296 uint32_t dac_index = (uint32_t) (node - clock_node_dac_start);
297 if (dac_index >= ARRAY_SIZE(ptr->DACCLK)) {
298 return status_invalid_argument;
299 }
300
301 ptr->DACCLK[dac_index] = (ptr->DACCLK[dac_index] & ~SYSCTL_DACCLK_MUX_MASK) | SYSCTL_DACCLK_MUX_SET(source);
302
303 return status_success;
304 }
305