STM32之串口通信

发布者:DelightfulSmile最新更新时间:2015-11-18 来源: eefocus关键字:STM32  串口通信 手机看文章 扫描二维码
随时随地手机看文章
实验目的:

实现利用串口1 不停的打印一个信息到电脑上,同时接收从串口发过来的数据,把发送过来的数据直接送回给电脑。

 

实验平台:

基于STM32F103C8T6的彩屏开发板

 

硬件接口:

        [转载]STM32之串口通信

 

注意:因为我的开发板上的串口和LED共用了PA9和PA10,所以在使用USART1时务必屏蔽LED,不然两者会互相影响而导致实现现象无法呈现。

 

相关寄存器:

1,串口时钟使能。串口作为STM32 的一个外设,其时钟由外设时钟使能寄存器控制,这

里我们使用的串口1 是在APB2ENR 寄存器的第14 位。

2,串口复位。串口1 的复位是通过配置APB2RSTR 寄存器的第14 位来实现的。。通过向该位写1来复位串口1,写0 结束复位。

3,串口波特率设置。每个串口都有一个自己独立的波特率寄存器USART_BRR

波特率的计算,STM32 的串口波特率计算公式如下:

                        [转载]STM32之串口通信
上式中, 是给串口的时钟(PCLK1 用于USART2、3、4、5,PCLK2 用于USART1);USARTDIV 是一个无符号定点数。我们只要得到USARTDIV 的值,就可以得到串口波特率寄存器USART1->BRR 的值。

4,串口控制。STM32 的每个串口都有3 个控制寄存器USART_CR1~3,串口的很多配置

都是通过这3 个寄存器来设置的

5,数据发送与接收。STM32 的发送与接收是通过数据寄存器USART_DR 来实现的,这是

一个双寄存器,包含了TDR 和RDR。

6,串口状态。串口的状态可以通过状态寄存器USART_SR 读取。

(注:详细的介绍使用请参考ST公司的数据手册)

 

程序设计:

(注:本人的usart.c usart.h delay.c delay.h sys.c sys.h是引用网上一位网友整理的)

 

    usart.h

#ifndef __USART_H

#define __USART_H

#include

#include "stdio.h"   

extern u8 USART_RX_BUF[64];     //接收缓冲,最大63个字节.末字节为换行符

extern u8 USART_RX_STA;         //接收状态标记   

 

//如果想串口中断接收,请不要注释以下宏定义

#define EN_USART1_RX //使能串口1接收

void uart_init(u32 pclk2,u32 bound);

 

#endif      

 

 

    usart.c

#include "sys.h"

#include "usart.h"

//加入以下代码,支持printf函数,而不需要选择use MicroLIB     

#if 1

#pragma import(__use_no_semihosting)            

//标准库需要的支持函数                

struct __FILE

{

  int handle;

 

 

 

};

 

FILE __stdout;      

//定义_sys_exit()以避免使用半主机模式   

_sys_exit(int x)

{

  x = x;

}

//重定义fputc函数

int fputc(int ch, FILE *f)

    

  while((USART1->SR&0X40)==0);//循环发送,直到发送完毕  

  USART1->DR = (u8) ch;     

  return ch;

}

#endif

//end

//////////////////////////////////////////////////////////////////

 

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

//串口1中断服务程序

//注意,读取USARTx->SR能避免莫名其妙的错误    

u8 USART_RX_BUF[64];     //接收缓冲,最大64个字节.

//接收状态

//bit7,接收完成标志

//bit6,接收到0x0d

//bit5~0,接收到的有效字节数目

u8 USART_RX_STA=0;       //接收状态标记    

 [page]

void USART1_IRQHandler(void)

