378 lines
10 KiB
Markdown
378 lines
10 KiB
Markdown
|
# 代码文件管理与组织
|
|||
|
|
|||
|
对于嵌入式项目,良好的文件管理结构至关重要。下面我将为您规划一个清晰的文件组织结构,适用于基于RT-Thread和GD32F470VET6的项目。
|
|||
|
|
|||
|
## 推荐的文件结构
|
|||
|
|
|||
|
```
|
|||
|
your_project_root/
|
|||
|
├── rt-thread/ # RT-Thread内核源码(作为子模块或直接复制)
|
|||
|
├── bsp/
|
|||
|
│ └── gd32/
|
|||
|
│ └── gd32f4xx/ # 您的BSP目录
|
|||
|
│ ├── applications/ # 应用层代码
|
|||
|
│ │ ├── main.c # 主入口文件
|
|||
|
│ │ ├── adc_app.c # ADC应用代码
|
|||
|
│ │ ├── adc_app.h
|
|||
|
│ │ ├── i2c_app.c # I2C应用代码
|
|||
|
│ │ ├── i2c_app.h
|
|||
|
│ │ └── SConscript # 应用层构建脚本
|
|||
|
│ ├── drivers/ # 驱动层代码
|
|||
|
│ │ ├── drv_adc.c # ADC驱动
|
|||
|
│ │ ├── drv_adc.h
|
|||
|
│ │ ├── drv_i2c.c # I2C驱动
|
|||
|
│ │ ├── drv_i2c.h
|
|||
|
│ │ ├── drv_usart.c # 串口驱动(调试用)
|
|||
|
│ │ ├── drv_usart.h
|
|||
|
│ │ ├── drv_gpio.c # GPIO驱动(如果需要)
|
|||
|
│ │ ├── drv_gpio.h
|
|||
|
│ │ └── SConscript # 驱动层构建脚本
|
|||
|
│ ├── libraries/ # 芯片厂商库
|
|||
|
│ │ ├── GD32F4xx_Standard_peripheral/
|
|||
|
│ │ │ ├── Include/
|
|||
|
│ │ │ └── Source/
|
|||
|
│ │ └── CMSIS/
|
|||
|
│ ├── board.c # 板级初始化
|
|||
|
│ ├── board.h # 板级相关定义
|
|||
|
│ ├── Kconfig # 配置选项
|
|||
|
│ ├── SConstruct # 构建脚本
|
|||
|
│ └── rtconfig.h # RT-Thread配置
|
|||
|
├── packages/ # RT-Thread软件包
|
|||
|
├── tools/ # 工具脚本
|
|||
|
└── documents/ # 项目文档
|
|||
|
```
|
|||
|
|
|||
|
## 各文件内容示例
|
|||
|
|
|||
|
### 1. 驱动层文件 (`drivers/drv_usart.c`)
|
|||
|
|
|||
|
```c
|
|||
|
#include "drv_usart.h"
|
|||
|
#include "gd32f4xx.h"
|
|||
|
|
|||
|
// 串口配置结构体
|
|||
|
static struct gd32_usart_config usart_config[] = {
|
|||
|
{
|
|||
|
.name = "uart0",
|
|||
|
.usart_periph = USART0,
|
|||
|
.rcu_periph_clock = RCU_USART0,
|
|||
|
.gpio_port = GPIOA,
|
|||
|
.rcu_gpio_clock = RCU_GPIOA,
|
|||
|
.tx_pin = GPIO_PIN_9,
|
|||
|
.rx_pin = GPIO_PIN_10,
|
|||
|
.baud_rate = 115200,
|
|||
|
.irq_type = USART0_IRQn
|
|||
|
},
|
|||
|
// 可以添加更多串口配置
|
|||
|
};
|
|||
|
|
|||
|
// 初始化所有配置的串口
|
|||
|
void gd32_usart_init(void)
|
|||
|
{
|
|||
|
for (int i = 0; i < sizeof(usart_config) / sizeof(usart_config[0]); i++) {
|
|||
|
gd32_usart_configure(&usart_config[i]);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// 配置单个串口
|
|||
|
void gd32_usart_configure(struct gd32_usart_config *config)
|
|||
|
{
|
|||
|
// 启用时钟
|
|||
|
rcu_periph_clock_enable(config->rcu_periph_clock);
|
|||
|
rcu_periph_clock_enable(config->rcu_gpio_clock);
|
|||
|
|
|||
|
// 配置GPIO
|
|||
|
gpio_af_set(config->gpio_port, GPIO_AF_7, config->tx_pin);
|
|||
|
gpio_af_set(config->gpio_port, GPIO_AF_7, config->rx_pin);
|
|||
|
gpio_mode_set(config->gpio_port, GPIO_MODE_AF, GPIO_PUPD_PULLUP, config->tx_pin);
|
|||
|
gpio_mode_set(config->gpio_port, GPIO_MODE_AF, GPIO_PUPD_PULLUP, config->rx_pin);
|
|||
|
gpio_output_options_set(config->gpio_port, GPIO_OTYPE_PP, GPIO_OSPEED_50MHZ, config->tx_pin);
|
|||
|
gpio_output_options_set(config->gpio_port, GPIO_OTYPE_PP, GPIO_OSPEED_50MHZ, config->rx_pin);
|
|||
|
|
|||
|
// 配置USART
|
|||
|
usart_deinit(config->usart_periph);
|
|||
|
usart_baudrate_set(config->usart_periph, config->baud_rate);
|
|||
|
usart_word_length_set(config->usart_periph, USART_WL_8BIT);
|
|||
|
usart_stop_bit_set(config->usart_periph, USART_STB_1BIT);
|
|||
|
usart_parity_config(config->usart_periph, USART_PM_NONE);
|
|||
|
usart_hardware_flow_rts_config(config->usart_periph, USART_RTS_DISABLE);
|
|||
|
usart_hardware_flow_cts_config(config->usart_periph, USART_CTS_DISABLE);
|
|||
|
usart_receive_config(config->usart_periph, USART_RECEIVE_ENABLE);
|
|||
|
usart_transmit_config(config->usart_periph, USART_TRANSMIT_ENABLE);
|
|||
|
|
|||
|
// 启用USART
|
|||
|
usart_enable(config->usart_periph);
|
|||
|
|
|||
|
// 配置中断(如果需要)
|
|||
|
// nvic_irq_enable(config->irq_type, 0, 0);
|
|||
|
// usart_interrupt_enable(config->usart_periph, USART_INT_RBNE);
|
|||
|
}
|
|||
|
|
|||
|
// 发送字符
|
|||
|
void gd32_usart_putchar(struct gd32_usart_config *config, char ch)
|
|||
|
{
|
|||
|
usart_data_transmit(config->usart_periph, (uint16_t)ch);
|
|||
|
while (usart_flag_get(config->usart_periph, USART_FLAG_TBE) == RESET);
|
|||
|
}
|
|||
|
|
|||
|
// 发送字符串
|
|||
|
void gd32_usart_puts(struct gd32_usart_config *config, const char *str)
|
|||
|
{
|
|||
|
while (*str) {
|
|||
|
gd32_usart_putchar(config, *str++);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// 获取串口配置 by name
|
|||
|
struct gd32_usart_config *gd32_usart_get_config(const char *name)
|
|||
|
{
|
|||
|
for (int i = 0; i < sizeof(usart_config) / sizeof(usart_config[0]); i++) {
|
|||
|
if (rt_strcmp(usart_config[i].name, name) == 0) {
|
|||
|
return &usart_config[i];
|
|||
|
}
|
|||
|
}
|
|||
|
return RT_NULL;
|
|||
|
}
|
|||
|
```
|
|||
|
|
|||
|
### 2. 驱动层头文件 (`drivers/drv_usart.h`)
|
|||
|
|
|||
|
```c
|
|||
|
#ifndef __DRV_USART_H__
|
|||
|
#define __DRV_USART_H__
|
|||
|
|
|||
|
#include <rtthread.h>
|
|||
|
#include <rtdevice.h>
|
|||
|
#include "gd32f4xx.h"
|
|||
|
|
|||
|
// 串口配置结构体
|
|||
|
struct gd32_usart_config {
|
|||
|
const char *name;
|
|||
|
uint32_t usart_periph;
|
|||
|
uint32_t rcu_periph_clock;
|
|||
|
uint32_t gpio_port;
|
|||
|
uint32_t rcu_gpio_clock;
|
|||
|
uint32_t tx_pin;
|
|||
|
uint32_t rx_pin;
|
|||
|
uint32_t baud_rate;
|
|||
|
uint8_t irq_type;
|
|||
|
};
|
|||
|
|
|||
|
// 函数声明
|
|||
|
void gd32_usart_init(void);
|
|||
|
void gd32_usart_configure(struct gd32_usart_config *config);
|
|||
|
void gd32_usart_putchar(struct gd32_usart_config *config, char ch);
|
|||
|
void gd32_usart_puts(struct gd32_usart_config *config, const char *str);
|
|||
|
struct gd32_usart_config *gd32_usart_get_config(const char *name);
|
|||
|
|
|||
|
#endif /* __DRV_USART_H__ */
|
|||
|
```
|
|||
|
|
|||
|
### 3. 应用层文件 (`applications/adc_app.c`)
|
|||
|
|
|||
|
```c
|
|||
|
#include "adc_app.h"
|
|||
|
#include "drv_usart.h"
|
|||
|
#include "drv_adc.h"
|
|||
|
#include <rtthread.h>
|
|||
|
#include <rtdevice.h>
|
|||
|
|
|||
|
// 定义调试串口
|
|||
|
#define DEBUG_USART "uart0"
|
|||
|
|
|||
|
// ADC调试线程
|
|||
|
static void adc_debug_thread_entry(void *parameter)
|
|||
|
{
|
|||
|
struct gd32_usart_config *usart_cfg = gd32_usart_get_config(DEBUG_USART);
|
|||
|
|
|||
|
if (usart_cfg == RT_NULL) {
|
|||
|
rt_kprintf("Error: Debug USART not found!\n");
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
// 初始化ADC
|
|||
|
if (gd32_adc_init() != RT_EOK) {
|
|||
|
gd32_usart_puts(usart_cfg, "Error: ADC initialization failed!\r\n");
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
gd32_usart_puts(usart_cfg, "ADC Debug Started\r\n");
|
|||
|
gd32_usart_puts(usart_cfg, "Sampling ADC channel 0 every second...\r\n");
|
|||
|
|
|||
|
while (1) {
|
|||
|
// 读取ADC值
|
|||
|
uint16_t adc_value = gd32_adc_read(0); // 假设读取通道0
|
|||
|
float voltage = (adc_value * 3.3f) / 4095.0f;
|
|||
|
|
|||
|
// 通过串口输出
|
|||
|
char buffer[64];
|
|||
|
rt_snprintf(buffer, sizeof(buffer), "ADC Value: %d, Voltage: %.2fV\r\n", adc_value, voltage);
|
|||
|
gd32_usart_puts(usart_cfg, buffer);
|
|||
|
|
|||
|
// 延时1秒
|
|||
|
rt_thread_mdelay(1000);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// 启动ADC调试
|
|||
|
int adc_debug_start(void)
|
|||
|
{
|
|||
|
rt_thread_t tid;
|
|||
|
|
|||
|
tid = rt_thread_create("adc_dbg",
|
|||
|
adc_debug_thread_entry,
|
|||
|
RT_NULL,
|
|||
|
1024,
|
|||
|
RT_THREAD_PRIORITY_MAX / 2,
|
|||
|
20);
|
|||
|
if (tid != RT_NULL) {
|
|||
|
rt_thread_startup(tid);
|
|||
|
return RT_EOK;
|
|||
|
}
|
|||
|
|
|||
|
return -RT_ERROR;
|
|||
|
}
|
|||
|
|
|||
|
// 导出到MSH命令
|
|||
|
MSH_CMD_EXPORT(adc_debug_start, Start ADC debug);
|
|||
|
```
|
|||
|
|
|||
|
### 4. 应用层头文件 (`applications/adc_app.h`)
|
|||
|
|
|||
|
```c
|
|||
|
#ifndef __ADC_APP_H__
|
|||
|
#define __ADC_APP_H__
|
|||
|
|
|||
|
#include <rtthread.h>
|
|||
|
|
|||
|
// 函数声明
|
|||
|
int adc_debug_start(void);
|
|||
|
|
|||
|
#endif /* __ADC_APP_H__ */
|
|||
|
```
|
|||
|
|
|||
|
### 5. 板级初始化 (`board.c`)
|
|||
|
|
|||
|
```c
|
|||
|
#include <rtthread.h>
|
|||
|
#include <rtdevice.h>
|
|||
|
#include "board.h"
|
|||
|
#include "drv_usart.h"
|
|||
|
|
|||
|
// 系统时钟配置
|
|||
|
void system_clock_config(void)
|
|||
|
{
|
|||
|
// 配置系统时钟为200MHz
|
|||
|
// ... GD32特定的时钟配置代码
|
|||
|
}
|
|||
|
|
|||
|
// 板级初始化
|
|||
|
void rt_hw_board_init(void)
|
|||
|
{
|
|||
|
// 配置系统时钟
|
|||
|
system_clock_config();
|
|||
|
|
|||
|
// 初始化调试串口
|
|||
|
gd32_usart_init();
|
|||
|
|
|||
|
// 打印启动信息
|
|||
|
struct gd32_usart_config *usart_cfg = gd32_usart_get_config("uart0");
|
|||
|
if (usart_cfg != RT_NULL) {
|
|||
|
gd32_usart_puts(usart_cfg, "\r\nRT-Thread GD32F470 Board Initialized\r\n");
|
|||
|
}
|
|||
|
|
|||
|
// 其他硬件初始化...
|
|||
|
|
|||
|
// 显示RT-Thread版本信息
|
|||
|
rt_show_version();
|
|||
|
|
|||
|
// 初始化系统堆内存
|
|||
|
#ifdef RT_USING_HEAP
|
|||
|
rt_system_heap_init((void *)HEAP_BEGIN, (void *)HEAP_END);
|
|||
|
#endif
|
|||
|
}
|
|||
|
```
|
|||
|
|
|||
|
### 6. 构建脚本 (`drivers/SConscript`)
|
|||
|
|
|||
|
```python
|
|||
|
from building import *
|
|||
|
|
|||
|
cwd = GetCurrentDir()
|
|||
|
src = Glob('*.c')
|
|||
|
path = [cwd]
|
|||
|
|
|||
|
# 定义编译组
|
|||
|
group = DefineGroup('Drivers', src, depend = [''], path = path)
|
|||
|
|
|||
|
Return('group')
|
|||
|
```
|
|||
|
|
|||
|
### 7. 应用层构建脚本 (`applications/SConscript`)
|
|||
|
|
|||
|
```python
|
|||
|
from building import *
|
|||
|
|
|||
|
cwd = GetCurrentDir()
|
|||
|
src = Glob('*.c')
|
|||
|
path = [cwd]
|
|||
|
|
|||
|
# 定义编译组
|
|||
|
group = DefineGroup('Applications', src, depend = [''], path = path)
|
|||
|
|
|||
|
Return('group')
|
|||
|
```
|
|||
|
|
|||
|
## 编译和调试流程
|
|||
|
|
|||
|
1. **环境设置**:
|
|||
|
|
|||
|
- 确保RT-Thread Env工具已正确设置
|
|||
|
- 配置工具链路径(对于IAR,需要在Env中设置相关路径)
|
|||
|
2. **配置项目**:
|
|||
|
|
|||
|
```bash
|
|||
|
# 在BSP目录下打开Env
|
|||
|
menuconfig
|
|||
|
```
|
|||
|
|
|||
|
- 启用USART设备驱动
|
|||
|
- 启用ADC设备驱动
|
|||
|
- 配置其他需要的组件
|
|||
|
3. **编译项目**:
|
|||
|
|
|||
|
```bash
|
|||
|
scons
|
|||
|
```
|
|||
|
|
|||
|
或使用IAR工程文件(如果已创建)
|
|||
|
4. **下载和调试**:
|
|||
|
|
|||
|
- 使用J-Link、GD-Link或ST-Link等调试器下载程序
|
|||
|
- 通过串口调试助手查看输出信息
|
|||
|
|
|||
|
## 调试建议
|
|||
|
|
|||
|
1. **分层调试**:
|
|||
|
|
|||
|
- 首先确保串口驱动正常工作
|
|||
|
- 然后调试ADC驱动
|
|||
|
- 最后实现应用逻辑
|
|||
|
2. **使用RT-Thread的Finsh控制台**:
|
|||
|
|
|||
|
- 可以通过串口直接输入命令控制应用程序
|
|||
|
- 使用 `list_device`命令查看已注册的设备
|
|||
|
- 使用 `adc_debug_start`命令启动ADC调试
|
|||
|
3. **添加调试宏**:
|
|||
|
|
|||
|
```c
|
|||
|
#define DEBUG_ADC
|
|||
|
#ifdef DEBUG_ADC
|
|||
|
#define ADC_DEBUG(fmt, ...) rt_kprintf("[ADC] " fmt, ##__VA_ARGS__)
|
|||
|
#else
|
|||
|
#define ADC_DEBUG(fmt, ...)
|
|||
|
#endif
|
|||
|
```
|
|||
|
|
|||
|
这样的文件组织结构清晰,便于维护和扩展,也符合RT-Thread的推荐实践。随着项目的发展,您可以轻松地添加新的驱动和应用模块。
|