项目场景:
用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_*函数操作能实现预期的功能。