前言
(1)学习本文之前,需要先学习前两篇文章。
(2)学习本文需要准备好TTL转USB模块。
函数介绍
ROM_GPIOPinConfigure()
配置GPIO引脚的复用功能。因为引脚不可能只有一个输出输入作用,还可能能够当作复用引脚。所以这个函数用于当IO需要用于其他功能的时候,调用这个函数。
/****** 函数声明 ******/
//这个存放在ROM
void ROM_GPIOPinConfigure(uint32_t ui32PinConfig);
//这个是存放在flash
void GPIOPinConfigure(uint32_t ui32PinConfig);
/****** 函数介绍 ******/
/* 作用 : 配置GPIO引脚的复用功能。
* 传入参数 : 参数在 pin_map.h 中选择,例如 GPIOPinConfigure(GPIO_PB0_U1RX);是把 PB0 复用为
U1RX
* 返回参数 : 无
*/
ROM_GPIOPinTypeUART()
配置UART外设使用的引脚。
/****** 函数声明 ******/
//这个存放在ROM
void ROM_GPIOPinTypeUART(uint32_t ui32Port,uint8_t ui8Pins);
//这个是存放在flash
void GPIOPinTypeUART(uint32_t ui32Port,uint8_t ui8Pins);
/****** 函数介绍 ******/
/* 作用 : 配置UART外设使用的引脚。
* 传入参数 :
* ui32Port : GPIO口的基地地址,GPIO_PORTx_BASE,x可为A,B,C,D,E,F,G,H
* ui8Pins : GPIO_PIN_X,x可为1,2,3,4,5,6,7
* 返回参数 : 无
*/
ROM_UARTConfigSetExpClk()
(1)设置UART的配置。
(2)这里需要注意一个点,UARTStdioConfig()函数也可以配置串口,但是我个人不建议使用这个配置。因为这个函数配置了之后,默认为使用 8 位数据,没有奇偶校验位,1 停止位。虽然我们后面也是这样配置。
(3)UARTStdioConfig()这个函数有一个很重要的点,如果是调用这个函数配置的串口,可以使用UARTprintf()函数,却不能使用UARTCharput()函数,用UARTConfigSetExpClk()配置的函数可以使用UARTCharput()函数,却不能使用UARTprintf()函数。
(4)UARTprintf()是一个伪printf()函数,使用方法与printf()函数一致。但是这样做的话会有两个问题:
<1>如果调用UARTprintf()函数,但是单片机有多个串口同时打开了,都需要输出。这样UARTprintf()函数到底是对应的哪一个串口呢?
<2>这个函数使用起来不灵活。比如,数据类型限制:UARTprintf函数只支持有限的数据类型,例如整数和浮点数等。如果需要输出其他数据类型,可能需要自己编写输出函数。格式化限制:UARTprintf函数使用标准的printf格式化字符串语法,但是并不支持所有的printf格式化字符串选项。例如,它不支持长整型(long)和长双精度浮点型(long double)等类型。缓冲区大小限制:UARTprintf函数使用一个内部的静态缓冲区来存储格式化后的字符串,因此输出的字符串长度受到缓冲区大小的限制。如果输出的字符串长度超过了缓冲区大小,可能会导致输出不完整或被截断。
/****** 函数声明 ******/
//这个存放在ROM
void ROM_UARTConfigSetExpClk(uint32_t ui32Base,uint32_t ui32UARTClk,uint32_t ui32Baud,uint32_t ui32Config);
//这个是存放在flash
void UARTConfigSetExpClk(uint32_t ui32Base,uint32_t ui32UARTClk,uint32_t ui32Baud,uint32_t ui32Config);
/****** 函数介绍 ******/
/* 作用 : 设置UART的配置。
* 传入参数 :
* ui32Base : UART端口的基地址
* ui32UARTClk : 提供给UART模块的时钟速率。
* ui32Baud : 波特率
* ui32Config : 可选参数如下:
* 数据位 : UART_CONFIG_WLEN_8(8数据位),
UART_CONFIG_WLEN_7(7数据位),
UART_CONFIG_WLEN_6(6数据位),
UART_CONFIG_WLEN_5(5数据位)
* 停止位 : UART_CONFIG_STOP_ONE(1停止位),
UART_CONFIG_STOP_TWO (2停止位)
* 校验位 : UART_CONFIG_PAR_NONE(无校验位) ,
UART_CONFIG_PAR_EVEN(偶校验),
UART_CONFIG_PAR_ODD (奇校验),
UART_CONFIG_PAR_ONE (1校验),
UART_CONFIG_PAR_ZERO (0校验)
* 返回参数 : 无
*/
ROM_UARTFIFODisable()
(1)用于禁用FIFO。
(2)FIFO是一个寄存器,串口通讯的数据可以将FIFO作为中转站,CPU需要发送数据,就将数据一次性发给FIFO,然后FIFO来处理。如果接收数据,也是先将数据接收到FIFO中,然后CPU再去处理。这样做的好处:
<1>这样做允许串口同时处理多个字符,而不需要等待每个字符都被处理完毕。这可以减少串口通信的延迟,提高通信效率。
<2>允许串口同时处理多个字符,而不需要等待每个字符都被处理完毕。这可以减少串口通信的延迟,提高通信效率。
<3>当使能FIFO缓冲区时,串口接收到的数据将被存储在FIFO缓冲区中,而不是直接传输到接收缓冲区。这可以减少中断的数量,从而提高系统响应速度。(注意:这样做的话串口的实时性就会降低很多!!!)
(3)因为我们想要串口接收到数据就马上进行反应,所以我们将FIFO禁用。
/****** 函数声明 ******/
//这个存放在ROM
void ROM_UARTFIFODisable(uint32_t ui32Base);
//这个是存放在flash
void ROM_UARTFIFODisable(uint32_t ui32Base);
/****** 函数介绍 ******/
/* 作用 : 用于禁用FIFO。
* 传入参数 :
* ui32Base :要禁用FIFO的UART端口的基地址
* 返回参数 : 无
*/
ROM_UARTIntEnable()
启用单独的UART中断源。
/****** 函数声明 ******/
//这个存放在ROM
void ROM_UARTIntEnable(uint32_t ui32Base,uint32_t ui32IntFlags);
//这个是存放在flash
void UARTIntEnable(uint32_t ui32Base,uint32_t ui32IntFlags);
/****** 函数介绍 ******/
/* 作用 : 用于禁用FIFO。
* 传入参数 :
* ui32Base :UART端口的基地址
* ui32IntFlags : UART_INT_9BIT(9位地址匹配中断),
UART_INT_OE(错误中断,接收到的数据中出现了错误的“停止位”),
UART_INT_BE(错误中断,数据接收器的缓冲区已满时,继续接收数据而导致之前接收到的数据被覆盖,从而引发中断),
UART_INT_PE(奇偶校验错误中断),
UART_INT_FE(帧错误中断),
UART_INT_RT(接收超时中断),
UART_INT_TX(发送中断),
UART_INT_RX(接收中断),
UART_INT_DSR(DSR中断),
UART_INT_DCD(DCD中断),
UART_INT_CTS(CTS中断),
UART_INT_RI(RI中断)
* 返回参数 : 无
*/
UARTIntRegister()
(1)UART中断函数注册
(2)这个的ROM函数似乎没有
/****** 函数声明 ******/
//这个是存放在flash
void UARTIntRegister(uint32_t ui32Base, void (*pfnHandler)(void));
/****** 函数介绍 ******/
/* 作用 : UART中断函数注册
* 传入参数 :
* ui32Base :UART端口的基地址
* void (*pfnHandler)(void) : 串口中断函数名字,此函数无返回值,无传入值
* 返回参数 : 无
*/
ROM_IntPrioritySet()
设置中断的优先级。
/****** 函数声明 ******/
//这个存放在ROM
void ROM_IntPrioritySet(uint32_t ui32Interrupt,uint8_t ui8Priority);
//这个是存放在flash
void IntPrioritySet(uint32_t ui32Interrupt,uint8_t ui8Priority);
/****** 函数介绍 ******/
/* 作用 : 设置中断的优先级
* 传入参数 :
* ui32Interrupt :指定所涉及的中断。在hw_ints.h中定义
* ui8Priority : 中断优先级,当同时断言多个中断时,具有最高优先级的中断将在较低优先级的中断之前被处理。xxx0 0000,只看 xxx 前三位。比如0的优先级比0xE0高。
* 返回参数 : 无
*/
printf重映射
(1)因为很多人喜欢使用printf重映射打印数据。所以这里我也进行简单的讲解。但是printf重映射只能映射一个串口,不过我下面还有一个方案,可以让所有的串口都能够printf重映射。
(2)我们只需要将x改动一下即可。同时需要打开MicroLIB。
//重新映射printf函数到UARTx,x可为1,2,3,4,5,6,7
int fputc(int ch, FILE *f){UARTCharPut(UARTx_BASE,ch); return (ch);}
int fgetc(FILE *f) {int ch=UARTCharGet(UARTx_BASE); return (ch);}
printf("* 作者 : CSND 风正豪 \r\n");
多个串口同时重映射
直接复制如下的函数,就可以直接将多个串口初始
/* 作用 : 用于多路串口重定义
* 传入参数 :
baseAddress : 要打印的串口地址,UARTx_BASE,x可为0,1,2,3,4,5,6,7
format : 需要打印的东西
... : 如果是打印字符,输入%c。有符号数字,%d。用法与printf一样
* 返回值 : 无
*/
void UART_printf(uint32_t baseAddress, const char *format,...)
{
uint32_t length;
va_list args;
uint32_t i;
char TxBuffer[128] = {0};
va_start(args, format);
length = vsnprintf((char*)TxBuffer, sizeof(TxBuffer), (char*)format, args);
va_end(args);
for(i = 0; i < length; i++)
{
while(UARTBusy(baseAddress));
UARTCharPut(baseAddress,TxBuffer[i]);
}
}
实操
#include "stdio.h"
#include <stdint.h>
#include <stdbool.h>
#include <string.h>
#include <stdarg.h>
#include "hw_memmap.h"
#include "hw_types.h"
#include "hw_gpio.h"
#include "hw_ints.h"
#include "debug.h"
#include "fpu.h"
#include "gpio.h"
#include "pin_map.h"
#include "rom.h"
#include "sysctl.h"
#include "uart.h"
#include "uartstdio.h"
#ifdef DEBUG
void
__error__(char *pcFilename, uint32_t ui32Line)
{
}
#endif
/*
*********************************************************************************************************
* 函 数 名: PrintfLogo
* 功能说明: 打印例程名称和例程发布日期, 接上串口线后,打开PC机的串口终端软件可以观察结果
* 形 参: 无
* 返 回 值: 无
*********************************************************************************************************
*/
void PrintfLogo(void)
{
printf("*************************************************************\n\r");
printf("* %s\r\n", "串口0控制RGB例程"); /* 打印例程名称 */
printf("* 发布日期 : %s\r\n", "2023年4月24日"); /* 打印例程日期 */
printf("* 使用芯片 : TM4C123GH6PZI7\r\n");
printf("* 作者 : CSND 风正豪 \r\n");
printf("*************************************************************\n\r");
}
/* 作用 : 用于多路串口重定义
* 传入参数 :
baseAddress : 要打印的串口地址,UARTx_BASE,x可为0,1,2,3,4,5,6,7
format : 需要打印的东西
... : 如果是打印字符,输入%c。有符号数字,%d。用法与printf一样
* 返回值 : 无
*/
void UART_printf(uint32_t baseAddress, const char *format,...)
{
uint32_t length;
va_list args;
uint32_t i;
char TxBuffer[128] = {0};
va_start(args, format);
length = vsnprintf((char*)TxBuffer, sizeof(TxBuffer), (char*)format, args);
va_end(args);
for(i = 0; i < length; i++)
{
while(UARTBusy(baseAddress));
UARTCharPut(baseAddress,TxBuffer[i]);
}
}
//串口0的接收中断
static uint8_t ch;
void UART0_IRQHandler(void)
{
//获取中断标志 原始中断状态 屏蔽中断标志
uint32_t flag = UARTIntStatus(UART0_BASE,1);
//清除中断标志
UARTIntClear(UART0_BASE,flag);
//判断FIFO是否还有数据
while(UARTCharsAvail(UART0_BASE))
{
//接收数据
ch=UARTCharGet(UART0_BASE);
}
}
void ConfigureUART0(void)
{
//使能UART使用的GPIO外设
ROM_SysCtlPeripheralEnable(SYSCTL_PERIPH_GPIOA);
//使能 UART0
ROM_SysCtlPeripheralEnable(SYSCTL_PERIPH_UART0);
//配置UART模式的GPIO引脚
ROM_GPIOPinConfigure(GPIO_PA0_U0RX);
ROM_GPIOPinConfigure(GPIO_PA1_U0TX);
ROM_GPIOPinTypeUART(GPIO_PORTA_BASE, GPIO_PIN_0 | GPIO_PIN_1);
//设置串口0,时钟源速率为系统时钟,波特率为9600,8数据位,1停止位,无校验位
UARTConfigSetExpClk(UART0_BASE, SysCtlClockGet(), 9600,(UART_CONFIG_WLEN_8 | UART_CONFIG_STOP_ONE |UART_CONFIG_PAR_NONE));
//禁用UART模块的FIFO缓冲区。减少串口接收的延迟,提高响应速度。
UARTFIFODisable(UART0_BASE);
//使能UART0接收中断
UARTIntEnable(UART0_BASE,UART_INT_RX);
//UART0中断函数注册
UARTIntRegister(UART0_BASE,UART0_IRQHandler);
//设置中断优先级
ROM_IntPrioritySet(INT_UART0,0x60);
}
//重新映射printf函数到UART0
int fputc(int ch, FILE *f){UARTCharPut(UART0_BASE,ch); return (ch);}
int fgetc(FILE *f) {int ch=UARTCharGet(UART0_BASE); return (ch);}
void RGB_Init(void)
{
ROM_SysCtlPeripheralEnable(SYSCTL_PERIPH_GPIOF); //使能GPIOF外设
ROM_GPIOPinTypeGPIOOutput(GPIO_PORTF_BASE, GPIO_PIN_5);//红色
ROM_GPIOPinTypeGPIOOutput(GPIO_PORTF_BASE, GPIO_PIN_6);//绿色
ROM_GPIOPinTypeGPIOOutput(GPIO_PORTF_BASE, GPIO_PIN_4);//蓝色
ROM_GPIOPinWrite(GPIO_PORTF_BASE, GPIO_PIN_4, GPIO_PIN_4);//置高位熄灭
ROM_GPIOPinWrite(GPIO_PORTF_BASE, GPIO_PIN_5, GPIO_PIN_5);//置高位熄灭
ROM_GPIOPinWrite(GPIO_PORTF_BASE, GPIO_PIN_6, GPIO_PIN_6);//置高位熄灭
}
int main(void)
{
ROM_FPUEnable();//使能浮点单元。这个函数必须在执行任何硬件浮点运算之前被调用;如果不这样做,将导致NOCP使用错误。
ROM_FPULazyStackingEnable();//浮点延迟堆栈,减少中断响应延迟
ROM_SysCtlClockSet(SYSCTL_SYSDIV_2_5 | SYSCTL_USE_PLL | SYSCTL_XTAL_16MHZ |SYSCTL_OSC_MAIN);//配置系统时钟,系统时钟频率400M/2/2.5=80M
RGB_Init();
ConfigureUART0();//初始化串口0
PrintfLogo();//串口打印版本信息
while(1)
{
if((int8_t)ch == 'a')
{
UARTprintf("Turn on the LED1\n"); //无法使用
UART_printf(UART0_BASE,"%s\n","hello world");
GPIOPinWrite(GPIO_PORTF_BASE, GPIO_PIN_5, !GPIO_PIN_5);//置低位点亮
SysCtlDelay(SysCtlClockGet() / 10);
GPIOPinWrite(GPIO_PORTF_BASE, GPIO_PIN_5, GPIO_PIN_5);//置高位熄灭
SysCtlDelay(SysCtlClockGet() / 10);
}
if((int8_t)ch == 'b')
{
UART_printf(UART0_BASE,"2+2=%d\n",4);
GPIOPinWrite(GPIO_PORTF_BASE, GPIO_PIN_6, !GPIO_PIN_6);//置低位点亮
SysCtlDelay(SysCtlClockGet() / 10);
GPIOPinWrite(GPIO_PORTF_BASE, GPIO_PIN_6, GPIO_PIN_6);//置高位熄灭
SysCtlDelay(SysCtlClockGet() / 10);
}
if((int8_t)ch == 'c')
{
UART_printf(UART0_BASE,"2+2.0=%f\n",4.0);
GPIOPinWrite(GPIO_PORTF_BASE, GPIO_PIN_4, !GPIO_PIN_4);//置低位点亮
SysCtlDelay(SysCtlClockGet() / 10);
GPIOPinWrite(GPIO_PORTF_BASE, GPIO_PIN_4, GPIO_PIN_4);//置高位熄灭
SysCtlDelay(SysCtlClockGet() / 10);
}
}
}