前言
-
前几篇已经 通过
STM32CubeMX
搭建了NUCLEO-L476RG
的 STM32L476RG 的 裸机工程,下载了 uC-OS2 V2.93 的源码,并把 uC-OS2 的源文件加入 Keil MDK5 工程,通过适配 Systick 系统定时器与 PendSV 实现任务调度,初步让 uC-OS2 运行起来 -
本篇适配 uC-OS2 的 串口驱动,实现 类似于 printf 的打印功能,让 uC-OS2 有串口运行信息
开发环境
-
win10 64位
-
Keil uVision5,MDK V5.36
-
uC-OS2 V2.93
-
开发板:NUCLEO-L476RG ,MCU 为 STM32L476RG
-
STM32CubeMX 6.9.1,用于生成 STM32的裸机工程
串口驱动
-
通过
STM32CubeMX
,已经配置了串口的驱动,默认串口的波特率是 115200 bps,不过没有串口输出的接口,需要自己完善一下 -
新建
uart.c
,驱动如下
#include "uart2.h"
#include <stdio.h>
#include <stdarg.h>
#include <string.h>
#include "stm32l4xx_hal.h"
static UART_HandleTypeDef huart2;
#define DBG_BUFF_MAX_LEN 256
static char rt_log_buf[DBG_BUFF_MAX_LEN] = { 0 };
void uart2_put(const char *fmt)
{
HAL_UART_Transmit(&huart2, (uint8_t *)fmt, strlen(fmt), 0xFFFF);
}
/* debug print */
int os_printf(const char *fmt, ...)
{
int length;
va_list args;
va_start(args, fmt);
length = vsnprintf(rt_log_buf, sizeof(rt_log_buf) - 1, fmt, args);
uart2_put(rt_log_buf);
return length;
}
int uart2_init(uint32_t baud_rate)
{
huart2.Instance = USART2;
huart2.Init.BaudRate = baud_rate;
huart2.Init.WordLength = UART_WORDLENGTH_8B;
huart2.Init.StopBits = UART_STOPBITS_1;
huart2.Init.Parity = UART_PARITY_NONE;
huart2.Init.Mode = UART_MODE_TX_RX;
huart2.Init.HwFlowCtl = UART_HWCONTROL_NONE;
huart2.Init.OverSampling = UART_OVERSAMPLING_16;
huart2.Init.OneBitSampling = UART_ONE_BIT_SAMPLE_DISABLE;
huart2.AdvancedInit.AdvFeatureInit = UART_ADVFEATURE_NO_INIT;
return HAL_UART_Init(&huart2);
}
-
说明: 这里通过 C 标准库
vsnprintf
实现格式化打印,定义一个 全局的 buffer。 -
新建
uart2.h
头文件
#ifndef __UART2_H__
#define __UART2_H__
#include <stdint.h>
int uart2_init(uint32_t baud_rate);
int os_printf(const char *fmt, ...);
#endif
main 函数调用
-
首先包含
#include "uart2.h"
,然后 初始化串口:uart2_init(UART2_BAUD_RATE);
,然后就可以使用os_printf
格式化打印了 -
格式化打印的意思就是可以像 printf 那样,把一个数值 打印输出 十进制、十六进制等工程,也可以
%s
输出一个字符串 -
修改后的 main.c 如下
#include "main.h"
#include "led.h"
#include "app_cfg.h"
#include "os.h"
#include "uart2.h"
void SystemClock_Config(void);
static void MX_GPIO_Init(void);
#define UART2_BAUD_RATE 115200
#define MCU_FREQUENCY 80000000
#define TASK_LED_PRIO 5
#define TASK_LED_STACK_SIZE 128
static OS_STK task_led_stack[TASK_LED_STACK_SIZE];
static void task_led_entry(void *p_arg)
{
int cnt = 0;
while (1)
{
led_grn_ctrl(1);
OSTimeDly(1000);
led_grn_ctrl(0);
OSTimeDly(1000);
cnt++;
os_printf("%s : cnt : %d\r\n", __func__, cnt);
}
}
void led_task_init(void)
{
OSTaskCreate(task_led_entry,(void *)0, &task_led_stack[TASK_LED_STACK_SIZE-1], TASK_LED_PRIO);
}
HAL_StatusTypeDef HAL_InitTick(uint32_t TickPriority)
{
return HAL_OK;
}
/**
* @brief The application entry point.
* @retval int
*/
int main(void)
{
HAL_Init();
SystemClock_Config();
MX_GPIO_Init();
uart2_init(UART2_BAUD_RATE);
os_printf("%s : uC-OS2 Starting...\r\n", __func__);
OSInit();
led_task_init();
OS_CPU_SysTickInitFreq(MCU_FREQUENCY);
OSStart();
return 0;
}
串口信息
-
编译、烧写到开发板,查看 串口的信息,当前配置的串口波特率是 115200 bps
-
当前串口打印信息正常
小结
-
本篇主要在 uC-OS2 上实现 类似于 printf 的串口格式化打印输出功能,适配串口驱动
-
接下来,继续研究 uC-OS2 上实现串口的 shell 等功能,不断的深入熟悉 uC-OS2 的各个模块