1 /*
2 * Copyright (c) 2022 Teslabs Engineering S.L.
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #define DT_DRV_COMPAT gd_gd32_cctl
8
9 #include <stdint.h>
10
11 #include <zephyr/arch/cpu.h>
12 #include <zephyr/device.h>
13 #include <zephyr/devicetree.h>
14 #include <zephyr/drivers/clock_control.h>
15 #include <zephyr/drivers/clock_control/gd32.h>
16
17 #include <gd32_regs.h>
18
19 /** RCU offset (from id cell) */
20 #define GD32_CLOCK_ID_OFFSET(id) (((id) >> 6U) & 0xFFU)
21 /** RCU configuration bit (from id cell) */
22 #define GD32_CLOCK_ID_BIT(id) ((id)&0x1FU)
23
24 #define CPU_FREQ DT_PROP(DT_PATH(cpus, cpu_0), clock_frequency)
25
26 /** AHB prescaler exponents */
27 static const uint8_t ahb_exp[16] = {
28 0U, 0U, 0U, 0U, 0U, 0U, 0U, 0U, 1U, 2U, 3U, 4U, 6U, 7U, 8U, 9U,
29 };
30 /** APB1 prescaler exponents */
31 static const uint8_t apb1_exp[8] = {
32 0U, 0U, 0U, 0U, 1U, 2U, 3U, 4U,
33 };
34 /** APB2 prescaler exponents */
35 static const uint8_t apb2_exp[8] = {
36 0U, 0U, 0U, 0U, 1U, 2U, 3U, 4U,
37 };
38
39 struct clock_control_gd32_config {
40 uint32_t base;
41 };
42
43 #if DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_timer)
44 /* timer identifiers */
45 #define TIMER_ID_OR_NONE(nodelabel) \
46 COND_CODE_1(DT_NODE_HAS_STATUS_OKAY(DT_NODELABEL(nodelabel)), \
47 (DT_CLOCKS_CELL(DT_NODELABEL(nodelabel), id),), ())
48
49 static const uint16_t timer_ids[] = {
50 TIMER_ID_OR_NONE(timer0) /* */
51 TIMER_ID_OR_NONE(timer1) /* */
52 TIMER_ID_OR_NONE(timer2) /* */
53 TIMER_ID_OR_NONE(timer3) /* */
54 TIMER_ID_OR_NONE(timer4) /* */
55 TIMER_ID_OR_NONE(timer5) /* */
56 TIMER_ID_OR_NONE(timer6) /* */
57 TIMER_ID_OR_NONE(timer7) /* */
58 TIMER_ID_OR_NONE(timer8) /* */
59 TIMER_ID_OR_NONE(timer9) /* */
60 TIMER_ID_OR_NONE(timer10) /* */
61 TIMER_ID_OR_NONE(timer11) /* */
62 TIMER_ID_OR_NONE(timer12) /* */
63 TIMER_ID_OR_NONE(timer13) /* */
64 TIMER_ID_OR_NONE(timer14) /* */
65 TIMER_ID_OR_NONE(timer15) /* */
66 TIMER_ID_OR_NONE(timer16) /* */
67 };
68 #endif /* DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_timer) */
69
clock_control_gd32_on(const struct device * dev,clock_control_subsys_t sys)70 static int clock_control_gd32_on(const struct device *dev,
71 clock_control_subsys_t sys)
72 {
73 const struct clock_control_gd32_config *config = dev->config;
74 uint16_t id = *(uint16_t *)sys;
75
76 sys_set_bit(config->base + GD32_CLOCK_ID_OFFSET(id),
77 GD32_CLOCK_ID_BIT(id));
78
79 return 0;
80 }
81
clock_control_gd32_off(const struct device * dev,clock_control_subsys_t sys)82 static int clock_control_gd32_off(const struct device *dev,
83 clock_control_subsys_t sys)
84 {
85 const struct clock_control_gd32_config *config = dev->config;
86 uint16_t id = *(uint16_t *)sys;
87
88 sys_clear_bit(config->base + GD32_CLOCK_ID_OFFSET(id),
89 GD32_CLOCK_ID_BIT(id));
90
91 return 0;
92 }
93
clock_control_gd32_get_rate(const struct device * dev,clock_control_subsys_t sys,uint32_t * rate)94 static int clock_control_gd32_get_rate(const struct device *dev,
95 clock_control_subsys_t sys,
96 uint32_t *rate)
97 {
98 const struct clock_control_gd32_config *config = dev->config;
99 uint16_t id = *(uint16_t *)sys;
100 uint32_t cfg;
101 uint8_t psc;
102
103 cfg = sys_read32(config->base + RCU_CFG0_OFFSET);
104
105 switch (GD32_CLOCK_ID_OFFSET(id)) {
106 #if defined(CONFIG_SOC_SERIES_GD32F4XX)
107 case RCU_AHB1EN_OFFSET:
108 case RCU_AHB2EN_OFFSET:
109 case RCU_AHB3EN_OFFSET:
110 #else
111 case RCU_AHBEN_OFFSET:
112 #endif
113 psc = (cfg & RCU_CFG0_AHBPSC_MSK) >> RCU_CFG0_AHBPSC_POS;
114 *rate = CPU_FREQ >> ahb_exp[psc];
115 break;
116 case RCU_APB1EN_OFFSET:
117 #if !defined(CONFIG_SOC_SERIES_GD32VF103) && \
118 !defined(CONFIG_SOC_SERIES_GD32A50X) && \
119 !defined(CONFIG_SOC_SERIES_GD32L23X)
120 case RCU_ADDAPB1EN_OFFSET:
121 #endif
122 psc = (cfg & RCU_CFG0_APB1PSC_MSK) >> RCU_CFG0_APB1PSC_POS;
123 *rate = CPU_FREQ >> apb1_exp[psc];
124 break;
125 case RCU_APB2EN_OFFSET:
126 psc = (cfg & RCU_CFG0_APB2PSC_MSK) >> RCU_CFG0_APB2PSC_POS;
127 *rate = CPU_FREQ >> apb2_exp[psc];
128 break;
129 default:
130 return -ENOTSUP;
131 }
132
133 #if DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_timer)
134 /* handle timer clocks */
135 for (size_t i = 0U; i < ARRAY_SIZE(timer_ids); i++) {
136 if (id != timer_ids[i]) {
137 continue;
138 }
139
140 #if defined(CONFIG_SOC_SERIES_GD32F4XX)
141 uint32_t cfg1 = sys_read32(config->base + RCU_CFG1_OFFSET);
142
143 /*
144 * The TIMERSEL bit in RCU_CFG1 controls the clock frequency of
145 * all the timers connected to the APB1 and APB2 domains.
146 *
147 * Up to a certain threshold value of APB{1,2} prescaler, timer
148 * clock equals to CK_AHB. This threshold value depends on
149 * TIMERSEL setting (2 if TIMERSEL=0, 4 if TIMERSEL=1). Above
150 * threshold, timer clock is set to a multiple of the APB
151 * domain clock CK_APB{1,2} (2 if TIMERSEL=0, 4 if TIMERSEL=1).
152 */
153
154 /* TIMERSEL = 0 */
155 if ((cfg1 & RCU_CFG1_TIMERSEL_MSK) == 0U) {
156 if (psc <= 2U) {
157 *rate = CPU_FREQ;
158 } else {
159 *rate *= 2U;
160 }
161 /* TIMERSEL = 1 */
162 } else {
163 if (psc <= 4U) {
164 *rate = CPU_FREQ;
165 } else {
166 *rate *= 4U;
167 }
168 }
169 #else
170 /*
171 * If the APB prescaler equals 1, the timer clock frequencies
172 * are set to the same frequency as that of the APB domain.
173 * Otherwise, they are set to twice the frequency of the APB
174 * domain.
175 */
176 if (psc != 1U) {
177 *rate *= 2U;
178 }
179 #endif /* CONFIG_SOC_SERIES_GD32F4XX */
180 }
181 #endif /* DT_HAS_COMPAT_STATUS_OKAY(gd_gd32_timer) */
182
183 return 0;
184 }
185
186 static enum clock_control_status
clock_control_gd32_get_status(const struct device * dev,clock_control_subsys_t sys)187 clock_control_gd32_get_status(const struct device *dev,
188 clock_control_subsys_t sys)
189 {
190 const struct clock_control_gd32_config *config = dev->config;
191 uint16_t id = *(uint16_t *)sys;
192
193 if (sys_test_bit(config->base + GD32_CLOCK_ID_OFFSET(id),
194 GD32_CLOCK_ID_BIT(id)) != 0) {
195 return CLOCK_CONTROL_STATUS_ON;
196 }
197
198 return CLOCK_CONTROL_STATUS_OFF;
199 }
200
201 static DEVICE_API(clock_control, clock_control_gd32_api) = {
202 .on = clock_control_gd32_on,
203 .off = clock_control_gd32_off,
204 .get_rate = clock_control_gd32_get_rate,
205 .get_status = clock_control_gd32_get_status,
206 };
207
208 static const struct clock_control_gd32_config config = {
209 .base = DT_REG_ADDR(DT_INST_PARENT(0)),
210 };
211
212 DEVICE_DT_INST_DEFINE(0, NULL, NULL, NULL, &config, PRE_KERNEL_1,
213 CONFIG_CLOCK_CONTROL_INIT_PRIORITY,
214 &clock_control_gd32_api);
215