当前位置:首页 > 单片机 > 单片机
[导读]串口设置的一般步骤可以总结为如下几个:1) 串口时钟使能, GPIO时钟使能2) 串口复位3)GPIO 端口模式设置4) 串口参数初始化5) 开启中断并且初始化 NVIC(如果需要开启中断才这个步骤) (如果需要开启中断才这个步骤

串口设置的一般步骤可以总结为如下几个:
1) 串口时钟使能, GPIO时钟使能
2) 串口复位
3)GPIO 端口模式设置
4) 串口参数初始化
5) 开启中断并且初始化 NVIC(如果需要开启中断才这个步骤) (如果需要开启中断才这个步骤)
6) 使能串口 使能串口

7) 编写中断处理函数


下面,我们就简单介绍这几个与串口基本配置直接相关的固件库函数。这些函数和 定义主要分布在 stm32f10x_usart.h ,stm32f10x_usart.c 文件中。

1.串口时钟使能。串口是挂载在APB2上的,所以使能函数为:

RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1)

2.串口复位。当外设出现异常的时候可以通过复位置 ,实现该外设的复位,然后重新配置这个外设让其重新工作的目。一般在系统刚开始配置时候,都会先执行复位该这个外 设达到让其重新工作的目。复位是在函数 USART_DeInit()完成:

void USART_DeInit(USART_TypeDef* USARTx)

3串口参数初始化


void USART_Init()函数:



voidUSART_Init(USART_TypeDef*USARTx,USART_InitTypeDef*USART_InitStruct);

作用:

根据指定参数初始化相应串口(波特率,字长,停止位,奇偶校验,硬件流控制等)

主要是用来初始化寄存器BRR以及CR1,CR2,CR3控制寄存器

使用范例:

USART_InitTypeDefUSART_InitStructure;

USART_InitStructure.USART_BaudRate= 9600;//波特率设置;

USART_InitStructure.USART_WordLength= USART_WordLength_8b;//字长为8位数据格式

USART_InitStructure.USART_StopBits= USART_StopBits_1;//一个停止位

USART_InitStructure.USART_Parity= USART_Parity_No;//无奇偶校验位

USART_InitStructure.USART_HardwareFlowControl= USART_HardwareFlowControl_None; //无硬件数据流控制

USART_InitStructure.USART_Mode= USART_Mode_Rx| USART_Mode_Tx; //收发模式

USART_Init(USART1,&USART_InitStructure);//初始化串口



4.void USART_Cmd()函数:


原型:

voidUSART_Cmd(USART_TypeDef*USARTx,FunctionalStateNewState);

作用:

使能相应的串口,用来设置寄存器CR1的串口使能位

使用范例:

USART_Cmd(USART1,ENABLE); //使能串口1


5.void USART_ITConfig()函数:

原型:voidUSART_ITConfig(USART_TypeDef*USARTx,

uint16_t USART_IT, FunctionalStateNewState);

作用:开启串口相应中断,设置串口控制寄存器

使用范例:

USART_ITConfig(USART1,USART_IT_RXNE, ENABLE); //开启读数据寄存器非空中断


6.USART_SendData()函数:

原型:


voidUSART_SendData(USART_TypeDef* USARTx, uint16_t Data);

作用:

发送数据到串口。

使用范例:

USART_SendData(USART1,0x12);

7.uint16_tUSART_ReceiveData()函数:

原型: uint16_t USART_ReceiveData(USART_TypeDef*USARTx)

获取串口最新接受的值。

使用范例:

USART_ReceiveData(USART1);

8.四个状态标志相关的函数:


FlagStatusUSART_GetFlagStatus(USART_TypeDef*USARTx,uint16_t USART_FLAG);

void USART_ClearFlag(USART_TypeDef*USARTx,uint16_t USART_FLAG);


ITStatusUSART_GetITStatus(USART_TypeDef*USARTx,uint16_t USART_IT);


void USART_ClearITPendingBit(USART_TypeDef*USARTx,uint16_t USART_IT);


以下是一个完整的初始化串口函数和一个中断服务函数:

//初始化IO 串口1
//bound:波特率
void uart_init(u32 bound){
//GPIO端口设置
GPIO_InitTypeDef GPIO_InitStructure;
USART_InitTypeDef USART_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;

RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE);//使能USART1,GPIOA时钟
USART_DeInit(USART1); //复位串口1
//USART1_TX PA.9
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //PA.9
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;//复用推挽输出
GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化PA9

//USART1_RX PA.10
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入
GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化PA10


//Usart1 NVIC 配置


NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;//抢占优先级3
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;//子优先级3
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//IRQ通道使能
NVIC_Init(&NVIC_InitStructure);//根据指定的参数初始化VIC寄存器

//USART 初始化设置


USART_InitStructure.USART_BaudRate = bound;//一般设置为9600;
USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为8位数据格式
USART_InitStructure.USART_StopBits = USART_StopBits_1;//一个停止位
USART_InitStructure.USART_Parity = USART_Parity_No;//无奇偶校验位
USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//无硬件数据流控制
USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;//收发模式


USART_Init(USART1, &USART_InitStructure); //初始化串口
USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启中断
USART_Cmd(USART1, ENABLE); //使能串口


}



void USART1_IRQHandler(void) //串口1中断服务程序
{
u8 Res;
#ifdef OS_TICKS_PER_SEC //如果时钟节拍数定义了,说明要使用ucosII了.
OSIntEnter();
#endif
if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) //接收中断(接收到的数据必须是0x0d 0x0a结尾)
{
Res =USART_ReceiveData(USART1);//(USART1->DR);//读取接收到的数据

if((USART_RX_STA&0x8000)==0)//接收未完成
{
if(USART_RX_STA&0x4000)//接收到了0x0d
{
if(Res!=0x0a)USART_RX_STA=0;//接收错误,重新开始
else USART_RX_STA|=0x8000;//接收完成了
}
else //还没收到0X0D
{
if(Res==0x0d)USART_RX_STA|=0x4000;
else
{
USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
USART_RX_STA++;
if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0;//接收数据错误,重新开始接收
}
}
}
}

#ifdef OS_TICKS_PER_SEC//如果时钟节拍数定义了,说明要使用ucosII了.
OSIntExit();
#endif
}


本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

这篇文章想分享笔者在嵌入式开发过程中常用的库函数,他们的使用方法,使用场景,使用好处以及头文件位置。1#include一般我在写C代码的时候都会包含此头文件,因为一旦包含此文件后,你就可以畅快的使用bool数据类型,而不...

关键字: 库函数

关注、星标公众号,直达精彩内容来源:知乎作者:Clarence这篇文章想分享笔者在嵌入式开发过程中常用的库函数,他们的使用方法,使用场景,使用好处以及头文件位置。1#include一般我在写C代码的时候都会包含此头文件,...

关键字: 库函数

基本概念阐述memcpy和memmove都是C语言的库函数,相比于 strcpy和 strncpy只能针对于字符类型的数组(),这两个函数可以拷贝其他类型的数组,对于 memcpy和 memmove的区别是什么呢?这里,...

关键字: 库函数 ov

在嵌入式Linux的C语言开发中,C语言的基本编程依然是最重要的内容。

关键字: 嵌入式 C语言 库函数

单片机编程软件的使用频率极高,采用单片机编程软件,可制造诸多系统。对于单片机编程软件,小编做过诸多介绍。本文对于单片机编程软件的介绍基于Keil,主要在于介绍该单片机编程软件是如何处理库函数以及寄存器的关系的。

关键字: 单片机编程软件 库函数 指数

杂记asla-lib库函数snd_pcm_open打开流程浅析ac97声卡intel8x0的DMA内存substream->dma_buffer什么时候被赋值浅析ac97声卡intel8x0的r

关键字: playback 库函数

stm32有两个看门狗,独立看门狗和窗口看门狗,其实两者的功能是类似的,只是喂狗的限制时间不同。独立看门狗是限制喂狗时间在0-x内,x由你的相关寄存器决定。喂狗的时间不能过晚。窗口看门狗,所以称之为窗口就是因为

关键字: STM32 库函数 操作寄存器 窗口看门狗

首先,I2C总线由两条线——串行数据(SDA)和串行时钟(SCL),这是同步通信,也是半双工通信,不能同时读写。每个器件都有一个唯一的地址识别,当总线空闲时I2C两条线都是高电平,只有当连接到总线的器件的输出级是

关键字: i2c stm32f429 库函数 读取eeprom

stm32f103最少有2个AD模数转换器,每个ADC都有18个通道,可以测量16个外部和2个内部模拟量。最大转换频率为1Mhz,也就是转换时间为1us(在 ADCCLK = 14Mhz,采样周期为1.5个时钟周期时)。...

关键字: STM32 库函数 操作寄存器 ad模数转换

问题及现象(STM32F103系列:http://www.y-ec.com/cpcp/class/?32.html)使用USART_SendData()函数非连续发送单个字符是没有问题的;当连续发送字符时(两个字符间没有...

关键字: STM32 库函数
关闭
关闭