当前位置:首页 > 单片机 > 单片机
[导读]一。单片机通信的知识1. 通信的两种方式2. 串行通信的三种传输方式半双工数据发送和接收数据不能同时传输,全双工发送和接收互不影响,数据传输可以同时进行。3. 串行通信的通信方式对于同步通信,除了一条数据线以外

一。单片机通信的知识

1. 通信的两种方式


2. 串行通信的三种传输方式

半双工数据发送和接收数据不能同时传输,全双工发送和接收互不影响,数据传输可以同时进行。


3. 串行通信的通信方式

对于同步通信,除了一条数据线以外,还需要一条时钟线,用于传输同步时钟信号,数据的每个位传输都是随着时钟传输。

异步通信没有时钟信号,这就需要通信双方事先要进行约定,比如UART通信要事先约定好波特率,才能保证通信数据的正确。

4. 常见的串行通信接口

5. STM32的串口通信接口



6. UART异步通信方式引脚接线方法



PC机上的DB9接口电平为232电平,不能直接跟单片机上TTL信号连接,两者的电平不兼容。

7. UART异步通信方式的引脚

对于大容量的STM32最多有5个串口。

RXD: 数据输入引脚,接收数据。

TXD: 数据输出引脚,发送数据。

二。UART异步通信的特点








奇偶校验位:比如传输8位数据,其中有3个1,如果是偶校验,因为前面有3个1,就需要在奇偶校验位加一个1,使传输的1为偶数个,如果前面传输的数据有4个1,就在奇偶校验位补0。奇校验则相反,如果前面的数据中有四个1,就需要在奇校验位补一个1,是1成为奇数个,如果前面有奇数个1,就在奇偶校验位补0。


设置奇偶校验是为了提高数据传输的准确率。

三。UART框图


1. 理解数据传输和接收过程。

2. 发送和接收共用一个波特率发生器。

对于STM32F103R8T6只有3个USART,串口1,串口2,串口3.

串口1的时钟来源于PCLK2

串口2-串口4的时钟来源于PCLK1.

PCLKx进来后进入USARTDIV进行分频。分频值的大小由USART_BRR寄存器进行配置。

时钟可以进行整数的分频,还可以进行小数的分频,比如可以 /36,还可以 /36.5(前面只是举了一个例子,实际上小数必须是1/16的整数倍),分频后再除以16,产生的时钟进入发送器时钟或接收器时钟。



SR寄存器发送接收数据过程中的标志位。




CR1寄存器一部分为发送接收使能位,另外还有中断使能位,可以编写中断函数,在中断函数中判断是哪个中断发生了。


四。串口常用的寄存器

1. SR状态寄存器


RXNE:读数据寄存器非空,DR寄存器已经收到数据置1

TC:发送完成,数据发送完成后置1

2. DR寄存器


只用到了位0--位8

3. BRR寄存器

波特率寄存器


用到低16位,低4位USARTDIV的小数部分

位4-位15,USARTDIV的整数部分


PCLK1用于串口2,3,4,5,为36M

PCLK2用于串口1,为72M

4. CR1寄存器

控制寄存器,主要有发送和接收使能,以及相关的中断使能



RE:接收使能

TE:发送使能

RXNEIE:接收缓冲区非空中断使能,在接收到完整的数据后如果在这里使能了中断,就可以产生中断。

五。串口操作常用的相关库函数

六。 串口配置的一般步骤




中断的通道在stm32f10x.h中

USART1的中断通道为USART1_IRQn。

中断文件的格式在启动文件 startup_stm32f10x_hd.s 中。

七。 串口几个重要函数

1.void USART_Init(USART_TypeDef* USARTx, USART_InitTypeDef* USART_InitStruct); 串口初始化函数

作用:初始化串口的一些重要参数,包括波特率,奇偶校验位,停止位等

2.void USART_ITConfig(USART_TypeDef* USARTx, uint16_t USART_IT, FunctionalState NewState);

作用:开启相应的串口的中断。

3.void USART_Cmd(USART_TypeDef* USARTx, FunctionalState NewState);

作用:使能相应的串口

4.void USART_SendData(USART_TypeDef* USARTx, uint16_t Data);

作用:串口发送数据,往串口发送一个数据

5.uint16_t USART_ReceiveData(USART_TypeDef* USARTx);

作用:串口接收数据