{

  u8 res;    

  if(USART1->SR&(1<<5))//接收到数据

  

      res=USART1->DR;

      if((USART_RX_STA&0x80)==0)//接收未完成

      {

          if(USART_RX_STA&0x40)//接收到了0x0d

          {

              if(res!=0x0a)

                  USART_RX_STA=0;//接收错误,重新开始

              else

                  USART_RX_STA|=0x80; //接收完成了

          }else //还没收到0X0D

          

              if(res==0x0d)

                  USART_RX_STA|=0x40;

              else

              {

                  USART_RX_BUF[USART_RX_STA&0X3F]=res;

                  USART_RX_STA++;

                  if(USART_RX_STA>63)USART_RX_STA=0;//接收数据错误,重新开始接收    

                  

          }

                                                    

                                            

}

#endif

//该函数的重点就是判断接收是否完成,通过检测是否收到0X0D、0X0A 的连续2 个字节//(0X0D 后跟0X0A 表示回车键)来检测是否结束。当检测到这个结束序列之后,就会置//位USART_RX_STA 的最高为来标记已经收到了一次数据。之后等待外部函数清空该位//之后才开始第二次接收。所接收的数据全部存放在USART_RX_BUF 里面,一次接收数//据不能超过64个字节,否则被丢弃。                                   

 

 

//初始化IO 串口1

//pclk2:PCLK2时钟频率(Mhz)

//bound:波特率

//CHECK OK

//091209

void uart_init(u32 pclk2,u32 bound)

    

  float temp;

  u16 mantissa;

  u16 fraction;     

  temp=(float)(pclk2*1000000)/(bound*16);//得到USARTDIV

  mantissa=temp;              //得到整数部分

  fraction=(temp-mantissa)*16; //得到小数部分

    mantissa<<=4;

  mantissa+=fraction;

  RCC->APB2ENR|=1<<2;   //使能PORTA口时钟 

  RCC->APB2ENR|=1<<14;  //使能串口时钟

  GPIOA->CRH&=0XFFFFF00F;

  GPIOA->CRH|=0X000008B0;//IO状态设置

       

  RCC->APB2RSTR|=1<<14;   //复位串口1

  RCC->APB2RSTR&=~(1<<14);//停止复位        

  //波特率设置

  USART1->BRR=mantissa; // 波特率设置

  USART1->CR1|=0X200C;  //1位停止,无校验位.

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

  //使能接收中断

  USART1->CR1|=1<<8;    //PE中断使能

  USART1->CR1|=1<<5;    //接收缓冲区非空中断使能         

  MY_NVIC_Init(3,3,USART1_IRQChannel,2);//组2,最低优先级

#endif

}

 

  主函数

#include

#include"common.h"   

int main(void)

       

  u8 t;

  u8 len;

  u16 times=0; 

  Stm32_Clock_Init(9); //系统时钟设置

  delay_init(72);      //延时初始化

  uart_init(72,9600); //串口初始化为9600 

  while(1)

  {

      if(USART_RX_STA&0x80)

                         

          len=USART_RX_STA&0x3f;//得到此次接收到的数据长度

          printf("n您发送的消息为:n");

          for(t=0;t

          {

              USART1->DR=USART_RX_BUF[t];

              while((USART1->SR&0X40)==0);//等待发送结束

          }

          printf("nn");//插入换行

          USART_RX_STA=0;

      }else

      {

          times++;

          if(timesP00==0)

          {

              printf("n 简单的串口实验n");

          }

          if(times 0==0)printf("请输入数据,以回车键结束n"); 

          delay_ms(10);  

      }

  

}

 

实验现象:

 

 [转载]STM32之串口通信
 


关键字:STM32  串口通信 引用地址:STM32之串口通信

上一篇:S3C2440存储控制器和MMU浅析
下一篇:stm32应用-简单的串口接收与发送程序

推荐阅读最新更新时间:2024-03-16 14:40

