13
13
#include <zephyr/drivers/can.h>
14
14
#include <zephyr/drivers/can/can_mcan.h>
15
15
#include <zephyr/drivers/clock_control.h>
16
+ #include <zephyr/drivers/clock_control/nrf_clock_control.h>
16
17
#include <zephyr/drivers/pinctrl.h>
17
18
#include <zephyr/irq.h>
18
19
@@ -27,7 +28,8 @@ struct can_nrf_config {
27
28
uint32_t mcan ;
28
29
uint32_t mrba ;
29
30
uint32_t mram ;
30
- const struct device * clock ;
31
+ const struct device * auxpll ;
32
+ const struct device * hsfll ;
31
33
const struct pinctrl_dev_config * pcfg ;
32
34
void (* irq_configure )(void );
33
35
uint16_t irq ;
@@ -54,7 +56,7 @@ static int can_nrf_get_core_clock(const struct device *dev, uint32_t *rate)
54
56
const struct can_mcan_config * mcan_config = dev -> config ;
55
57
const struct can_nrf_config * config = mcan_config -> custom ;
56
58
57
- return clock_control_get_rate (config -> clock , NULL , rate );
59
+ return clock_control_get_rate (config -> auxpll , NULL , rate );
58
60
}
59
61
60
62
static DEVICE_API (can , can_nrf_api ) = {
@@ -131,17 +133,41 @@ static const struct can_mcan_ops can_mcan_nrf_ops = {
131
133
.clear_mram = can_nrf_clear_mram ,
132
134
};
133
135
136
+ static int configure_hsfll (const struct device * dev , bool on )
137
+ {
138
+ const struct can_mcan_config * mcan_config = dev -> config ;
139
+ const struct can_nrf_config * config = mcan_config -> custom ;
140
+ struct nrf_clock_spec spec = { 0 };
141
+
142
+ /* If CAN is on, HSFLL frequency >= AUXPLL frequency */
143
+ if (on ) {
144
+ int ret ;
145
+
146
+ ret = clock_control_get_rate (dev , NULL , & spec .frequency );
147
+ if (ret < 0 ) {
148
+ return ret ;
149
+ }
150
+ }
151
+
152
+ return nrf_clock_control_request_sync (config -> hsfll , & spec , K_FOREVER );
153
+ }
154
+
134
155
static int can_nrf_init (const struct device * dev )
135
156
{
136
157
const struct can_mcan_config * mcan_config = dev -> config ;
137
158
const struct can_nrf_config * config = mcan_config -> custom ;
138
159
int ret ;
139
160
140
- if (!device_is_ready (config -> clock )) {
161
+ if (!device_is_ready (config -> auxpll ) || ! device_is_ready ( config -> hsfll )) {
141
162
return - ENODEV ;
142
163
}
143
164
144
- ret = clock_control_on (config -> clock , NULL );
165
+ ret = configure_hsfll (dev , true);
166
+ if (ret < 0 ) {
167
+ return ret ;
168
+ }
169
+
170
+ ret = clock_control_on (config -> auxpll , NULL );
145
171
if (ret < 0 ) {
146
172
return ret ;
147
173
}
@@ -186,8 +212,9 @@ static int can_nrf_init(const struct device *dev)
186
212
.mcan = CAN_MCAN_DT_INST_MCAN_ADDR(n), \
187
213
.mrba = CAN_MCAN_DT_INST_MRBA(n), \
188
214
.mram = CAN_MCAN_DT_INST_MRAM_ADDR(n), \
189
- .clock = DEVICE_DT_GET(DT_INST_CLOCKS_CTLR(n )), \
215
+ .auxpll = DEVICE_DT_GET(DT_INST_CLOCKS_CTLR_BY_NAME(n, auxpll )), \
190
216
.pcfg = PINCTRL_DT_INST_DEV_CONFIG_GET(n), \
217
+ .hsfll = DEVICE_DT_GET(DT_INST_CLOCKS_CTLR_BY_NAME(n, hsfll)), \
191
218
.irq = DT_INST_IRQN(n), \
192
219
.irq_configure = can_nrf_irq_configure##n, \
193
220
}; \
0 commit comments