123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229 |
- /*
- * Copyright (c) 2006-2023, RT-Thread Development Team
- *
- * SPDX-License-Identifier: Apache-2.0
- *
- * Change Logs:
- * Date Author Notes
- * 2023-10-20 ChuShicheng first version
- */
- #include "drv_pwm.h"
- #include "board.h"
- #include "hardware/pwm.h"
- #ifdef BSP_USING_PWM
- #define DBG_LEVEL DBG_LOG
- #include <rtdbg.h>
- #define LOG_TAG "DRV.PWM"
- struct pico_pwm
- {
- struct rt_device_pwm pwm_device;
- rt_uint8_t slice_num;
- rt_uint8_t pin_a;
- rt_uint8_t pin_b;
- rt_uint8_t a_all_flag; /* 1: Two pins simultaneously output PWM A channel */
- rt_uint8_t b_all_flag; /* 1: Two pins simultaneously output PWM B channel */
- char *name;
- };
- static struct pico_pwm pico_pwm_obj[] =
- {
- #ifdef BSP_USING_PWM0
- {
- .name = "pwm0",
- .slice_num = 0,
- .pin_a = BSP_PWM0_A_PIN,
- .pin_b = BSP_PWM0_B_PIN,
- #ifdef BSP_PWM0_A_ALL
- .a_all_flag = 1,
- #endif /* BSP_PWM0_A_ALL */
- #ifdef BSP_PWM0_B_ALL
- .b_all_flag = 1,
- #endif /* BSP_PWM0_B_ALL */
- },
- #endif /* BSP_USING_PWM0 */
- #ifdef BSP_USING_PWM1
- {
- .name = "pwm1",
- .slice_num = 1,
- .pin_a = BSP_PWM1_A_PIN,
- .pin_b = BSP_PWM1_B_PIN,
- #ifdef BSP_PWM1_A_ALL
- .a_all_flag = 1,
- #endif /* BSP_PWM1_A_ALL */
- #ifdef BSP_PWM1_B_ALL
- .b_all_flag = 1,
- #endif /* BSP_PWM1_B_ALL */
- },
- #endif /* BSP_USING_PWM1 */
- #ifdef BSP_USING_PWM2
- {
- .name = "pwm2",
- .slice_num = 2,
- .pin_a = BSP_PWM2_A_PIN,
- .pin_b = BSP_PWM2_B_PIN,
- #ifdef BSP_PWM2_A_ALL
- .a_all_flag = 1,
- #endif /* BSP_PWM2_A_ALL */
- #ifdef BSP_PWM2_B_ALL
- .b_all_flag = 1,
- #endif /* BSP_PWM2_B_ALL */
- },
- #endif /* BSP_USING_PWM2 */
- #ifdef BSP_USING_PWM3
- {
- .name = "pwm3",
- .slice_num = 3,
- .pin_a = BSP_PWM3_A_PIN,
- .pin_b = BSP_PWM3_B_PIN,
- #ifdef BSP_PWM3_A_ALL
- .a_all_flag = 1,
- #endif /* BSP_PWM3_A_ALL */
- #ifdef BSP_PWM3_B_ALL
- .b_all_flag = 1,
- #endif /* BSP_PWM3_B_ALL */
- },
- #endif /* BSP_USING_PWM3 */
- #ifdef BSP_USING_PWM4
- {
- .name = "pwm4",
- .slice_num = 4,
- .pin_a = BSP_PWM4_A_PIN,
- .pin_b = BSP_PWM4_B_PIN,
- #ifdef BSP_PWM4_A_ALL
- .a_all_flag = 1,
- #endif /* BSP_PWM4_A_ALL */
- #ifdef BSP_PWM4_B_ALL
- .b_all_flag = 1,
- #endif /* BSP_PWM4_B_ALL */
- },
- #endif /* BSP_USING_PWM4 */
- #ifdef BSP_USING_PWM5
- {
- .name = "pwm5",
- .slice_num = 5,
- .pin_a = BSP_PWM5_A_PIN,
- .pin_b = BSP_PWM5_B_PIN,
- #ifdef BSP_PWM5_A_ALL
- .a_all_flag = 1,
- #endif /* BSP_PWM5_A_ALL */
- #ifdef BSP_PWM5_B_ALL
- .b_all_flag = 1,
- #endif /* BSP_PWM5_B_ALL */
- },
- #endif /* BSP_USING_PWM5 */
- #ifdef BSP_USING_PWM6
- {
- .name = "pwm6",
- .slice_num = 6,
- .pin_a = BSP_PWM6_A_PIN,
- .pin_b = BSP_PWM6_B_PIN,
- #ifdef BSP_PWM6_A_ALL
- .a_all_flag = 1,
- #endif /* BSP_PWM6_A_ALL */
- #ifdef BSP_PWM6_B_ALL
- .b_all_flag = 1,
- #endif /* BSP_PWM6_B_ALL */
- },
- #endif /* BSP_USING_PWM6 */
- #ifdef BSP_USING_PWM7
- {
- .name = "pwm7",
- .slice_num = 7,
- .pin_a = BSP_PWM7_A_PIN,
- .pin_b = BSP_PWM7_B_PIN,
- },
- #endif /* BSP_USING_PWM7 */
- };
- static rt_err_t _pwm_set(rt_uint8_t slice_num, struct rt_pwm_configuration *configuration)
- {
- uint32_t period_hz = 1000000000 / configuration->period;
- uint32_t pulse = 1000000000 / configuration->pulse;
- pwm_config config = pwm_get_default_config();
- pwm_config_set_clkdiv(&config, 1.33f);
- pwm_init(slice_num, &config, true);
- pwm_set_wrap(slice_num, 100000000 / period_hz);
- pwm_set_chan_level(slice_num, configuration->channel, 100000000 / pulse);
- return RT_EOK;
- }
- static rt_err_t _pwm_get(rt_uint8_t slice_num, struct rt_pwm_configuration *configuration)
- {
- uint32_t period_hz = pwm_hw->slice[slice_num].top;
- configuration->period = period_hz * 10;
- configuration->pulse = pwm_hw->slice[slice_num].cc * 10;
- return RT_EOK;
- }
- static rt_err_t _pwm_control(struct rt_device_pwm *device, int cmd, void *arg)
- {
- struct pico_pwm *_pwm = rt_container_of(device, struct pico_pwm, pwm_device);
- struct rt_pwm_configuration *configuration = (struct rt_pwm_configuration *)arg;
- switch (cmd)
- {
- case PWM_CMD_ENABLE:
- pwm_set_enabled (_pwm->slice_num, RT_TRUE);
- return RT_EOK;
- case PWM_CMD_DISABLE:
- pwm_set_enabled (_pwm->slice_num, RT_FALSE);
- return RT_EOK;
- case PWM_CMD_SET:
- return _pwm_set(_pwm->slice_num, configuration);
- case PWM_CMD_GET:
- return _pwm_get(_pwm->slice_num, configuration);
- default:
- return -RT_EINVAL;
- }
- }
- static struct rt_pwm_ops _pwm_ops =
- {
- _pwm_control
- };
- int rt_hw_pwm_init(void)
- {
- int result = RT_EOK;
- for (int i = 0; i < sizeof(pico_pwm_obj) / sizeof(pico_pwm_obj[0]); i++)
- {
- if(pico_pwm_obj[i].a_all_flag)
- {
- gpio_set_function(pico_pwm_obj[i].slice_num * 2, GPIO_FUNC_PWM);
- gpio_set_function(pico_pwm_obj[i].slice_num * 2 + 16, GPIO_FUNC_PWM);
- }
- else
- {
- gpio_set_function(pico_pwm_obj[i].pin_a, GPIO_FUNC_PWM);
- }
- if(pico_pwm_obj[i].b_all_flag)
- {
- gpio_set_function(pico_pwm_obj[i].slice_num * 2 + 1, GPIO_FUNC_PWM);
- gpio_set_function(pico_pwm_obj[i].slice_num * 2 + 17, GPIO_FUNC_PWM);
- }
- else
- {
- gpio_set_function(pico_pwm_obj[i].pin_b, GPIO_FUNC_PWM);
- }
- result = rt_device_pwm_register(&pico_pwm_obj[i].pwm_device, pico_pwm_obj[i].name, &_pwm_ops, 0);
- if(result != RT_EOK)
- {
- LOG_E("%s register fail.", pico_pwm_obj[i].name);
- }
- }
- return result;
- }
- INIT_DEVICE_EXPORT(rt_hw_pwm_init);
- #endif /* BSP_USING_PWM */
|