解决rt_pin_get返回错误码的问题

发布于:2025-09-06 ⋅ 阅读:(19) ⋅ 点赞:(0)

项目场景:

用RT-Thread Studio创建新项目,项目参数如下图所示

      


问题描述

在main函数中写了简单的用按钮开关LED灯的功能,代码如下图所示,编译和烧写程序后发现不能实现预期的功能,而如果改为GET_PIN的 方式后可以实现预期的功能。

#define LED1 GET_PIN(A, 0)
void turnLed1(void* args);
int main(void)
{
    rt_kprintf("ex03-key-led Hello RT-Thread!\n");
    rt_pin_mode(LED1, PIN_MODE_OUTPUT);
    rt_pin_write(LED1, PIN_HIGH);

    // key2 会是系统没有实现的错误号
    rt_base_t key2 = rt_pin_get("PF.10");
    rt_kprintf("key2 = %d\n", key2);

    // rt_base_t key3 = GET_PIN(F, 10);
    rt_base_t key3 = rt_pin_get("PF.10");
    rt_kprintf("key3 = %d\n", key3);

    rt_pin_mode(key3, PIN_MODE_INPUT_PULLDOWN);
    rt_err_t result = rt_pin_attach_irq(key3, PIN_IRQ_MODE_RISING, turnLed1, RT_NULL);
    if(result != RT_EOK)
    {
        rt_kprintf("rt_pin_attach_irq failed,err=%d\n", result);
    }
    if(rt_pin_irq_enable(key3, PIN_IRQ_ENABLE) != RT_EOK)
    {
        rt_kprintf("rt_pin_irq_enable failed");
    }
    while (1)
    {

        rt_thread_mdelay(2000);
    }

    return RT_EOK;
}

原因分析:

在网上搜索,发现有一些关于这个问题的主题,基本都是建议用GET_PIN来获取,这篇文章主要是从RT-Thread的设备驱动层来分析产生这个问题的原因,并在驱动层解决rt_pin_get存在的问题。下面我们从源码逐步来分析这个问题的原因,本节追朔了较多源码,可以跳过本节而直接使用下一节的解决方案。rt_pin_get函数位于rtthread/component/drivers/misc/pin.c文件中,

rt_base_t rt_pin_get(const char *name)
{
    RT_ASSERT(_hw_pin.ops != RT_NULL);
    RT_ASSERT(name[0] == 'P');
    
    if(_hw_pin.ops->pin_get == RT_NULL)
    {
        return -RT_ENOSYS;
    }

    return _hw_pin.ops->pin_get(name);
}

它最终是通过调用_hw_pin.ops->pin_get函数来获取引脚编号,在调用pin_get前,会先判断_hw_pin.ops->pin_get函数指针是否为空。因为这个函数指针为空,所以调用这个函数会返回错误码-RT_ENOSYS错误码,在不同的版本中RT_ENOSYS定义的整数可能不一样,把这个整数当引脚编号在rt_pin_*函数中使用会导致功能无法实现。

        _hw_pin这个对象是在rt_device_pin_register中初始化

int rt_device_pin_register(const char *name, const struct rt_pin_ops *ops, void *user_data)
{
    _hw_pin.parent.type         = RT_Device_Class_Miscellaneous;
    _hw_pin.parent.rx_indicate  = RT_NULL;
    _hw_pin.parent.tx_complete  = RT_NULL;

#ifdef RT_USING_DEVICE_OPS
    _hw_pin.parent.ops          = &pin_ops;
#else
    _hw_pin.parent.init         = RT_NULL;
    _hw_pin.parent.open         = RT_NULL;
    _hw_pin.parent.close        = RT_NULL;
    _hw_pin.parent.read         = _pin_read;
    _hw_pin.parent.write        = _pin_write;
    _hw_pin.parent.control      = _pin_control;
#endif

    _hw_pin.ops                 = ops;
    _hw_pin.parent.user_data    = user_data;

    /* register a character device */
    rt_device_register(&_hw_pin.parent, name, RT_DEVICE_FLAG_RDWR);

    return 0;
}

rt_device_pin_register函数是在rt_hw_pin_init函数中被调用

