STM32学习笔记——十、六USART——串口通讯
- 0 前言
- 1 串口通讯协议简介
-
- 1.1 物理层
-
- 1.1.1 电平标准
- 1.1.2 RS-232信号线
- 1.2 协议层
-
- 1.2.1 波特率
- 1.2.2 通讯的起始和停止信号
- 1.2.3 有效数据
- 1.2.4 数据校验
- 2 STM32的USART简介
- 3 USART功能框图
-
- 3.1.1 功能引脚
- 3.1.2 数据寄存器
- 3.1.3 控制器
- 3.1.4 发送器
- 3.1.5 接收器
- 3.1.6 小数波特率生成
- 3.1 校验控制
- 3.2 中断控制
- 4 USART初始化结构体详解
-
- 4.1 USART 初始化结构体
- 4.2 USART时钟初始化结构体
- 5 USART1接发通信实验
-
- 5.1 硬件设计
- 5.2 软件设计
-
- 5.2.1 GPIO和USART 宏定义
- 5.2.2 嵌套向量中断控制器NVIC配置
- 5.2.3 USART初始化配置
- 5.2.4 字符发送
- 5.2.5 USART 中断服务函数
- 5.2.6 主函数
- 5.3 下载验证
- 7 补充说明
0 前言
本章参考资料:
- 《STM32F10X-中文参考手册》USART 章节
- 配合《STM32F10X-中文参考手册》USART 章节一起阅读
- 以STM32F103 系列控制器资源讲解
1 串口通讯协议简介
- 串口通讯(Serial Communication) :一种设备间非常常用的串行通讯方式
- 在计算机科学里,大部分复杂的问题都可以通过分层来简化。
- 对于通讯协议,我们也以分层的方式来理解:
- 最基本的是把它分为物理层和协议层。
- 物理层:规定通讯系统中具有机械、电子功能部分的特性,确保原始数据在物理媒体的传输。
- 协议层:主要规定通讯逻辑,统一收发双方的数据打包、解包标准。
- 简单来说物理层规定我们用嘴巴还是用肢体来交流,协议层则规定我们用中文还是英文来交流。
1.1 物理层
- 串口通讯的物理层有很多标准及变种,我们主要讲解RS-232 标准。
- RS-232 标准:主要规定了信号的用途、通讯接口以及信号的电平标准。
- 两个通讯设备的“DB9 接口”之间通过串口信号线建立起连接,串口信号线中使用“RS-232 标准”传输数据信号。
- 由于RS-232 电平标准的信号不能直接被控制器直接识别,所以这些信号会经过一个“电平转换芯片”转换成控制器能识别的“TTL 标准”的电平信号,才能实现通讯。
1.1.1 电平标准
根据通讯使用的电平标准不同,串口通讯可分为TTL 标准及RS-232 标准
- 常见的电子电路中常使用TTL 的电平标准,理想状态下,使用5V 表示二进制逻辑1,使用0V 表示逻辑0。
- 使用RS232电平标准为了增加串口通讯的远距离传输及抗干扰能力,它使用-15V 表示逻辑1,+15V 表示逻辑0。
1.1.2 RS-232信号线
- 在最初的应用中,RS-232 串口标准常用于计算机、路由与调制调解器(MODEN,俗称“猫”) 之间的通讯
- 在这种通讯系统中,设备被分为数据终端设备DTE(计算机、路由) 和数据通讯设备DCE(调制调解器)。
- 在旧式的台式计算机中一般会有RS-232 标准的COM 口(也称DB9 接口),电脑主板上的COM 口及串口线。
- 其中接线口以针式引出信号线的称为公头,以孔式引出信号线的称为母头。
- 在计算机中一般引出公头接口,而在调制调解器设备中引出的一般为母头,使用上图中的串口线即可把它与计算机连接起来。
- 通讯时,串口线中传输的信号就是使用前面讲解的RS-232 标准调制的。
- 在这种应用场合下,DB9 接口中的公头及母头的各个引脚的标准信号线接法见图DB9 标准的公头及母头接法及表DB9 信号线说明。
- 计算机端的DB9公头标准接发
- 由于两个通讯设备之间的收发信号(RXD 与TXD)应交叉相连,所以调制调解器端的DB9 母头的收发信号接法一般与公头的相反,两个设备之间连接时,只要使用“直通型”的串口线连接起来即
- 串口线中的RTS、CTS、DSR、DTR 及DCD 信号,使用逻辑1 表示信号有效,逻辑0 表示信号无效。
- 例如,当计算机端控制DTR 信号线表示为逻辑1 时,它是为了告知远端的调制调解器,本机已准备好接收数据,0 则表示还没准备就绪。
- 在目前的其它工业控制使用的串口通讯中,一般只使用RXD、TXD 以及GND 三条信号线,直接传输数据信号,而RTS、CTS、DSR、DTR 及DCD 信号都被裁剪掉了。
1.2 协议层
串口通讯的数据包由发送设备通过自身的TXD 接口传输到接收设备的RXD 接口。
- 在串口通讯的协议层中,规定了数据包的内容,它由启始位、主体数据、校验位以及停止位组成,通讯双方的数据包格式要约定一致才能正常收发数据,其组成见图串口数据包的基本组成。
1.2.1 波特率
以串口异步通讯为例:
- 异步通讯中由于没有时钟信号(如前面讲解的DB9 接口中是没有时钟信号的),所以两个通讯设备之间需要约定好波特率,即每个码元的长度,以便对信号进行解码。
- 上图串口数据包的基本组成中用虚线分开的每一格就是代表一个码元。常见的波特率为4800、9600、115200 等。
1.2.2 通讯的起始和停止信号
串口通讯的一个数据包从起始信号开始,直到停止信号结束。
- 数据包的起始信号由一个逻辑0 的数据位表示
- 数据包的停止信号可由0.5、1、1.5 或2 个逻辑1 的数据位表示
- 只要双方约定一致即可
1.2.3 有效数据
在数据包的起始位之后紧接着的就是要传输的主体数据内容,也称为有效数据
1.2.4 数据校验
在有效数据之后,有一个可选的数据校验位
- 由于数据通信相对更容易受到外部干扰导致传输数据出现偏差,可以在传输过程加上校验位来解决这个问题
- 校验方法:
- 奇校验(odd)
- 偶校验(even)
- 0 校验(space)
- 1 校验(mark)
- 无校验(noparity)
- 奇校验要求有效数据和校验位中“1”的个数为奇数,比如一个8 位长的有效数据为:01101001,此时总共有4 个“1”,为达到奇校验效果,校验位为“1”,最后传输的数据将是8 位的有效数据加上1 位的校验位总共9 位。
- 偶校验与奇校验要求刚好相反,要求帧数据和校验位中“1”的个数为偶数,比如数据帧:11001010,此时数据帧“1”的个数为4 个,所以偶校验位为“0”。
- 0 校验是不管有效数据中的内容是什么,校验位总为“0”,1 校验是校验位总为“1”。
2 STM32的USART简介
通用同步异步收发器(Universal Synchronous Asynchronous Receiver and Transmitter) 是一个串行通信设备,可以灵活地与外部设备进行全双工数据交换。
- 还有一个UART(Universal Asynchronous Receiver and Transmitter):
- 在USART 基础上裁剪掉了同步通信功能,只有异步通信。
- 简单区分同步和异步就是看通信时需不需要对外提供时钟输出,我们平时用的串口通信基本都是UART。
- 串行通信一般是以帧格式传输数据,即是一帧一帧的传输,每帧包含有起始信号、数据信息、停止信息,可能还有校验信息。
- USART 就是对这些传输参数有具体规定,当然也不是只有唯一 一个参数值,很多参数值都可以自定义设置,只是增强它的兼容性。
- USART 满足外部设备对工业标准NRZ 异步串行数据格式的要求,并且使用了小数波特率发生器,可以提供多种波特率,使得它的应用更加广泛。
- USART 支持同步单向通信和半双工单线通信;还支持局域互连网络LIN、智能卡(SmartCard) 协议与lrDA(红外线数据协会) SIR ENDEC 规范。
- USART 支持使用DMA,可实现高速数据通信。
- USART 在STM32 应用最多莫过于**“打印”程序信息**,一般在硬件设计时都会预留一个USART通信接口连接电脑,用于在调试程序是可以把一些调试信息“打印”在电脑端的串口调试助手工具上,从而了解程序运行是否正确、如果出错哪具体哪里出错等等。
3 USART功能框图
USART的功能框图:包含了USART的最核心内容
3.1.1 功能引脚
- TX:发送数据输出引脚。
- RX:接收数据输入引脚。
- SW_RX:数据接收引脚,只用于单线和智能卡模式,属于内部引脚,没有具体外部引脚。
- nRTS:请求以发送(Request To Send),n 表示低电平有效。如果使能RTS 流控制,当USART 接收器准备好接收新数据时就会将nRTS 变成低电平;当接收寄存器已满时,nRTS 将被设置为高电平。该引脚只适用于硬件流控制。
- nCTS:清除以发送(Clear To Send),n 表示低电平有效。如果使能CTS 流控制,发送器在发送下一帧数据之前会检测nCTS 引脚,如果为低电平,表示可以发送数据,如果为高电平则在发送完当前数据帧之后停止发送。该引脚只适用于硬件流控制。
- SCLK:发送器时钟输出引脚。这个引脚仅适用于同步模式。
- USART 引脚在STM32F103ZET6 芯片具体分布见表STM32F103ZET6 芯片的USART 引脚。
- STM32F103VET6 系统控制器有三个USART 和两个UART。
- 其中USART1 和时钟来源于APB2 总线时钟,其最大频率为72MHz
- 其他四个的时钟来源于APB1 总线时钟,其最大频率为36MHz。
- UART 只是异步传输功能,所以没有SCLK、nCTS 和nRTS 功能引脚。
3.1.2 数据寄存器
- USART 数据寄存器(USART_DR) 只有低9 位有效,并且第9 位数据是否有效要取决于USART控制寄存器1(USART_CR1) 的M 位设置,当M 位为0 时表示8 位数据字长,当M 位为1 表示9位数据字长,我们一般使用8 位数据字长。
- USART_DR 包含了已发送的数据或者接收到的数据。USART_DR 实际是包含了两个寄存器:
- 一个专门用于发送的可写TDR,当进行发送操作时,往USART_DR写入数据会自动存储在TDR 内
- 一个专门用于接收的可读RDR,当进行读取操作时,向USART_DR 读取数据会自动提取RDR数据
- TDR 和RDR 都是介于系统总线和移位寄存器之间
- 串行通信是一个位一个位传输的
- 发送时把TDR 内容转移到发送移位寄存器,然后把移位寄存器数据每一位发送出去
- 接收时把接收到的每一位顺序保存在接收移位寄存器内然后才转移到RDR
- USART 支持DMA 传输,可以实现高速数据传输
3.1.3 控制器
- USART 有专门控制发送的发送器、控制接收的接收器,还有唤醒单元、中断控制等等。
- 使用USART 之前需要向USART_CR1 寄存器的UE 位置1 使能USART,UE 位用来开启供给给串口的时钟。
- 发送或者接收数据字长可选8 位或9 位,由USART_CR1 的M 位控制。
3.1.4 发送器
- 当USART_CR1 寄存器的发送使能位TE 置1 时,启动数据发送:
- 发送移位寄存器的数据会在TX引脚输出,低位在前,高位在后
- 如果是同步模式SCLK 也输出时钟信号。
- 一个字符帧发送需要三个部分:起始位+ 数据帧+ 停止位
- 起始位是一个位周期的低电平,位周期就是每一位占用的时间
- 数据帧就是我们要发送的8 位或9 位数据,数据是从最低位开始传输的
- 停止位是一定时间周期的高电平
- 停止位时间长短是可以通过 USART 控制寄存器2(USART_CR2) 的STOP[1:0] 位控制,可选0.5个、1 个、1.5 个和2 个停止位。默认使用1 个停止位。
- 2 个停止位适用于正常USART 模式、单线模式和调制解调器模式。0.5 个和1.5 个停止位用于智能卡模式。
- 当选择8 位字长,使用1 个停止位时,具体发送字符时序图见图字符发送时序图。
- 当发送使能位TE 置1 之后:
- 发送器开始会先发送一个空闲帧(一个数据帧长度的高电平)
- 接下来就可以往USART_DR 寄存器写入要发送的数据
- 在写入最后一个数据后,需要等待USART 状态寄存器(USART_SR) 的TC 位为1,表示数据传输完成
- 如果USART_CR1 寄存器的TCIE 位置1,将产生中断。
- 在发送数据时,编程的时候有几个比较重要的标志位我们来总结下
3.1.5 接收器
- 如果将USART_CR1 寄存器的RE 位置1,使能USART 接收,使得接收器在RX 线开始搜索起始位。
- 在确定到起始位后就根据 RX 线电平状态把数据存放在接收移位寄存器内。
- 接收完成后就把接收移位寄存器数据移到RDR 内,并把USART_SR 寄存器的RXNE 位置1
- 同时如果USART_CR2 寄存器的RXNEIE 置1 的话可以产生中断。
- 在接收数据时,编程的时候有几个比较重要的标志位我们来总结下
3.1.6 小数波特率生成
波特率指数据信号对载波的调制速率,它用单位时间内载波调制状态改变次数来表示,单位为波特。
比特率指单位时间内传输的比特数,单位bit/s(bps)。
- 对于USART 波特率与比特率相等,以后不区分这两个概念。
- 波特率越大,传输速率越快。
- USART 的发送器和接收器使用相同的波特率。计算公式如下
- 其中,fPLCK 为USART 时钟,USARTDIV 是一个存放在波特率寄存器(USART_BRR) 的一个无符号定点数。
- 其中DIV_Mantissa[11:0] 位定义USARTDIV 的整数部分,DIV_Fraction[3:0] 位定义USARTDIV 的小数部分。
- 例如:DIV_Mantissa=24(0x18),DIV_Fraction=10(0x0A),此时USART_BRR 值为0x18A;那么USARTDIV
的小数位10/16=0.625;整数位24,最终USARTDIV 的值为24.625。
- 如果知道USARTDIV 值为27.68,那么DIV_Fraction=16*0.68=10.88,最接近的正整数为11,所以DIV_Fraction[3:0] 为0xB;DIV_Mantissa= 整数(27.68)=27,即为0x1B。
- 波特率的常用值有2400、9600、19200、115200。
- 下面以实例讲解如何设定寄存器值得到波特率的值:
- USART1 使用APB2 总线时钟,最高可达72MHz,其他USART 的最高频率为36MHz。
- 选取 USART1 作为实例讲解,即fPLCK=72MHz。为得到115200bps 的波特率,此时:
- 解得USARTDIV=39.0625,可算得DIV_Fraction=0.0625*16=1=0x01,DIV_Mantissa=39=0x27,即
应该设置USART_BRR 的值为0x271。
3.1 校验控制
- STM32F103 系列控制器USART 支持奇偶校验
- 当使用校验位时,串口传输的长度将是8 位的数据帧加上1 位的校验位总共9 位,此时USART_CR1 寄存器的M 位需要设置为1,即9 数据位。
- 将USART_CR1 寄存器的PCE 位置1 就可以启动奇偶校验控制,奇偶校验由硬件自动完成
- 启动了奇偶校验控制之后:
- 在发送数据帧时会自动添加校验位,接收数据时自动验证校验位
- 接收数据时如果出现奇偶校验位验证失败,会见USART_SR 寄存器的PE 位置1,并可以产生奇偶校验中断。
- 使能了奇偶校验控制后,每个字符帧的格式将变成:起始位+ 数据帧+ 校验位+ 停止位。
3.2 中断控制
USART 有多个中断请求事件,具体见表USART 中断请求。
4 USART初始化结构体详解
标准库函数对每个外设都建立了一个初始化结构体,比如USART_InitTypeDef,结构体成员用于设置外设工作参数,并由外设初始化配置函数,比如USART_Init() 调用,这些设定参数将会设置外设相应的寄存器,达到配置外设工作环境的目的。
- 初始化结构体和初始化库函数配合使用是标准库精髓所在,理解了初始化结构体每个成员意义基本上就可以对该外设运用自如了。
- 初始化结构体定义在stm32f10x_usart.h 文件中
- 初始化库函数定义在stm32f10x_usart.c 文件中
- 编程时我们可以结合这两个文件内注释使用。
4.1 USART 初始化结构体
typedef struct {
uint32_t USART_BaudRate;
uint16_t USART_WordLength;
uint16_t USART_StopBits;
uint16_t USART_Parity;
uint16_t USART_Mode;
uint16_t USART_HardwareFlowControl;
} USART_InitTypeDef;
-
- USART_BaudRate:波特率设置。一般设置为2400、9600、19200、115200。标准库函数会根据设定值计算得到USARTDIV 值,从而设置USART_BRR 寄存器值。
-
- USART_WordLength:数据帧字长,可选8 位或9 位。它设定USART_CR1 寄存器的M 位的值。如果没有使能奇偶校验控制,一般使用8 数据位;如果使能了奇偶校验则一般设置为9 数据位。
-
- USART_StopBits:停止位设置,可选0.5 个、1 个、1.5 个和2 个停止位,它设定USART_CR2寄存器的STOP[1:0] 位的值,一般我们选择1 个停止位。
-
- USART_Parity:奇偶校验控制选择,可选USART_Parity_No(无校验)、USART_Parity_Even(偶校验) 以及USART_Parity_Odd(奇校验),它设定USART_CR1 寄存器的PCE 位和PS 位的值。
-
- USART_Mode:USART 模式选择,有USART_Mode_Rx 和USART_Mode_Tx,允许使用逻辑或运算选择两个,它设定USART_CR1 寄存器的RE 位和TE 位。
-
- USART_HardwareFlowControl:硬件流控制选择,只有在硬件流控制模式才有效,可选有使能RTS、使能CTS、同时使能RTS 和CTS、不使能硬件流。
4.2 USART时钟初始化结构体
当使用同步模式时需要配置SCLK 引脚输出脉冲的属性,标准库使用一个时钟初始化结构体USART_ClockInitTypeDef 来设置,该结构体内容也只有在同步模式才需要设置。
typedef struct {
uint16_t USART_Clock;
uint16_t USART_CPOL;
uint16_t USART_CPHA;
uint16_t USART_LastBit;
} USART_ClockInitTypeDef;
-
- USART_Clock:同步模式下SCLK 引脚上时钟输出使能控制,可选禁止时钟输出(USART_Clock_Disable) 或开启时钟输出(USART_Clock_Enable);如果使用同步模式发送,一般都需要开启时钟。它设定USART_CR2 寄存器的CLKEN 位的值。
-
- USART_CPOL:同步模式下SCLK 引脚上输出时钟极性设置,可设置在空闲时SCLK 引脚为低电平(USART_CPOL_Low) 或高电平(USART_CPOL_High)。它设定USART_CR2 寄存器的CPOL位的值。
-
- USART_CPHA:同步模式下SCLK 引脚上输出时钟相位设置,可设置在时钟第一个变化沿捕获数据(USART_CPHA_1Edge) 或在时钟第二个变化沿捕获数据。它设定USART_CR2 寄存器的CPHA 位的值。USART_CPHA 与USART_CPOL 配合使用可以获得多种模式时钟关系。
-
- USART_LastBit:选择在发送最后一个数据位的时候时钟脉冲是否在SCLK 引脚输出,可以是不输出脉冲(USART_LastBit_Disable)、输出脉冲(USART_LastBit_Enable)。它设定USART_CR2 寄存器的LBCL 位的值。
5 USART1接发通信实验
- USART 只需两根信号线即可完成双向通信,对硬件要求低,使得很多模块都预留USART 接口来实现与其他模块或者控制器进行数据传输,比如GSM 模块,WIFI 模块、蓝牙模块等等。在硬件设计时,注意还需要一根“共地线”。
- 经常使用USART 来实现控制器与电脑之间的数据传输,通过USART 发送到串口调试助手,这样我们可以非常清楚程序的运行状态
- 可以将数据发送到串口调试助手,还可以在串口调试助手发送数据给控制器
- 编写一个程序实现开发板与电脑通信,在开发板上电时通过USART 发送一串字符串给电脑,然后开发板进入中断接收等待状态,如果电脑有发送数据过来,开发板就会产生中断,我们在中断服务函数接收数据,并马上把数据返回发送给电脑。
5.1 硬件设计
- 利用USART 实现开发板与电脑通信,需要用到一个USB 转USART 的IC
- 选择CH340G芯片来实现这个功能,CH340G 是一个USB 总线的转接芯片,实现USB 转USART、USB 转lrDA红外或者USB 转打印机接口,我们使用其USB 转USART 功能
- 将CH340G 的TXD 引脚与USART1 的RX 引脚连接,CH340G 的RXD 引脚与USART1 的
TX 引脚连接
- CH340G 芯片集成在开发板上,其地线(GND) 已与控制器的GND 连通
5.2 软件设计
- 创建了两个文件:bsp_usart.c 和 bsp _usart.h 文件用来存放USART 驱动程序及相关宏定义。
- 编程要点:
-
- 使能RX 和TX 引脚GPIO 时钟和USART 时钟;
-
- 初始化GPIO,并将GPIO 复用到USART 上;
-
- 配置USART 参数;
-
- 配置中断控制器并使能USART 接收中断;
-
- 使能USART;
-
- 在USART 接收中断服务函数实现数据接收和发送。
- 代码分析
5.2.1 GPIO和USART 宏定义
1
4
5
6 #define DEBUG_USARTx USART1
7 #define DEBUG_USART_CLK RCC_APB2Periph_USART1
8 #define DEBUG_USART_APBxClkCmd RCC_APB2PeriphClockCmd
9 #define DEBUG_USART_BAUDRATE 115200
10
11
12 #define DEBUG_USART_GPIO_CLK (RCC_APB2Periph_GPIOA)
13 #define DEBUG_USART_GPIO_APBxClkCmd RCC_APB2PeriphClockCmd
14
15 #define DEBUG_USART_TX_GPIO_PORT GPIOA
16 #define DEBUG_USART_TX_GPIO_PIN GPIO_Pin_9
17 #define DEBUG_USART_RX_GPIO_PORT GPIOA
18 #define DEBUG_USART_RX_GPIO_PIN GPIO_Pin_10
19
20 #define DEBUG_USART_IRQ USART1_IRQn
21 #define DEBUG_USART_IRQHandler USART1_IRQHandler
- 使用宏定义方便程序移植和升级
- 开发板中的CH340G 的收发引脚默认通过跳帽连接到USART1
- 如果想使用其他串口,可以把CH340G 跟USART1 直接的连接跳帽拔掉,然后再把其他串口的IO 用杜邦线接到CH340G 的收发引脚即可
- 使用USART1,设定波特率为115200,选定USART 的GPIO 为PA9 和PA10
5.2.2 嵌套向量中断控制器NVIC配置
1 static void NVIC_Configuration(void)
2 {
3 NVIC_InitTypeDef NVIC_InitStructure;
4
5
6 NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
7
8
9 NVIC_InitStructure.NVIC_IRQChannel = DEBUG_USART_IRQ;
10
11 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
12
13 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
14
15 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
16
17 NVIC_Init(&NVIC_InitStructure);
18 }
- 在中断章节已对嵌套向量中断控制器的工作机制做了详细的讲解,这里我们就直接使用,配置USART 作为中断源,因为本实验没有使用其他中断,对优先级没什么具体要求。
5.2.3 USART初始化配置
1 void USART_Config(void)
2 {
3 GPIO_InitTypeDef GPIO_InitStructure;
4 USART_InitTypeDef USART_InitStructure;
5
6
7 DEBUG_USART_GPIO_APBxClkCmd(DEBUG_USART_GPIO_CLK, ENABLE);
8
9
10 DEBUG_USART_APBxClkCmd(DEBUG_USART_CLK, ENABLE);
11
12
13 GPIO_InitStructure.GPIO_Pin = DEBUG_USART_TX_GPIO_PIN;
14 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
15 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
16 GPIO_Init(DEBUG_USART_TX_GPIO_PORT, &GPIO_InitStructure);
17
18
19 GPIO_InitStructure.GPIO_Pin = DEBUG_USART_RX_GPIO_PIN;
20 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
21 GPIO_Init(DEBUG_USART_RX_GPIO_PORT, &GPIO_InitStructure);
22
23
24
25 USART_InitStructure.USART_BaudRate = DEBUG_USART_BAUDRATE;
26
27 USART_InitStructure.USART_WordLength = USART_WordLength_8b;
28
29 USART_InitStructure.USART_StopBits = USART_StopBits_1;
30
31 USART_InitStructure.USART_Parity = USART_Parity_No ;
32
33 USART_InitStructure.USART_HardwareFlowControl =
34 USART_HardwareFlowControl_None;
35
36 USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
37
38 USART_Init(DEBUG_USARTx, &USART_InitStructure);
39
40
41 NVIC_Configuration();
42
43
44 USART_ITConfig(DEBUG_USARTx, USART_IT_RXNE, ENABLE);
45
46
47 USART_Cmd(DEBUG_USARTx, ENABLE);
48 }
- 使用GPIO_InitTypeDef 和USART_InitTypeDef 结构体定义一个GPIO 初始化变量以及一个USART初始化变量,这两个结构体内容我们之前已经有详细讲解。
- 调用RCC_APB2PeriphClockCmd 函数开启GPIO 端口时钟,使用GPIO 之前必须开启对应端口的时钟。使用RCC_APB2PeriphClockCmd 函数开启USART 时钟。
- 使用GPIO 之前都需要初始化配置它,并且还要添加特殊设置,因为我们使用它作为外设的引脚,一般都有特殊功能。我们在初始化时需要把它的模式设置为复用功能。这里把串口的Tx 引脚配置为复用推挽输出,Rx 引脚为浮空输入,数据完全由外部输入决定。
- 接下来,我们配置USART1 通信参数为:波特率115200,字长为8,1 个停止位,没有校验位,不使用硬件流控制,收发一体工作模式,然后调用USART 初始化函数完成配置。
- 程序用到USART 接收中断,需要配置NVIC,这里调用NVIC_Configuration 函数完成配置。配置完NVIC 之后调用USART_ITConfig 函数使能USART 接收中断。
最后调用USART_Cmd 函数使能USART,这个函数最终配置的是USART_CR1 的UE 位,具体的作用是开启USART 的工作时钟,没有时钟那USART 这个外设自然就工作不了。
5.2.4 字符发送
1
2 void Usart_SendByte( USART_TypeDef * pUSARTx, uint8_t ch)
3 {
4
5 USART_SendData(pUSARTx,ch);
6
7
8 while (USART_GetFlagStatus(pUSARTx, USART_FLAG_TXE) == RESET);
9 }
10
11
12 void Usart_SendString( USART_TypeDef * pUSARTx, char *str)
13 {
14 unsigned int k=0;
15 do {
16 Usart_SendByte( pUSARTx, *(str + k) );
17 k++;
18 } while (*(str + k)!='\0');
19
20
21 while (USART_GetFlagStatus(pUSARTx,USART_FLAG_TC)==RESET) {
22 }
23 }
- Usart_SendByte 函数用来在指定USART 发送一个ASCLL 码值字符,它有两个形参,第一个为USART,第二个为待发送的字符。
- 它是通过调用库函数USART_SendData 来实现的,并且增加了等待发送完成功能。
- 通过使用USART_GetFlagStatus 函数来获取USART 事件标志来实现发送完成功能等待,它接收两个参数,一个是USART,一个是事件标志。
- 这里我们循环检测发送数据寄存器为空这个标志,当跳出while 循环时说明发送数据寄存器为空这个事实。
- Usart_SendString 函数用来发送一个字符串,它实际是调用Usart_SendByte 函数发送每个字符,直到遇到空字符才停止发送。最后使用循环检测发送完成的事件标志TC 来实现保证数据发送完成后才退出函数。
5.2.5 USART 中断服务函数
1 void DEBUG_USART_IRQHandler(void)
2 {
3 uint8_t ucTemp;
4 if (USART_GetITStatus(DEBUG_USARTx,USART_IT_RXNE)!=RESET) {
5 ucTemp = USART_ReceiveData( DEBUG_USARTx );
6 USART_SendData(DEBUG_USARTx,ucTemp);
7 }
8 }
- 存放在stm32f10x_it.c 文件中的,该文件用来集中存放外设中断服务函数。
- 当我们使能了中断并且中断发生时就会执行这里的中断服务函数。
- USART-3 使能了USART 接收中断,当USART 有接收到数据就会执行USART_IRQHandler 函数。
- USART_GetITStatus 函数与USART_GetFlagStatus 函数类似用来获取标志位状态,但USART_GetITStatus 函数是专门用来获取中断事件标志的,并返回该标志位状态。
- 使用if 语句来判断是否是真的产生USART 数据接收这个中断事件,如果是真的就使用USART数据读取函数USART_ReceiveData 读取数据到指定存储区。
- 然后再调用USART 数据发送函数USART_SendData 把数据又发送给源设备,即PC 端的串口调试助手。
5.2.6 主函数
1 int main(void)
2 {
3
4 USART_Config();
5
6 Usart_SendString( DEBUG_USARTx,"这是一个串口中断接收回显实验\n");
7
8 while (1) {
9
10 }
11 }
- 调用USART_Config 函数完成USART 初始化配置,包括GPIO 配置,USART 配置,接收中断使能等等信息。
- 接下来调用字符发送函数把数据发送给串口调试助手。
- 最后主函数什么都不做,静静地等待USART 接收中断的产生,并在中断服务函数把数据回传。
5.3 下载验证
- 保证开发板相关硬件连接正确,用USB 线连接开发板的USB 转串口跟电脑,在电脑端打开串口调试助手并配置好相关参数:115200 8-N-1,把编译好的程序下载到开发板,此时串口调试助手即可收到开发板发过来的数据。
- 我们在串口调试助手发送区域输入任意字符,点击发送按钮,马上在串口调试助手接收区即可看到相同的字符。
7 补充说明
对于STM32 芯片:
- 各个引脚可以做什么外设功能或者电气特性参数在数据手册查阅
- 引脚和外设的功能特性和对应寄存器的使用在参考手册查阅。
- 以下对于F103 芯片,当根据数据手册确定要使用某个引脚做某个外设功能时,在参考手册如下查阅外设的GPIO 应该如何配置,当对各种配置不了解时以能在手册查阅出的结果为准。
- 关于串口中断的使用,以串口1 举例,在参考手册、库内定义和启动文件等可以看出串口1 使用USART1_IRQn 一个中断源,使用USART1_IRQHandler 一个中断函数,回顾EXIT 章节的补充说明有类似意思,对于串口1 这里,如果使能了多个标志位的中断,它们会在各自的时序流程触发中断,因此在中断函数里面用多个GetITStatus 函数判断标志并来做此标志的流程操作,查阅手册中的寄存器描述了解各标志位的时序和如何清除标志位等,某些标志位优先以手册中有描述的软件清除流程而可以不使用ClearFlag 函数。
- 上面例程只使用了接收非空中断,可以参考补充例程内的接收非空中断加空闲中断的使用,除此之外还有其他也能够使能中断的标志位,但不需要都用到,当加更多标志位中断时会让中断流程和主流程的逻辑配合变更复杂。
- 以上补充总结对之后的外设学习也是同样道理,注意对于GPIO 配置和外设在不同系列芯片会存在差异,重点在于掌握整体概念,学会在不同芯片的手册中查阅对应内容。