6.void USART_DeInit(USART_TypeDef* USARTx);

作用:串口复位

7.void USART1_IRQHandler(void) //串口1中断服务函数

{

}

通过以上几个函数就可以实现数据的发送和接收

中断状态的获取,复位相关的一些函数

1. FlagStatus USART_GetFlagStatus(USART_TypeDef* USARTx, uint16_t USART_FLAG);

2. void USART_ClearFlag(USART_TypeDef* USARTx, uint16_t USART_FLAG);

3. ITStatus USART_GetITStatus(USART_TypeDef* USARTx, uint16_t USART_IT);

4. void USART_ClearITPendingBit(USART_TypeDef* USARTx, uint16_t USART_IT);

八。串口配置的一般步骤

1) 串口时钟使能,GPIO 时钟使能,需要使能GPIOA和USART1的时钟

2) 串口复位

3) GPIO 端口模式设置

4) 串口参数初始化

5) 开启中断并且初始化 NVIC(如果需要开启中断才需要这个步骤)

6) 使能串口

7) 编写中断处理函数

1. 配置RXD,TXD引脚

发送TXD(PA9)设置为推挽复用输出

接收RXD(PA10)设置为浮空或带上拉输入

在misc.c文件中有中断分组的配置函数

void NVIC_PriorityGroupConfig(uint32_t NVIC_PriorityGroup)

例:串口初始化

void My_USART1_Init(void)

{

GPIO_InitTypeDef GPIO_InitStru;

USART_InitTypeDef USART_InitStru;

NVIC_InitTypeDef NVIC_InitStru;

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);

RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);//USART1时钟来自PCLK2

GPIO_InitStru.GPIO_Mode= GPIO_Mode_AF_PP; //推挽复用输出

GPIO_InitStru.GPIO_Pin = GPIO_Pin_9;

GPIO_InitStru.GPIO_Speed = GPIO_Speed_10MHz;

GPIO_Init(GPIOA,&GPIO_InitStru);

GPIO_InitStru.GPIO_Mode= GPIO_Mode_IN_FLOATING; //浮空输入

GPIO_InitStru.GPIO_Pin = GPIO_Pin_10;

GPIO_InitStru.GPIO_Speed = GPIO_Speed_10MHz;

GPIO_Init(GPIOA,&GPIO_InitStru);

USART_InitStru.USART_BaudRate = 115200; //波特率为115200

USART_InitStru.USART_HardwareFlowControl = USART_HardwareFlowControl_None; //硬件流控制 为无

USART_InitStru.USART_Mode = USART_Mode_Rx|USART_Mode_Tx; //工作模式为发送和接收

USART_InitStru.USART_Parity = USART_Parity_No; //没有奇偶校验

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

USART_InitStru.USART_WordLength = USART_WordLength_8b; //字长为8(因为没有奇偶校验)

USART_Init(USART1,&USART_InitStru);

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

USART_ITConfig(USART1,USART_IT_RXNE,ENABLE); //设置串口中断的类型,这里选择接收缓冲区非空 产生中断

NVIC_InitStru.NVIC_IRQChannel = USART1_IRQn; //定义在哪个通道,在顶层文件stm32f10x.h文件中 定义

NVIC_InitStru.NVIC_IRQChannelCmd = ENABLE;

NVIC_InitStru.NVIC_IRQChannelPreemptionPriority = 1; //抢占优先级 0-3

NVIC_InitStru.NVIC_IRQChannelSubPriority = 1; //子优先级 0-3

NVIC_Init(&NVIC_InitStru); //设置中断抢占优先级和响应优先级

}

中断服务函数的格式在启动文件startup_stm32f10x_hd.s中定义

voidUSART1_IRQHandler(void)

{

u8 res;

if( USART_GetITStatus(USART1,USART_IT_RXNE)) //判断是否是接收到数据产生的中断

{

res = USART_ReceiveData(USART1); //接收数据

USART_SendData(USART1,res); //重新发送回这个数据

}

}

//初始化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

//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); //初始化串口


#if EN_USART1_RX //如果使能了接收

//注:在usart.h头文件中定义了

//#define EN_USART1_RX 1 //使能(1),禁止(0)串口1中断接收

//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); //根据指定的参数初始化NVIC寄存器

USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启接收中断,接收数据时产生中断

#endif

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

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

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