STM32+HC05串口蓝牙设计简易的蓝牙音箱
一、环境介绍 MCU: STM32F103C8T6 蓝牙模块: HC05 (串口蓝牙) 音频解码模块: VS1053B OLED显示屏: 0.96寸SPI接口OLED 开发软件: Keil5 上位机: 使用QT设计Android端APP 二、功能介绍 Android手机打开APP,设置好参数之后,选择音乐文件发送给蓝牙音箱设备端,HC05蓝牙收到数据之后,再传递给VS1053进行播放。程序里采用环形缓冲区,接收HC05蓝牙传递的数据,设置好传递的参数之后,基本播放音乐是很流畅的。 完整项目源码下载地址: https://download.csdn.net/download/xiaolong1126626
[单片机]
STM32+HC05串口蓝牙设计简易的蓝牙音箱
运用LabView控制DS3900串口通信模块
本篇应用笔记讨论了DS3900串口通信模块和LabView的使用问题,LabView是适用于嵌入式应用的图形化界面开发平台。本文可作为面向DS3900的LabView界面用户指南。 介绍 LabView作为嵌入式应用的图形化界面开发平台得到了广泛应用。DS3900串口通信模块是一款通用接口板,借助该接口板,用户可使用PC串口与具有I²C接口的器件通信。DS3900的指令集允许应用软件与I²C器件直接通信。 本应用笔记是面向DS3900的LabView界面的用户指南。首先,用户需要安装LabView。为用户提供了标准VI,以加载并运行程序。可以下载与本应用笔记相关的LabView代码(ZIP,321K)。 使用LabVie
[电源管理]
运用LabView控制DS3900<font color='red'>串口通信</font>模块
STM32】HAL库 STM32CubeMX教程七---PWM输出(呼吸灯)
前言: 本系列教程将 对应外设原理,HAL库与STM32CubeMX结合在一起讲解,使您可以更快速的学会各个模块的使用 所用工具: 1、芯片: STM32F407ZET6/ STM32F103ZET6 2、STM32CubeMx软件 3、IDE: MDK-Keil软件 4、STM32F1xx/STM32F4xxHAL库 知识概括: 通过本篇博客您将学到: PWM工作原理 STM32CubeMX创建PWM例程 HAL库定时器PWM函数库 PWM创建呼吸灯 什么是PWM 脉冲宽度调制(PWM),是英文“Pulse Width Modulation”的缩写,简称脉宽调制,是利用微处理器的数字输出来对模拟电路进行控制的一种非
[单片机]
【<font color='red'>STM32</font>】HAL库 STM32CubeMX教程七---PWM输出(呼吸灯)
关于STM32的中断问题集锦
1、STM32的EXIT的库函数 问:EXTI_GetFlagStatus和EXTI_GetITStatus区别是什么? 答:一个是获取状态也的,一个是获取中断的。你可以看一下函数上面的说明。 2、STM32的外部中断 问:STM32的外部中断响应的最大频率是多少? 答:应该是2-3个机器周期 3、判断中断是否会被响应的依据是什么? 答:1、首先是占先式优先级,其次是副优先级;2、占先式优先级决定是否会有中断嵌套;3、Reset、NMI、HardFault优先级为负(高于普通中断优先级)且不可调整。 4、什么是占先式优先级? 答:占先式优先级(pre-emptionpriority):高占先式优
[单片机]
STM32的Flash写了保护怎么办
关于STM32对内部Flash的保护 为了防止对Flash的非法访问,所有STM32的芯片都提供对Flash的保护,具体分为写保护和读保护。 如果对Flash设置了写保护,那就无法对Flash进行编程和擦除。在开发STM32的时候,如果出现这种情况,通常仿真器都支持对Flash进行解锁,像jlink,stlink等仿真器都支持这个功能。 在使用MDK进行调试的时候,可能会遇到如下图所示的报错信息,这时候就要排查Flash是不是被保护起来了。 读保护即大家通常说的“加密”,是作用于整个Flash存储区域,相关文章:STM32等单片机程序加密的方法。一旦设置了Flash的读保护,内置的Flash存储区只能通过程序的正常
[单片机]
<font color='red'>STM32</font>的Flash写了保护怎么办
stm32串口通信调试总结
本文分为两部分,即”以USART1为例的串口初始化”和“调试中遇到的问题” 以USART1为例的串口初始化 本程序调用了stm32自带的固件库,工程中具体的文件见下图: 一.GPIO及USART1初始化结构体变量定义 GPIO_InitTypeDef GPIO_InitStructure; USART_InitTypeDef USART_InitStructure;12 二.串口时钟及GPIO端口时钟使能 USART1是挂在APB2总线上的外设。 TX,RX分别是PA9,PA10端口的复用。 要使用到端口复用,就要使能端口的时钟,并使能相应外设的时钟。这里可使用|同时使能这两个时钟。 RCC_APB2PeriphCloc
[单片机]
<font color='red'>stm32</font><font color='red'>串口通信</font>调试总结
stm32 串口通讯不成功的解决办法
首先要注意所用到的USART是否用到了复用功能 千万别忘了打开复用时钟!!!!!!!!! 代码如下:Hello! everyone,welcome to class! #include void delay_ms(u16 x) { u8 t; while(x--){for(t=0;t 120;t++);} } void GPIO_Configuration(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC | RCC_APB2Periph_AFIO, ENABLE); GPIO_InitS
[单片机]
<font color='red'>stm32</font> 串口通讯不成功的解决办法
STM32在程序中禁用JTAG功能后不能进行仿真的解决办法
STM32在程序中禁用JTAG功能后不能进行仿真会出现不能进行JTAG仿真的现象。 在程序中,通常禁止JTAG的函数如下所示: //改变指定管脚的映射 GPIO_Remap_SWJ_Disable SWJ 完全失能(JTAG+SW-DP) GPIO_PinRemapConfig(GPIO_Remap_SWJ_Disable, ENABLE); //改变指定管脚的映射 GPIO_Remap_SWJ_JTAGDisable JTAG-DP 失能 + SW-DP使能 GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable , ENABLE); 当设置了以上语句后,当运行了这两个语句后
[单片机]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
随便看看
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved