1 /*
2 * Copyright 2023 NXP
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #include <zephyr/device.h>
8 #include <zephyr/drivers/pwm.h>
9 #include <zephyr/drivers/pinctrl.h>
10 #include <zephyr/drivers/clock_control.h>
11
12 #include <Emios_Mcl_Ip.h>
13 #include <Emios_Pwm_Ip.h>
14
15 #ifdef CONFIG_PWM_CAPTURE
16 #include <Emios_Icu_Ip.h>
17 #endif
18
19 #define LOG_MODULE_NAME nxp_s32_emios_pwm
20 #include <zephyr/logging/log.h>
21 LOG_MODULE_REGISTER(LOG_MODULE_NAME, CONFIG_PWM_LOG_LEVEL);
22
23 #define DT_DRV_COMPAT nxp_s32_emios_pwm
24
25 /*
26 * Need to fill to this array at runtime, cannot do at build time like
27 * the HAL over configuration tool due to limitation of the integration
28 */
29 #if EMIOS_PWM_IP_USED
30 extern uint8 eMios_Pwm_Ip_IndexInChState[EMIOS_PWM_IP_INSTANCE_COUNT][EMIOS_PWM_IP_CHANNEL_COUNT];
31 #endif
32
33 #ifdef CONFIG_PWM_CAPTURE
34 extern uint8 eMios_Icu_Ip_IndexInChState[EMIOS_ICU_IP_INSTANCE_COUNT][EMIOS_ICU_IP_NUM_OF_CHANNELS];
35
36 /* We need maximum three edges for measure both period and cycle */
37 #define MAX_NUM_EDGE 3
38
39 struct pwm_nxp_s32_capture_data {
40 bool continuous;
41 bool inverted;
42 bool pulse_capture;
43 bool period_capture;
44 void *user_data;
45 pwm_capture_callback_handler_t callback;
46 eMios_Icu_ValueType edge_buff[MAX_NUM_EDGE];
47 };
48 #endif
49
50 struct pwm_nxp_s32_data {
51 uint32_t emios_clk;
52 #if EMIOS_PWM_IP_USED
53 uint8_t start_pwm_ch;
54 #endif
55
56 #ifdef CONFIG_PWM_CAPTURE
57 struct pwm_nxp_s32_capture_data capture[EMIOS_ICU_IP_NUM_OF_CHANNELS];
58 #endif
59 };
60
61 #if EMIOS_PWM_IP_USED
62 struct pwm_nxp_s32_pulse_info {
63 uint8_t pwm_pulse_channels;
64 Emios_Pwm_Ip_ChannelConfigType *pwm_info;
65 };
66 #endif
67
68 struct pwm_nxp_s32_config {
69 eMIOS_Type *base;
70 uint8_t instance;
71 const struct device *clock_dev;
72 clock_control_subsys_t clock_subsys;
73 const struct pinctrl_dev_config *pincfg;
74
75 #if EMIOS_PWM_IP_USED
76 struct pwm_nxp_s32_pulse_info *pulse_info;
77 #endif
78
79 #ifdef CONFIG_PWM_CAPTURE
80 eMios_Icu_Ip_ConfigType * icu_cfg;
81 #endif
82 };
83
84 #if EMIOS_PWM_IP_USED
85 #ifdef EMIOS_PWM_IP_MODE_OPWFMB_USED
pwm_nxp_s32_set_cycles_internal_timebase(uint8_t instance,uint32_t channel,uint32_t period_cycles,uint32_t pulse_cycles)86 static int pwm_nxp_s32_set_cycles_internal_timebase(uint8_t instance, uint32_t channel,
87 uint32_t period_cycles, uint32_t pulse_cycles)
88 {
89 bool need_update = false;
90
91 if ((period_cycles > EMIOS_PWM_IP_MAX_CNT_VAL) ||
92 (period_cycles <= EMIOS_PWM_IP_MIN_CNT_VAL)) {
93 LOG_ERR("period_cycles is out of range");
94 return -EINVAL;
95 }
96
97 if (Emios_Pwm_Ip_GetPeriod(instance, channel) != period_cycles) {
98 Emios_Pwm_Ip_SetPeriod(instance, channel, period_cycles);
99 need_update = true;
100 }
101
102 if (Emios_Pwm_Ip_GetDutyCycle(instance, channel) != pulse_cycles) {
103 need_update = true;
104 if (Emios_Pwm_Ip_SetDutyCycle(instance, channel, pulse_cycles)) {
105 LOG_ERR("Cannot set pulse cycles");
106 return -EIO;
107 }
108 }
109
110 if (need_update) {
111 /* Force match so that the new period, duty cycle takes effect immediately */
112 Emios_Pwm_Ip_ForceMatchTrailingEdge(instance, channel, true);
113 }
114
115 return 0;
116 }
117 #endif
118
119 #if defined(EMIOS_PWM_IP_MODE_OPWMCB_USED) || defined(EMIOS_PWM_IP_MODE_OPWMB_USED)
pwm_nxp_s32_set_cycles_external_timebase(uint8_t instance,uint32_t channel,uint32_t period_cycles,uint32_t pulse_cycles)120 static int pwm_nxp_s32_set_cycles_external_timebase(uint8_t instance, uint32_t channel,
121 uint32_t period_cycles, uint32_t pulse_cycles)
122 {
123 uint8_t master_channel;
124
125 if ((period_cycles > EMIOS_PWM_IP_MAX_CNT_VAL) ||
126 (period_cycles <= EMIOS_PWM_IP_MIN_CNT_VAL)) {
127 LOG_ERR("period_cycles is out of range");
128 return -EINVAL;
129 }
130
131 if (Emios_Pwm_Ip_GetPeriod(instance, channel) != period_cycles) {
132 /*
133 * This mode uses internal counter, so change period and cycle
134 * don't effect to the others
135 */
136 master_channel = Emios_Pwm_Ip_GetMasterBusChannel(instance, channel);
137
138 if (Emios_Mcl_Ip_SetCounterBusPeriod(instance, master_channel, period_cycles)) {
139 LOG_ERR("Cannot set counter period");
140 return -EIO;
141 }
142 }
143
144 if (Emios_Pwm_Ip_GetDutyCycle(instance, channel) != pulse_cycles) {
145 if (Emios_Pwm_Ip_SetDutyCycle(instance, channel, pulse_cycles)) {
146 LOG_ERR("Cannot set pulse cycles");
147 return -EIO;
148 }
149 }
150
151 return 0;
152 }
153 #endif
154
pwm_nxp_s32_set_cycles(const struct device * dev,uint32_t channel,uint32_t period_cycles,uint32_t pulse_cycles,pwm_flags_t flags)155 static int pwm_nxp_s32_set_cycles(const struct device *dev, uint32_t channel,
156 uint32_t period_cycles, uint32_t pulse_cycles,
157 pwm_flags_t flags)
158 {
159 const struct pwm_nxp_s32_config *config = dev->config;
160 struct pwm_nxp_s32_data *data = dev->data;
161
162 Emios_Pwm_Ip_ChannelConfigType *pwm_info;
163 uint8_t logic_ch;
164
165 if (channel >= EMIOS_PWM_IP_CHANNEL_COUNT) {
166 LOG_ERR("invalid channel %d", channel);
167 return -EINVAL;
168 }
169
170 if (eMios_Pwm_Ip_IndexInChState[config->instance][channel] >=
171 EMIOS_PWM_IP_NUM_OF_CHANNELS_USED) {
172 LOG_ERR("Channel %d is not configured for PWM", channel);
173 return -EINVAL;
174 }
175
176 logic_ch = eMios_Pwm_Ip_IndexInChState[config->instance][channel] - data->start_pwm_ch;
177 pwm_info = &config->pulse_info->pwm_info[logic_ch];
178
179 if ((flags & PWM_POLARITY_MASK) == pwm_info->OutputPolarity) {
180 LOG_ERR("Only support configuring output polarity at boot time");
181 return -ENOTSUP;
182 }
183
184 switch (pwm_info->Mode) {
185 #ifdef EMIOS_PWM_IP_MODE_OPWFMB_USED
186 case EMIOS_PWM_IP_MODE_OPWFMB_FLAG:
187 return pwm_nxp_s32_set_cycles_internal_timebase(config->instance, channel,
188 period_cycles, pulse_cycles);
189 #endif
190
191 #ifdef EMIOS_PWM_IP_MODE_OPWMCB_USED
192 case EMIOS_PWM_IP_MODE_OPWMCB_TRAIL_EDGE_FLAG:
193 case EMIOS_PWM_IP_MODE_OPWMCB_LEAD_EDGE_FLAG:
194 if ((period_cycles % 2)) {
195 LOG_ERR("OPWMCB mode: period must be an even number");
196 return -EINVAL;
197 }
198
199 return pwm_nxp_s32_set_cycles_external_timebase(config->instance, channel,
200 (period_cycles + 2) / 2,
201 pulse_cycles);
202 #endif
203
204 #if defined(EMIOS_PWM_IP_MODE_OPWMB_USED)
205 case EMIOS_PWM_IP_MODE_OPWMB_FLAG:
206 if ((Emios_Pwm_Ip_GetPhaseShift(config->instance, channel) +
207 pulse_cycles) > period_cycles) {
208 LOG_ERR("OPWMB mode: new duty cycle + phase shift must <= new period");
209 return -EINVAL;
210 }
211
212 return pwm_nxp_s32_set_cycles_external_timebase(config->instance, channel,
213 period_cycles, pulse_cycles);
214 #endif
215
216 default:
217 /* Never reach here */
218 break;
219 }
220
221 return 0;
222 }
223 #endif
224
225 #ifdef CONFIG_PWM_CAPTURE
pwm_nxp_s32_capture_calc(eMios_Icu_ValueType first_cnt,eMios_Icu_ValueType second_cnt)226 static ALWAYS_INLINE eMios_Icu_ValueType pwm_nxp_s32_capture_calc(eMios_Icu_ValueType first_cnt,
227 eMios_Icu_ValueType second_cnt)
228 {
229 if (first_cnt < second_cnt) {
230 return second_cnt - first_cnt;
231 }
232
233 /* Counter top value is always 0xFFFF */
234 return EMIOS_ICU_IP_COUNTER_MASK - first_cnt + second_cnt;
235 }
236
pwm_nxp_s32_pulse_calc(bool inverted,eMios_Icu_ValueType * edge_buff,eMios_Icu_Ip_LevelType input_state)237 static ALWAYS_INLINE eMios_Icu_ValueType pwm_nxp_s32_pulse_calc(bool inverted,
238 eMios_Icu_ValueType *edge_buff,
239 eMios_Icu_Ip_LevelType input_state)
240 {
241 eMios_Icu_ValueType first_cnt, second_cnt;
242
243 if (input_state ^ inverted) {
244 /* 3 edges captured is raise, fall, raise */
245 first_cnt = edge_buff[0];
246 second_cnt = edge_buff[1];
247 } else {
248 /* 3 edges captured is fall, raise, fall */
249 first_cnt = edge_buff[1];
250 second_cnt = edge_buff[2];
251 }
252
253 return pwm_nxp_s32_capture_calc(first_cnt, second_cnt);
254 }
255
pwm_nxp_s32_capture_configure(const struct device * dev,uint32_t channel,pwm_flags_t flags,pwm_capture_callback_handler_t cb,void * user_data)256 static int pwm_nxp_s32_capture_configure(const struct device *dev,
257 uint32_t channel,
258 pwm_flags_t flags,
259 pwm_capture_callback_handler_t cb,
260 void *user_data)
261 {
262 const struct pwm_nxp_s32_config *config = dev->config;
263 struct pwm_nxp_s32_data *data = dev->data;
264
265 if (channel >= EMIOS_ICU_IP_NUM_OF_CHANNELS) {
266 LOG_ERR("Invalid channel %d", channel);
267 return -EINVAL;
268 }
269
270 if (!flags) {
271 LOG_ERR("Invalid PWM capture flag");
272 return -EINVAL;
273 }
274
275 if (eMios_Icu_Ip_IndexInChState[config->instance][channel] >=
276 EMIOS_ICU_IP_NUM_OF_CHANNELS_USED) {
277 LOG_ERR("Channel %d is not configured for PWM", channel);
278 return -EINVAL;
279 }
280
281 /* If interrupt is enabled --> channel is on-going */
282 if (config->base->CH.UC[channel].C & eMIOS_C_FEN_MASK) {
283 LOG_ERR("Channel %d is busy", channel);
284 return -EBUSY;
285 }
286
287 data->capture[channel].continuous = (flags & PWM_CAPTURE_MODE_MASK);
288 data->capture[channel].inverted = (flags & PWM_POLARITY_MASK);
289 data->capture[channel].pulse_capture = (flags & PWM_CAPTURE_TYPE_PULSE);
290 data->capture[channel].period_capture = (flags & PWM_CAPTURE_TYPE_PERIOD);
291 data->capture[channel].callback = cb;
292 data->capture[channel].user_data = user_data;
293
294 return 0;
295 }
296
pwm_nxp_s32_capture_enable(const struct device * dev,uint32_t channel)297 static int pwm_nxp_s32_capture_enable(const struct device *dev, uint32_t channel)
298 {
299 const struct pwm_nxp_s32_config *config = dev->config;
300 struct pwm_nxp_s32_data *data = dev->data;
301
302 eMios_Icu_Ip_EdgeType edge;
303 uint8_t num_edge;
304
305 if (channel >= EMIOS_ICU_IP_NUM_OF_CHANNELS) {
306 LOG_ERR("Invalid channel %d", channel);
307 return -EINVAL;
308 }
309
310 if (eMios_Icu_Ip_IndexInChState[config->instance][channel] >=
311 EMIOS_ICU_IP_NUM_OF_CHANNELS_USED) {
312 LOG_ERR("Channel %d is not configured for PWM", channel);
313 return -EINVAL;
314 }
315
316 if (!data->capture[channel].callback) {
317 LOG_ERR("Callback is not configured");
318 return -EINVAL;
319 }
320
321 /* If interrupt is enabled --> channel is on-going */
322 if (config->base->CH.UC[channel].C & eMIOS_C_FEN_MASK) {
323 LOG_ERR("Channel %d is busy", channel);
324 return -EBUSY;
325 }
326
327 /* If just measure period, we just need 2 edges */
328 if (data->capture[channel].period_capture && !data->capture[channel].pulse_capture) {
329 num_edge = 2U;
330 edge = EMIOS_ICU_RISING_EDGE;
331 } else {
332 num_edge = 3U;
333 edge = EMIOS_ICU_BOTH_EDGES;
334 }
335
336 Emios_Icu_Ip_SetActivation(config->instance, channel, edge);
337
338 Emios_Icu_Ip_EnableNotification(config->instance, channel);
339
340 Emios_Icu_Ip_StartTimestamp(config->instance, channel,
341 data->capture[channel].edge_buff,
342 MAX_NUM_EDGE, num_edge);
343
344 return 0;
345 }
346
pwm_nxp_s32_capture_disable(const struct device * dev,uint32_t channel)347 static int pwm_nxp_s32_capture_disable(const struct device *dev, uint32_t channel)
348 {
349 const struct pwm_nxp_s32_config *config = dev->config;
350
351 if (channel >= EMIOS_ICU_IP_NUM_OF_CHANNELS) {
352 LOG_ERR("Invalid channel %d", channel);
353 return -EINVAL;
354 }
355
356 if (eMios_Icu_Ip_IndexInChState[config->instance][channel] >=
357 EMIOS_ICU_IP_NUM_OF_CHANNELS_USED) {
358 LOG_ERR("Channel %d is not configured for PWM", channel);
359 return -EINVAL;
360 }
361
362 Emios_Icu_Ip_StopTimestamp(config->instance, channel);
363
364 return 0;
365 }
366
pwm_nxp_s32_get_master_bus(const struct device * dev,uint32_t channel)367 static int pwm_nxp_s32_get_master_bus(const struct device *dev, uint32_t channel)
368 {
369 const struct pwm_nxp_s32_config *config = dev->config;
370 uint8_t bus_select, master_bus;
371
372 bus_select = (config->base->CH.UC[channel].C & eMIOS_C_BSL_MASK) >> eMIOS_C_BSL_SHIFT;
373
374 switch (bus_select) {
375 case 0:
376 master_bus = 23U;
377 break;
378 case 1:
379 master_bus = (channel < 8U) ? 0U : ((channel < 16U) ? 8U : 16U);
380 break;
381 case 2:
382 master_bus = 22U;
383 break;
384 default:
385 /* Default is internal counter */
386 master_bus = channel;
387 break;
388 }
389
390 return master_bus;
391 }
392 #endif
393
pwm_nxp_s32_get_cycles_per_sec(const struct device * dev,uint32_t channel,uint64_t * cycles)394 static int pwm_nxp_s32_get_cycles_per_sec(const struct device *dev,
395 uint32_t channel,
396 uint64_t *cycles)
397 {
398 const struct pwm_nxp_s32_config *config = dev->config;
399 struct pwm_nxp_s32_data *data = dev->data;
400
401 uint8_t master_bus = 0xFFU;
402 uint8_t internal_prescaler, global_prescaler;
403
404 #if EMIOS_PWM_IP_USED
405 if (eMios_Pwm_Ip_IndexInChState[config->instance][channel] <
406 EMIOS_PWM_IP_NUM_OF_CHANNELS_USED) {
407 master_bus = Emios_Pwm_Ip_GetMasterBusChannel(config->instance, channel);
408 }
409 #endif
410
411 #ifdef CONFIG_PWM_CAPTURE
412 if (eMios_Icu_Ip_IndexInChState[config->instance][channel] <
413 EMIOS_ICU_IP_NUM_OF_CHANNELS_USED) {
414 master_bus = pwm_nxp_s32_get_master_bus(dev, channel);
415 }
416 #endif
417
418 if (master_bus == 0xFF) {
419 LOG_ERR("Channel %d is not configured for PWM", channel);
420 return -EINVAL;
421 }
422
423 internal_prescaler = (config->base->CH.UC[master_bus].C2 & eMIOS_C2_UCEXTPRE_MASK) >>
424 eMIOS_C2_UCEXTPRE_SHIFT;
425
426 /* Clock source for internal prescaler is from either eMIOS or eMIOS / global prescaler */
427 if (config->base->CH.UC[master_bus].C2 & eMIOS_C2_UCPRECLK_MASK) {
428 *cycles = data->emios_clk / (internal_prescaler + 1);
429 } else {
430 global_prescaler = (config->base->MCR & eMIOS_MCR_GPRE_MASK) >>
431 eMIOS_MCR_GPRE_SHIFT;
432 *cycles = data->emios_clk / ((internal_prescaler + 1) * (global_prescaler + 1));
433 }
434
435 return 0;
436 }
437
438 #if EMIOS_PWM_IP_USED
pwm_nxp_s32_pulse_gen_init(const struct device * dev)439 static int pwm_nxp_s32_pulse_gen_init(const struct device *dev)
440 {
441 const struct pwm_nxp_s32_config *config = dev->config;
442 struct pwm_nxp_s32_data *data = dev->data;
443
444 const Emios_Pwm_Ip_ChannelConfigType *pwm_info;
445
446 uint8_t ch_id;
447 static uint8_t logic_ch;
448
449 data->start_pwm_ch = logic_ch;
450
451 for (ch_id = 0; ch_id < config->pulse_info->pwm_pulse_channels; ch_id++) {
452 pwm_info = &config->pulse_info->pwm_info[ch_id];
453 eMios_Pwm_Ip_IndexInChState[config->instance][pwm_info->ChannelId] = logic_ch++;
454 Emios_Pwm_Ip_InitChannel(config->instance, pwm_info);
455 }
456
457 return 0;
458 }
459 #endif
460
461 #ifdef CONFIG_PWM_CAPTURE
pwm_nxp_s32_pulse_capture_init(const struct device * dev)462 static int pwm_nxp_s32_pulse_capture_init(const struct device *dev)
463 {
464 const struct pwm_nxp_s32_config *config = dev->config;
465
466 const eMios_Icu_Ip_ChannelConfigType *icu_info;
467
468 uint8_t ch_id;
469 static uint8_t logic_ch;
470
471 for (ch_id = 0; ch_id < config->icu_cfg->nNumChannels; ch_id++) {
472 icu_info = &(*config->icu_cfg->pChannelsConfig)[ch_id];
473 eMios_Icu_Ip_IndexInChState[config->instance][icu_info->hwChannel] = logic_ch++;
474 }
475
476 if (Emios_Icu_Ip_Init(config->instance, config->icu_cfg)) {
477 return -EINVAL;
478 }
479
480 return 0;
481 }
482
pwm_nxp_s32_capture_callback(const struct device * dev,uint32_t channel)483 static void pwm_nxp_s32_capture_callback(const struct device *dev, uint32_t channel)
484 {
485 const struct pwm_nxp_s32_config *config = dev->config;
486 struct pwm_nxp_s32_data *data = dev->data;
487
488 uint32_t period = 0, pulse = 0;
489
490 if (data->capture[channel].period_capture && !data->capture[channel].pulse_capture) {
491 period = pwm_nxp_s32_capture_calc(data->capture[channel].edge_buff[0],
492 data->capture[channel].edge_buff[1]);
493 } else {
494 if (data->capture[channel].pulse_capture) {
495 pulse = pwm_nxp_s32_pulse_calc(data->capture[channel].inverted,
496 data->capture[channel].edge_buff,
497 Emios_Icu_Ip_GetInputLevel(config->instance,
498 channel));
499 }
500
501 if (data->capture[channel].period_capture) {
502 period = pwm_nxp_s32_capture_calc(data->capture[channel].edge_buff[0],
503 data->capture[channel].edge_buff[2]);
504 }
505 }
506
507 if (!data->capture[channel].continuous) {
508 Emios_Icu_Ip_StopTimestamp(config->instance, channel);
509 }
510
511 data->capture[channel].callback(dev, channel, period, pulse, 0,
512 data->capture[channel].user_data);
513 }
514 #endif
515
pwm_nxp_s32_init(const struct device * dev)516 static int pwm_nxp_s32_init(const struct device *dev)
517 {
518 const struct pwm_nxp_s32_config *config = dev->config;
519 struct pwm_nxp_s32_data *data = dev->data;
520
521 int err = 0;
522
523 if (!device_is_ready(config->clock_dev)) {
524 return -ENODEV;
525 }
526
527 if (clock_control_get_rate(config->clock_dev, config->clock_subsys,
528 &data->emios_clk)) {
529 return -EINVAL;
530 }
531
532 err = pinctrl_apply_state(config->pincfg, PINCTRL_STATE_DEFAULT);
533 if (err < 0) {
534 return err;
535 }
536
537 #if EMIOS_PWM_IP_USED
538 err = pwm_nxp_s32_pulse_gen_init(dev);
539 if (err < 0) {
540 return err;
541 }
542 #endif
543
544 #ifdef CONFIG_PWM_CAPTURE
545 err = pwm_nxp_s32_pulse_capture_init(dev);
546 if (err < 0) {
547 return err;
548 }
549 #endif
550
551 return err;
552 }
553
554 static DEVICE_API(pwm, pwm_nxp_s32_driver_api) = {
555 .set_cycles = pwm_nxp_s32_set_cycles,
556 .get_cycles_per_sec = pwm_nxp_s32_get_cycles_per_sec,
557 #ifdef CONFIG_PWM_CAPTURE
558 .configure_capture = pwm_nxp_s32_capture_configure,
559 .enable_capture = pwm_nxp_s32_capture_enable,
560 .disable_capture = pwm_nxp_s32_capture_disable,
561 #endif
562 };
563
564 /*
565 * If timebase is configured in MCB up/down count mode: pwm period = (2 * master bus's period - 2)
566 */
567 #define EMIOS_PWM_PERIOD_TIME_BASE(node_id) \
568 COND_CODE_1(DT_ENUM_HAS_VALUE(node_id, mode, MCB_UP_DOWN_COUNTER), \
569 (2 * DT_PROP_BY_PHANDLE(node_id, master_bus, period) - 2), \
570 (DT_PROP_BY_PHANDLE(node_id, master_bus, period)))
571
572 #define EMIOS_PWM_IS_MODE_OPWFMB(node_id) \
573 DT_ENUM_HAS_VALUE(node_id, pwm_mode, OPWFMB)
574
575 #define EMIOS_PWM_IS_MODE_OPWMCB(node_id) \
576 UTIL_OR(DT_ENUM_HAS_VALUE(node_id, pwm_mode, OPWMCB_TRAIL_EDGE), \
577 DT_ENUM_HAS_VALUE(node_id, pwm_mode, OPWMCB_LEAD_EDGE)) \
578
579 #define EMIOS_PWM_IS_MODE_OPWMB(node_id) \
580 DT_ENUM_HAS_VALUE(node_id, pwm_mode, OPWMB)
581
582 #define EMIOS_PWM_IS_MODE_SAIC(node_id) \
583 DT_ENUM_HAS_VALUE(node_id, pwm_mode, SAIC)
584
585 #define EMIOS_PWM_IS_CAPTURE_MODE(node_id) \
586 EMIOS_PWM_IS_MODE_SAIC(node_id)
587
588 #define EMIOS_PWM_LOG(node_id, msg) \
589 DT_NODE_PATH(node_id) ": " DT_PROP(node_id, pwm_mode) ": " msg \
590
591 #define EMIOS_PWM_VERIFY_MASTER_BUS(node_id) \
592 BUILD_ASSERT(BIT(DT_PROP(node_id, channel)) & \
593 DT_PROP_BY_PHANDLE(node_id, master_bus, channel_mask), \
594 EMIOS_PWM_LOG(node_id, "invalid master bus"));
595
596 #define EMIOS_PWM_PULSE_GEN_COMMON_VERIFY(node_id) \
597 BUILD_ASSERT(DT_NODE_HAS_PROP(node_id, duty_cycle), \
598 EMIOS_PWM_LOG(node_id, "duty-cycle must be configured")); \
599 BUILD_ASSERT(DT_NODE_HAS_PROP(node_id, polarity), \
600 EMIOS_PWM_LOG(node_id, "polarity must be configured")); \
601 BUILD_ASSERT(DT_NODE_HAS_PROP(node_id, input_filter), \
602 EMIOS_PWM_LOG(node_id, "input-filter is not used"));
603
604 #define EMIOS_PWM_VERIFY_MODE_OPWFMB(node_id) \
605 EMIOS_PWM_PULSE_GEN_COMMON_VERIFY(node_id) \
606 BUILD_ASSERT(DT_NODE_HAS_PROP(node_id, period), \
607 EMIOS_PWM_LOG(node_id, "period must be configured")); \
608 BUILD_ASSERT(IN_RANGE(DT_PROP(node_id, period), EMIOS_PWM_IP_MIN_CNT_VAL + 1, \
609 EMIOS_PWM_IP_MAX_CNT_VAL), \
610 EMIOS_PWM_LOG(node_id, "period is out of range")); \
611 BUILD_ASSERT(DT_PROP(node_id, duty_cycle) <= DT_PROP(node_id, period), \
612 EMIOS_PWM_LOG(node_id, "duty-cycle must <= period")); \
613 BUILD_ASSERT(!DT_NODE_HAS_PROP(node_id, master_bus), \
614 EMIOS_PWM_LOG(node_id, "master-bus must not be configured")); \
615 BUILD_ASSERT(DT_PROP(node_id, dead_time) == 0, \
616 EMIOS_PWM_LOG(node_id, "dead-time is not used")); \
617 BUILD_ASSERT(DT_PROP(node_id, phase_shift) == 0, \
618 EMIOS_PWM_LOG(node_id, "phase-shift is not used"));
619
620 #define EMIOS_PWM_VERIFY_MODE_OPWMCB(node_id) \
621 EMIOS_PWM_PULSE_GEN_COMMON_VERIFY(node_id) \
622 BUILD_ASSERT(DT_ENUM_HAS_VALUE(DT_PHANDLE(node_id, master_bus), mode, \
623 MCB_UP_DOWN_COUNTER), \
624 EMIOS_PWM_LOG(node_id, "master-bus must be configured in MCB up-down")); \
625 BUILD_ASSERT((DT_PROP(node_id, duty_cycle) + DT_PROP(node_id, dead_time)) <= \
626 EMIOS_PWM_PERIOD_TIME_BASE(node_id), \
627 EMIOS_PWM_LOG(node_id, "duty-cycle + dead-time must <= period")); \
628 BUILD_ASSERT(DT_PROP(node_id, dead_time) <= DT_PROP(node_id, duty_cycle), \
629 EMIOS_PWM_LOG(node_id, "dead-time must <= duty-cycle")); \
630 BUILD_ASSERT(DT_PROP(node_id, phase_shift) == 0, \
631 EMIOS_PWM_LOG(node_id, "phase-shift is not used")); \
632 BUILD_ASSERT(!DT_NODE_HAS_PROP(node_id, period), \
633 EMIOS_PWM_LOG(node_id, "period is not used," \
634 " driver takes the value from master bus")); \
635 BUILD_ASSERT(!DT_NODE_HAS_PROP(node_id, prescaler), \
636 EMIOS_PWM_LOG(node_id, "prescaler is not used," \
637 " driver takes the value from master bus")); \
638 BUILD_ASSERT(DT_ENUM_HAS_VALUE(node_id, prescaler_src, PRESCALED_CLOCK), \
639 EMIOS_PWM_LOG(node_id, "prescaler-src is not used," \
640 " always use prescalered source")); \
641
642 #define EMIOS_PWM_VERIFY_MODE_OPWMB(node_id) \
643 EMIOS_PWM_PULSE_GEN_COMMON_VERIFY(node_id) \
644 BUILD_ASSERT(DT_ENUM_HAS_VALUE(DT_PHANDLE(node_id, master_bus), mode, MCB_UP_COUNTER), \
645 EMIOS_PWM_LOG(node_id, "master-bus must be configured in MCB up")); \
646 BUILD_ASSERT(!DT_NODE_HAS_PROP(node_id, period), \
647 EMIOS_PWM_LOG(node_id, "period is not used," \
648 " driver takes the value from master bus")); \
649 BUILD_ASSERT((DT_PROP(node_id, duty_cycle) + DT_PROP(node_id, phase_shift)) <= \
650 EMIOS_PWM_PERIOD_TIME_BASE(node_id), \
651 EMIOS_PWM_LOG(node_id, "duty-cycle + phase-shift must <= period")); \
652 BUILD_ASSERT(DT_PROP(node_id, dead_time) == 0, \
653 EMIOS_PWM_LOG(node_id, "dead-time is not used")); \
654 BUILD_ASSERT(!DT_NODE_HAS_PROP(node_id, prescaler), \
655 EMIOS_PWM_LOG(node_id, "prescaler is not used")); \
656 BUILD_ASSERT(DT_ENUM_HAS_VALUE(node_id, prescaler_src, PRESCALED_CLOCK), \
657 EMIOS_PWM_LOG(node_id, "prescaler-src is not used," \
658 " always use prescalered source")); \
659
660 #define EMIOS_PWM_VERIFY_MODE_SAIC(node_id) \
661 IF_ENABLED(DT_NODE_HAS_PROP(node_id, master_bus), \
662 (BUILD_ASSERT( \
663 DT_ENUM_HAS_VALUE(DT_PHANDLE(node_id, master_bus), mode, MCB_UP_COUNTER), \
664 EMIOS_PWM_LOG(node_id, "master-bus must be configured in MCB up"));)) \
665 IF_ENABLED(DT_NODE_HAS_PROP(node_id, master_bus), \
666 (BUILD_ASSERT(DT_PROP_BY_PHANDLE(node_id, master_bus, period) == 0xFFFF, \
667 EMIOS_PWM_LOG(node_id, "master-bus period must be 0xFFFF"));)) \
668 IF_ENABLED(UTIL_NOT(DT_NODE_HAS_PROP(node_id, master_bus)), \
669 (BUILD_ASSERT( \
670 BIT(DT_PROP(node_id, channel)) & DT_PROP(DT_GPARENT(node_id), internal_cnt),\
671 EMIOS_PWM_LOG(node_id, "master-bus must be chosen," \
672 " channel do not have has counter")))); \
673 IF_ENABLED(UTIL_NOT(DT_NODE_HAS_PROP(node_id, master_bus)), \
674 (BUILD_ASSERT(DT_NODE_HAS_PROP(node_id, prescaler), \
675 EMIOS_PWM_LOG(node_id, "if use internal counter, prescaler must" \
676 " be configured")))); \
677 BUILD_ASSERT(!DT_NODE_HAS_PROP(node_id, duty_cycle), \
678 EMIOS_PWM_LOG(node_id, "duty-cycle is not used")); \
679 BUILD_ASSERT(!DT_NODE_HAS_PROP(node_id, polarity), \
680 EMIOS_PWM_LOG(node_id, "polarity is not used")); \
681 BUILD_ASSERT(!DT_NODE_HAS_PROP(node_id, period), \
682 EMIOS_PWM_LOG(node_id, "period is not used")); \
683 BUILD_ASSERT(DT_ENUM_HAS_VALUE(node_id, prescaler_src, PRESCALED_CLOCK), \
684 EMIOS_PWM_LOG(node_id, "prescaler-src is not used," \
685 " always use prescalered source"));
686
687 #define _EMIOS_PWM_VERIFY_CONFIG(node_id) \
688 IF_ENABLED(DT_NODE_HAS_PROP(node_id, master_bus), \
689 (EMIOS_PWM_VERIFY_MASTER_BUS(node_id))) \
690 IF_ENABLED(EMIOS_PWM_IS_MODE_OPWFMB(node_id), \
691 (EMIOS_PWM_VERIFY_MODE_OPWFMB(node_id))) \
692 IF_ENABLED(EMIOS_PWM_IS_MODE_OPWMCB(node_id), \
693 (EMIOS_PWM_VERIFY_MODE_OPWMCB(node_id))) \
694 IF_ENABLED(EMIOS_PWM_IS_MODE_OPWMB(node_id), \
695 (EMIOS_PWM_VERIFY_MODE_OPWMB(node_id))) \
696 IF_ENABLED(EMIOS_PWM_IS_MODE_SAIC(node_id), \
697 (EMIOS_PWM_VERIFY_MODE_SAIC(node_id)))
698
699 #if EMIOS_PWM_IP_USED
700 /* Macros used to glue devicetree with RTD's definition */
701 #define EMIOS_PWM_BUS_A EMIOS_PWM_IP_BUS_A
702 #define EMIOS_PWM_BUS_B EMIOS_PWM_IP_BUS_BCDE
703 #define EMIOS_PWM_BUS_C EMIOS_PWM_IP_BUS_BCDE
704 #define EMIOS_PWM_BUS_D EMIOS_PWM_IP_BUS_BCDE
705 #define EMIOS_PWM_BUS_E EMIOS_PWM_IP_BUS_BCDE
706 #define EMIOS_PWM_BUS_F EMIOS_PWM_IP_BUS_F
707
708 #define EMIOS_PWM_BUS(mode) DT_CAT(EMIOS_PWM_, mode)
709 #define EMIOS_PWM_MODE(mode) DT_CAT3(EMIOS_PWM_IP_MODE_, mode, _FLAG)
710 #define EMIOS_PWM_POLARITY(mode) DT_CAT(EMIOS_PWM_IP_, mode)
711 #define EMIOS_PWM_PS_SRC(mode) DT_CAT(EMIOS_PWM_IP_PS_SRC_, mode)
712
713 #define _EMIOS_PWM_PULSE_GEN_CONFIG(node_id) \
714 IF_ENABLED(UTIL_NOT(EMIOS_PWM_IS_CAPTURE_MODE(node_id)), \
715 ({ \
716 .ChannelId = DT_PROP(node_id, channel), \
717 .Mode = EMIOS_PWM_MODE(DT_STRING_TOKEN(node_id, pwm_mode)), \
718 .InternalPsSrc = EMIOS_PWM_PS_SRC(DT_STRING_TOKEN(node_id, prescaler_src)), \
719 .InternalPs = DT_PROP_OR(node_id, prescaler, \
720 DT_PROP_BY_PHANDLE(node_id, master_bus, prescaler)) - 1,\
721 .Timebase = COND_CODE_1(DT_NODE_HAS_PROP(node_id, master_bus), \
722 (EMIOS_PWM_BUS(DT_STRING_TOKEN( \
723 DT_PHANDLE(node_id, master_bus), bus_type))), \
724 (EMIOS_PWM_IP_BUS_INTERNAL)), \
725 .PhaseShift = DT_PROP(node_id, phase_shift), \
726 .DeadTime = DT_PROP(node_id, dead_time), \
727 .OutputDisableSource = EMIOS_PWM_IP_OUTPUT_DISABLE_NONE, \
728 .OutputPolarity = EMIOS_PWM_POLARITY(DT_STRING_TOKEN(node_id, polarity)), \
729 .DebugMode = DT_PROP(node_id, freeze), \
730 .PeriodCount = DT_PROP_OR(node_id, period, EMIOS_PWM_PERIOD_TIME_BASE(node_id)),\
731 .DutyCycle = DT_PROP(node_id, duty_cycle), \
732 },))
733
734 #define EMIOS_PWM_PULSE_GEN_CONFIG(n) \
735 const Emios_Pwm_Ip_ChannelConfigType emios_pwm_##n##_init[] = { \
736 DT_INST_FOREACH_CHILD_STATUS_OKAY(n, _EMIOS_PWM_PULSE_GEN_CONFIG) \
737 }; \
738 const struct pwm_nxp_s32_pulse_info emios_pwm_##n##_info = { \
739 .pwm_pulse_channels = ARRAY_SIZE(emios_pwm_##n##_init), \
740 .pwm_info = (Emios_Pwm_Ip_ChannelConfigType *)emios_pwm_##n##_init, \
741 };
742
743 #define EMIOS_PWM_PULSE_GEN_GET_CONFIG(n) \
744 .pulse_info = (struct pwm_nxp_s32_pulse_info *)&emios_pwm_##n##_info,
745 #else
746 #define EMIOS_PWM_PULSE_GEN_CONFIG(n)
747 #define EMIOS_PWM_PULSE_GEN_GET_CONFIG(n)
748 #endif
749
750 #ifdef CONFIG_PWM_CAPTURE
751 /* Macros used to glue devicetree with RTD's definition */
752 #define EMIOS_ICU_BUS_A EMIOS_ICU_BUS_A
753 #define EMIOS_ICU_BUS_B EMIOS_ICU_BUS_DIVERSE
754 #define EMIOS_ICU_BUS_C EMIOS_ICU_BUS_DIVERSE
755 #define EMIOS_ICU_BUS_D EMIOS_ICU_BUS_DIVERSE
756 #define EMIOS_ICU_BUS_E EMIOS_ICU_BUS_DIVERSE
757 #define EMIOS_ICU_BUS_F EMIOS_ICU_BUS_F
758
759 #define DIGITAL_FILTER_0 EMIOS_DIGITAL_FILTER_BYPASSED
760 #define DIGITAL_FILTER_2 EMIOS_DIGITAL_FILTER_02
761 #define DIGITAL_FILTER_4 EMIOS_DIGITAL_FILTER_04
762 #define DIGITAL_FILTER_8 EMIOS_DIGITAL_FILTER_08
763 #define DIGITAL_FILTER_16 EMIOS_DIGITAL_FILTER_16
764
765 #define EMIOS_PWM_CAPTURE_FILTER(filter) DT_CAT(DIGITAL_FILTER_, filter)
766 #define EMIOS_PWM_CAPTURE_MODE(mode) DT_CAT(EMIOS_ICU_, mode)
767 #define EMIOS_PWM_CAPTURE_BUS(mode) DT_CAT(EMIOS_ICU_, mode)
768
769 #define EMIOS_PWM_CAPTURE_CB(n, ch) \
770 DT_CAT5(pwm_nxp_s32_, n, _channel_, ch, _capture_callback)
771
772 #define EMIOS_PWM_CALLBACK_DECLARE(node_id, n) \
773 void EMIOS_PWM_CAPTURE_CB(n, DT_PROP(node_id, channel))(void) \
774 { \
775 pwm_nxp_s32_capture_callback(DEVICE_DT_INST_GET(n), DT_PROP(node_id, channel)); \
776 } \
777
778 #define _EMIOS_PWM_PULSE_CAPTURE_CONFIG(node_id, n) \
779 IF_ENABLED(EMIOS_PWM_IS_CAPTURE_MODE(node_id), \
780 ({ \
781 .hwChannel = DT_PROP(node_id, channel), \
782 .ucMode = EMIOS_PWM_CAPTURE_MODE(DT_STRING_TOKEN(node_id, pwm_mode)), \
783 .FreezeEn = DT_PROP(node_id, freeze), \
784 .Prescaler = COND_CODE_1(DT_NODE_HAS_PROP(node_id, master_bus), \
785 (DT_PROP_BY_PHANDLE(node_id, master_bus, prescaler)), \
786 (DT_PROP(node_id, prescaler))) - 1, \
787 .CntBus = COND_CODE_1(DT_NODE_HAS_PROP(node_id, master_bus), \
788 (EMIOS_PWM_CAPTURE_BUS(DT_STRING_TOKEN( \
789 DT_PHANDLE(node_id, master_bus), bus_type))),\
790 (EMIOS_ICU_BUS_INTERNAL_COUNTER)), \
791 .chMode = EMIOS_ICU_MODE_TIMESTAMP, \
792 .chSubMode = EMIOS_ICU_MODE_WITHOUT_DMA, \
793 .measurementMode = EMIOS_ICU_NO_MEASUREMENT, \
794 .edgeAlignement = EMIOS_ICU_BOTH_EDGES, \
795 .Filter = EMIOS_PWM_CAPTURE_FILTER(DT_PROP(node_id, input_filter)), \
796 .callback = NULL_PTR, \
797 .logicChStateCallback = NULL_PTR, \
798 .callbackParams = 255U, \
799 .bWithoutInterrupt = FALSE, \
800 .timestampBufferType = EMIOS_ICU_CIRCULAR_BUFFER, \
801 .eMiosChannelNotification = EMIOS_PWM_CAPTURE_CB(n, DT_PROP(node_id, channel)), \
802 .eMiosOverflowNotification = NULL_PTR, \
803 },))
804
805 #define EMIOS_PWM_PULSE_CAPTURE_CONFIG(n) \
806 DT_INST_FOREACH_CHILD_STATUS_OKAY_VARGS(n, EMIOS_PWM_CALLBACK_DECLARE, n) \
807 const eMios_Icu_Ip_ChannelConfigType emios_pwm_##n##_capture_init[] = { \
808 DT_INST_FOREACH_CHILD_STATUS_OKAY_VARGS(n, _EMIOS_PWM_PULSE_CAPTURE_CONFIG, n) \
809 }; \
810 const eMios_Icu_Ip_ConfigType emios_pwm_##n##_capture_info = { \
811 .nNumChannels = ARRAY_SIZE(emios_pwm_##n##_capture_init), \
812 .pChannelsConfig = &emios_pwm_##n##_capture_init, \
813 };
814
815 #define EMIOS_PWM_PULSE_CAPTURE_GET_CONFIG(n) \
816 .icu_cfg = (eMios_Icu_Ip_ConfigType *)&emios_pwm_##n##_capture_info,
817 #else
818 #define EMIOS_PWM_PULSE_CAPTURE_CONFIG(n)
819 #define EMIOS_PWM_PULSE_CAPTURE_GET_CONFIG(n)
820 #endif
821
822 #define EMIOS_PWM_VERIFY_CONFIG(n) \
823 DT_INST_FOREACH_CHILD_STATUS_OKAY(n, _EMIOS_PWM_VERIFY_CONFIG)
824
825 #define EMIOS_NXP_S32_INSTANCE_CHECK(idx, node_id) \
826 ((DT_REG_ADDR(node_id) == IP_EMIOS_##idx##_BASE) ? idx : 0)
827
828 #define EMIOS_NXP_S32_GET_INSTANCE(node_id) \
829 LISTIFY(__DEBRACKET eMIOS_INSTANCE_COUNT, EMIOS_NXP_S32_INSTANCE_CHECK, (|), node_id)
830
831 #define PWM_NXP_S32_INIT_DEVICE(n) \
832 PINCTRL_DT_INST_DEFINE(n); \
833 EMIOS_PWM_VERIFY_CONFIG(n) \
834 EMIOS_PWM_PULSE_GEN_CONFIG(n) \
835 EMIOS_PWM_PULSE_CAPTURE_CONFIG(n) \
836 static const struct pwm_nxp_s32_config pwm_nxp_s32_config_##n = { \
837 .base = (eMIOS_Type *)DT_REG_ADDR(DT_INST_PARENT(n)), \
838 .instance = EMIOS_NXP_S32_GET_INSTANCE(DT_INST_PARENT(n)), \
839 .clock_dev = DEVICE_DT_GET(DT_CLOCKS_CTLR(DT_INST_PARENT(n))), \
840 .clock_subsys = (clock_control_subsys_t)DT_CLOCKS_CELL(DT_INST_PARENT(n), name),\
841 .pincfg = PINCTRL_DT_INST_DEV_CONFIG_GET(n), \
842 EMIOS_PWM_PULSE_GEN_GET_CONFIG(n) \
843 EMIOS_PWM_PULSE_CAPTURE_GET_CONFIG(n) \
844 }; \
845 static struct pwm_nxp_s32_data pwm_nxp_s32_data_##n; \
846 DEVICE_DT_INST_DEFINE(n, \
847 &pwm_nxp_s32_init, \
848 NULL, \
849 &pwm_nxp_s32_data_##n, \
850 &pwm_nxp_s32_config_##n, \
851 POST_KERNEL, \
852 CONFIG_PWM_INIT_PRIORITY, \
853 &pwm_nxp_s32_driver_api);
854
855 DT_INST_FOREACH_STATUS_OKAY(PWM_NXP_S32_INIT_DEVICE)
856