从点亮LED到驱动外设:手把手教你用RT-Thread玩转星火一号开发板
从点亮LED到驱动外设手把手教你用RT-Thread玩转星火一号开发板第一次拿到星火一号开发板时面对密密麻麻的芯片引脚和陌生的RT-Thread环境不少开发者会感到无从下手。本文将带你从最基础的LED控制开始逐步深入SPI Flash读写、串口通信等核心功能用实际项目串联起RT-Thread的开发全流程。1. 开发环境搭建与第一个LED程序星火一号开发板搭载的STM32F407ZGT6芯片拥有168MHz主频和丰富外设资源但要让这颗心脏跳动起来首先需要配置好开发环境。不同于裸机开发RT-Thread作为实时操作系统提供了更高效的资源管理方式。必备工具清单MDK Keil 5.30或IAR 8.50ST-Link/V2调试器RT-Thread ENV工具PuTTY/Tera Term串口终端安装完基础工具后通过以下命令克隆BSP源码git clone https://github.com/RT-Thread/rt-thread.git cd rt-thread/bsp/stm32/stm32f407-rt-spark打开工程后在applications/main.c中添加LED控制代码#include rtthread.h #include rtdevice.h #define LED_PIN GET_PIN(F, 11) void led_thread_entry(void *parameter) { rt_pin_mode(LED_PIN, PIN_MODE_OUTPUT); while(1) { rt_pin_write(LED_PIN, PIN_HIGH); rt_thread_mdelay(500); rt_pin_write(LED_PIN, PIN_LOW); rt_thread_mdelay(500); } } int main(void) { rt_thread_t tid rt_thread_create(led, led_thread_entry, RT_NULL, 512, 20, 10); if(tid ! RT_NULL) rt_thread_startup(tid); return RT_EOK; }提示RT-Thread的PIN驱动框架统一了不同芯片的GPIO操作接口GET_PIN(F,11)对应PF11引脚板载绿色LED2. 深入RT-Thread驱动框架2.1 设备驱动模型解析RT-Thread采用经典的UNIX设备驱动模型所有外设都被抽象为rt_device结构体。通过以下代码可以查看当前已注册设备#include rtdevice.h void list_devices(void) { struct rt_device *dev RT_NULL; rt_ubase_t index 0; rt_kprintf(--- Registered Devices ---\n); while ((dev rt_device_find_by_index(index)) ! RT_NULL) { rt_kprintf(%-16s %s\n, dev-parent.name, dev-flag RT_DEVICE_FLAG_ACTIVATED ? ACTIVE : INACTIVE); } } MSH_CMD_EXPORT(list_devices, List all registered devices);2.2 SPI Flash驱动实战星火一号板载W25Q12816MB SPI Flash通过以下步骤实现读写操作在ENV工具中启用SPI驱动menuconfig → Hardware Drivers Config → On-chip Peripheral Drivers → Enable SPI1添加Flash设备初始化代码#include drv_spi.h #include spi_flash.h static int spi_flash_init(void) { __HAL_RCC_GPIOB_CLK_ENABLE(); struct rt_spi_device *spi_dev; rt_hw_spi_device_attach(spi1, spi10, GPIOB, GPIO_PIN_14); spi_dev (struct rt_spi_device *)rt_device_find(spi10); struct rt_spi_configuration cfg; cfg.data_width 8; cfg.mode RT_SPI_MODE_0 | RT_SPI_MSB; cfg.max_hz 30 * 1000 * 1000; rt_spi_configure(spi_dev, cfg); return RT_EOK; } INIT_DEVICE_EXPORT(spi_flash_init);使用SFUD组件操作Flashvoid test_flash(void) { struct rt_spi_device *spi_dev rt_device_find(spi10); struct rt_spi_flash_device *flash rt_sfud_flash_probe(W25Q128, spi10); /* 读写测试 */ rt_uint8_t buf[256], rbuf[256]; rt_memset(buf, 0x55, sizeof(buf)); rt_spi_flash_write(flash, 0, buf, sizeof(buf)); rt_spi_flash_read(flash, 0, rbuf, sizeof(rbuf)); if(rt_memcmp(buf, rbuf, sizeof(buf)) 0) { rt_kprintf(SPI Flash test success!\n); } } MSH_CMD_EXPORT(test_flash, Test W25Q128 operations);3. 多线程与IPC实战3.1 创建管理多个线程RT-Thread的线程调度器支持优先级抢占下面示例创建三个不同优先级的线程static rt_thread_t tid1, tid2, tid3; static void thread1_entry(void *param) { while(1) { rt_kprintf([Thread1] priority 8 running\n); rt_thread_mdelay(1000); } } static void thread2_entry(void *param) { while(1) { rt_kprintf([Thread2] priority 10 running\n); rt_thread_mdelay(1500); } } void thread_sample(void) { tid1 rt_thread_create(thread1, thread1_entry, RT_NULL, 1024, 8, 10); tid2 rt_thread_create(thread2, thread2_entry, RT_NULL, 1024, 10, 10); rt_thread_startup(tid1); rt_thread_startup(tid2); }3.2 使用消息队列通信线程间通信是RTOS的核心功能以下示例展示消息队列的使用#include rtthread.h #define MSG_QUEUE_SIZE 10 static struct rt_messagequeue mq; static rt_uint8_t msg_pool[MSG_QUEUE_SIZE * sizeof(rt_uint32_t)]; static void sender_thread_entry(void *param) { rt_uint32_t count 0; while(1) { if(rt_mq_send(mq, count, sizeof(count)) RT_EOK) { rt_kprintf([Sender] send message: %d\n, count); } rt_thread_mdelay(500); } } static void receiver_thread_entry(void *param) { rt_uint32_t rx_data; while(1) { if(rt_mq_recv(mq, rx_data, sizeof(rx_data), RT_WAITING_FOREVER) RT_EOK) { rt_kprintf([Receiver] got message: %d\n, rx_data); } } } int ipc_sample(void) { rt_mq_init(mq, msgq, msg_pool, sizeof(rt_uint32_t), sizeof(msg_pool), RT_IPC_FLAG_FIFO); rt_thread_t sender rt_thread_create(sender, sender_thread_entry, RT_NULL, 512, 15, 10); rt_thread_t receiver rt_thread_create(receiver, receiver_thread_entry, RT_NULL, 512, 15, 10); rt_thread_startup(sender); rt_thread_startup(receiver); return RT_EOK; } MSH_CMD_EXPORT(ipc_sample, Message queue sample);4. 外设进阶开发技巧4.1 ADC采样与数据处理星火一号开发板提供了多个ADC通道以下代码实现周期采样与滤波#include rtdevice.h #define ADC_DEV_NAME adc1 #define ADC_CHANNEL 5 static void adc_thread_entry(void *param) { rt_adc_device_t adc_dev (rt_adc_device_t)rt_device_find(ADC_DEV_NAME); rt_adc_enable(adc_dev, ADC_CHANNEL); float avg 0; rt_uint32_t samples[10]; while(1) { /* 采集10次取平均 */ for(int i0; i10; i) { samples[i] rt_adc_read(adc_dev, ADC_CHANNEL); rt_thread_mdelay(10); } /* 移动平均滤波 */ avg 0; for(int i0; i10; i) avg samples[i]; avg / 10; rt_kprintf(ADC value: %.2f\n, avg * 3.3 / 4096); rt_thread_mdelay(1000); } } int adc_sample(void) { rt_thread_t tid rt_thread_create(adc, adc_thread_entry, RT_NULL, 512, 20, 10); if(tid) rt_thread_startup(tid); return RT_EOK; }4.2 硬件定时器精确定时对于需要精确时间控制的场景可以使用STM32的硬件定时器#include rtdevice.h static rt_device_t hwtimer_dev; static rt_err_t timeout_cb(rt_device_t dev, rt_size_t size) { rt_kprintf(Hardware timer timeout!\n); return RT_EOK; } int timer_sample(void) { hwtimer_dev rt_device_find(timer2); rt_device_set_rx_indicate(hwtimer_dev, timeout_cb); rt_device_control(hwtimer_dev, HWTIMER_CTRL_FREQ_SET, (void *)1000000); rt_hwtimer_mode_t mode HWTIMER_MODE_PERIOD; rt_device_control(hwtimer_dev, HWTIMER_CTRL_MODE_SET, mode); rt_hwtimerval_t timeout_s {.sec1, .usec0}; rt_device_write(hwtimer_dev, 0, timeout_s, sizeof(timeout_s)); return RT_EOK; }5. 调试与性能优化5.1 使用ulog组件RT-Thread的ulog组件提供分级日志功能在rtconfig.h中配置#define ULOG_USING_SYSLOG #define ULOG_OUTPUT_LVL_D #define ULOG_OUTPUT_TIME #define ULOG_OUTPUT_LEVEL #define ULOG_OUTPUT_THREAD_NAME在代码中使用不同级别日志#include ulog.h void log_demo(void) { LOG_D(This is debug message); LOG_I(System running time: %d, rt_tick_get()); LOG_W(Low memory warning!); LOG_E(I2C communication failed); }5.2 内存使用分析通过以下命令查看系统资源占用msh free total memory: 192048 used memory: 65432 maximum allocated memory: 72320 msh list_thread thread pri status sp stack size max used left tick error ------ --- ------ -- ---------- -------- --------- --- led 20 running 0x00000060 0x00000200 28% 0x0000000a 000 tshell 10 suspend 0x00000080 0x00001000 15% 0x00000014 000在项目开发中遇到SPI通信不稳定时通过逻辑分析仪抓取波形发现时钟极性配置错误。修改RT_SPI_MODE_0为RT_SPI_MODE_3后问题解决这种实战经验往往比文档更有价值。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2574895.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!