请选择 进入手机版 | 继续访问电脑版
查看: 635|回复: 1

[原创] LPC55S69对接RT-Thread PWM设备框架

[复制链接]

该用户从未签到

6

主题

9

帖子

0

注册会员

Rank: 2

积分
79
最后登录
2024-2-28
发表于 2023-3-10 21:44:17 | 显示全部楼层 |阅读模式
LPC55S69对接RT-Thread PWM设备框架
在使用 RT-Thread 的 bsp pwm 的时候,注意到 lpc55sxx 系列只对接了通用定时器2中的通道1作为 PWM 输出。但其实 LPC55S69 具备非常多的 PWM 资源。于是根据 RT-Thread 设备驱动框架了,对接了其他通用定时器的 PWM 输出。
开始对接创建PWM设备
之前该 BSP 中的 drv_pwm 已经对接了基本的接口,但只考虑了一个通道作为 PWM 输出,在实际的使用中具有较大的局限性,不能充分发挥作用。我们首先基于PWM设备基类结构派生出新的 PWM 设备结构体:
struct lpc_pwm
{
    struct rt_device_pwm pwm_device;
    CTIMER_Type *    tim;
    uint32_t channel;
    char *name;
};
因为 PWM 设备不止一个 ,所以我们定义一个全局的 PWM 设备表来同时创建多个 PWM 设备:
static struct lpc_pwm lpc_pwm_obj[] =
{
#if defined(BSP_USING_CTIMER1_MAT0) || defined(BSP_USING_CTIMER1_MAT1) || \
        defined(BSP_USING_CTIMER1_MAT2)
    {
       .tim                     = CTIMER1,
       .name                    = "pwm1",
       .channel                 = RT_NULL
    },
#endif

#if defined(BSP_USING_CTIMER2_MAT0) || defined(BSP_USING_CTIMER2_MAT1) || \
        defined(BSP_USING_CTIMER2_MAT2)
    {
       .tim                     = CTIMER2,
       .name                    = "pwm2",
       .channel                 = RT_NULL
    },
#endif
};
/* 可以根据实际需求自行扩展 */
选择指定定时器的具体通道作为 PWM 输出:
static void pwm_get_channel(void)
{
#ifdef BSP_USING_CTIMER1_MAT0
    lpc_pwm_obj[PWM1_INDEX].channel |= 1 << 0;
#endif
#ifdef BSP_USING_CTIMER1_MAT1
    lpc_pwm_obj[PWM1_INDEX].channel |= 1 << 1;
#endif
#ifdef BSP_USING_CTIMER1_MAT2
    lpc_pwm_obj[PWM1_INDEX].channel |= 1 << 2;
#endif
#ifdef BSP_USING_CTIMER2_MAT0
    lpc_pwm_obj[PWM2_INDEX].channel |= 1 << 0;
#endif
#ifdef BSP_USING_CTIMER2_MAT1
    lpc_pwm_obj[PWM2_INDEX].channel |= 1 << 1;
#endif
#ifdef BSP_USING_CTIMER2_MAT2
    lpc_pwm_obj[PWM2_INDEX].channel |= 1 << 2;
#endif
}
/* 可以根据实际需求自行扩展 */实现PWM设备的操作方法
PWM 设备只有一个 control 方法, control 方法使用设备控制字 cmd 来区分操作,分别有ENABLE、DISABLE、SET、GET。这部分原先已经基本实现过了,只需添加新的定时器判断分支即可。
注册PWM设备
原先是使用 rt_hw_pwm_init() 即可注册定时器2通道1这一个定时器设备,但现在需要注册的是多个 PWM 设备,于是编写 lpc_pwm_init() 将 PWM 设备表中的多个 PWM 设备逐个进行注册。
static int lpc_pwm_init(void)
{
    int i = 0;
    int result = RT_EOK;

    pwm_get_channel();

    for (i = 0; i < sizeof(lpc_pwm_obj) / sizeof(lpc_pwm_obj[0]); i++)
    {
        /* pwm init */
        if (rt_hw_pwm_init(&lpc_pwm_obj) != RT_EOK)
        {
            LOG_E("%s init failed", lpc_pwm_obj.name);
            result = -RT_ERROR;
            goto __exit;
        }
        else
        {
            LOG_D("%s init success", lpc_pwm_obj.name);

            /* register pwm device */
            if (rt_device_pwm_register(&lpc_pwm_obj.pwm_device, lpc_pwm_obj.name, &lpc_drv_ops, lpc_pwm_obj.tim) == RT_EOK)
            {
                LOG_D("%s register success", lpc_pwm_obj.name);
            }
            else
            {
                LOG_E("%s register failed", lpc_pwm_obj.name);
                result = -RT_ERROR;
            }
        }
    }

__exit:
    return result;
}
INIT_DEVICE_EXPORT(lpc_pwm_init);
至此,对于该 BSP 的 drv_pwm 的重构基本完成,主要目的是更全面的对接 RT-Thread 设备框架,能够同时使用多路 PWM 输出,提高其适用性和灵活性。
后续优化
之后希望通过进一步研究功能强大的 SCTime/PWM ,并将其也添加到设备驱动框架中来。
欢迎大家交流~
回复

使用道具 举报

  • TA的每日心情
    开心
    2024-3-26 15:16
  • 签到天数: 266 天

    [LV.8]以坛为家I

    3298

    主题

    6545

    帖子

    0

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    31998
    最后登录
    2024-4-9
    发表于 2023-3-13 13:59:43 | 显示全部楼层
    支持一下
    签到签到
    回复

    使用道具 举报

    您需要登录后才可以回帖 注册/登录

    本版积分规则

    关闭

    站长推荐上一条 /4 下一条

    Archiver|手机版|小黑屋|恩智浦技术社区

    GMT+8, 2024-4-16 13:57 , Processed in 0.112443 second(s), 19 queries , MemCache On.

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表