int rt_hw_pin_init(void)
{
#if defined(__HAL_RCC_GPIOA_CLK_ENABLE)
    __HAL_RCC_GPIOA_CLK_ENABLE();
#endif
    
#if defined(__HAL_RCC_GPIOB_CLK_ENABLE)
    __HAL_RCC_GPIOB_CLK_ENABLE();
#endif
    
#if defined(__HAL_RCC_GPIOC_CLK_ENABLE)
    __HAL_RCC_GPIOC_CLK_ENABLE();
#endif
    
#if defined(__HAL_RCC_GPIOD_CLK_ENABLE)
    __HAL_RCC_GPIOD_CLK_ENABLE();
#endif

#if defined(__HAL_RCC_GPIOE_CLK_ENABLE)
    __HAL_RCC_GPIOE_CLK_ENABLE();
#endif

#if defined(__HAL_RCC_GPIOF_CLK_ENABLE)
    __HAL_RCC_GPIOF_CLK_ENABLE();
#endif

#if defined(__HAL_RCC_GPIOG_CLK_ENABLE)
    #ifdef SOC_SERIES_STM32L4
        HAL_PWREx_EnableVddIO2();
    #endif
    __HAL_RCC_GPIOG_CLK_ENABLE();
#endif

#if defined(__HAL_RCC_GPIOH_CLK_ENABLE)
    __HAL_RCC_GPIOH_CLK_ENABLE();
#endif

#if defined(__HAL_RCC_GPIOI_CLK_ENABLE)
    __HAL_RCC_GPIOI_CLK_ENABLE();
#endif

#if defined(__HAL_RCC_GPIOJ_CLK_ENABLE)
    __HAL_RCC_GPIOJ_CLK_ENABLE();
#endif

#if defined(__HAL_RCC_GPIOK_CLK_ENABLE)
    __HAL_RCC_GPIOK_CLK_ENABLE();
#endif

    return rt_device_pin_register("pin", &_stm32_pin_ops, RT_NULL);
}

可以看到这里使用了_stm32_pin_ops对象去初始化pin的内核对象,_stm32_pin_ops是一个结构体对象。对比下面的结构体定义和对象定义,可以看出没有为pin_get函数指针赋初值,所以rt_pin_get中会返回-RT_NOSYS。

struct rt_pin_ops
{
    void (*pin_mode)(struct rt_device *device, rt_base_t pin, rt_base_t mode);
    void (*pin_write)(struct rt_device *device, rt_base_t pin, rt_base_t value);
    int (*pin_read)(struct rt_device *device, rt_base_t pin);

    /* TODO: add GPIO interrupt */
    rt_err_t (*pin_attach_irq)(struct rt_device *device, rt_int32_t pin,
                      rt_uint32_t mode, void (*hdr)(void *args), void *args);
    rt_err_t (*pin_detach_irq)(struct rt_device *device, rt_int32_t pin);
    rt_err_t (*pin_irq_enable)(struct rt_device *device, rt_base_t pin, rt_uint32_t enabled);
    rt_base_t (*pin_get)(const char *name);
};

const static struct rt_pin_ops _stm32_pin_ops =
{
    stm32_pin_mode,
    stm32_pin_write,
    stm32_pin_read,
    stm32_pin_attach_irq,
    stm32_pin_dettach_irq,
    stm32_pin_irq_enable,
};

解决方案:

在drv_gpio.c中新增一个根据名称获取pin引脚编号的函数如下

static rt_base_t stm32_pin_get(const char *name)
{
    int port, pin, sz, n;
    sz = rt_strlen(name);
    if ((sz == 4 || sz == 5) && name[0] == 'P' && name[2] == '.')
    {
        port = name[1] - 'A';
        pin  = name[3] - '0';
        if (0 <= port && port < 26 && 0 <= pin && pin <= 9)
        {
            if (sz == 5)
            {
                n = name[4] - '0';
                pin = (0 <= n && n <= 9) ? (pin * 10 + n) : 16; // 该mcu一个port16个引脚
            }
            int index = 16 * port + pin;
            if (index < sizeof(pins)/sizeof(struct pin_index))
            {
                return pins[index].index;
            }
        }
    }

    return -1;
}

在drv_gpio.c中修改_stm32_pin_ops结构体为如下内容

const static struct rt_pin_ops _stm32_pin_ops =
{
    stm32_pin_mode,
    stm32_pin_write,
    stm32_pin_read,
    stm32_pin_attach_irq,
    stm32_pin_dettach_irq,
    stm32_pin_irq_enable,
    stm32_pin_get,
};

经过上述修改,重新编译项目和烧录软件到开发板上运行,rt_pin_*函数操作能实现预期的功能。


网站公告

今日签到

点亮在社区的每一天
去签到