开发板芯片:STM32F407ZGT6

PA9-USART1_TX,PA10-USART1_RX;

PF9-LED0,PF10-LED1;

一、串口1配置过程(不使用串口中断):

1.使能时钟,包括GPIO时钟和串口1时钟使能,注意它们是挂载在不同的时钟总线上的。

    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);    //使能端口时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE); //使能串口时钟

2.端口映射到串口1。

    GPIO_PinAFConfig(GPIOA, GPIO_PinSource9, GPIO_AF_USART1);//端口映射到串口
GPIO_PinAFConfig(GPIOA, GPIO_PinSource10, GPIO_AF_USART1);

3.GPIO初始化,注意配置模式时,要设置为复用模式,其他和GPIO基本配置相同

    IO_Init。GPIO_Mode = GPIO_Mode_AF;            //设置为复用模式
IO_Init。GPIO_OType = GPIO_OType_PP;
IO_Init。GPIO_Pin = GPIO_Pin_9 | GPIO_Pin_10;
IO_Init。GPIO_PuPd = GPIO_PuPd_UP;
IO_Init。GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &IO_Init);

4.串口1初始化,包括:波特率,发送接收使能,数据位,停止位,奇偶校验,硬件控制流

    UT_Init。USART_BaudRate = ;        //波特率
UT_Init。USART_HardwareFlowControl = USART_HardwareFlowControl_None; //无硬件控制流
UT_Init。USART_Mode = USART_Mode_Rx | USART_Mode_Tx; //发送接收
UT_Init。USART_Parity = USART_Parity_No; //无奇偶校验
UT_Init。USART_StopBits = USART_StopBits_1; //停止位1
UT_Init。USART_WordLength = USART_WordLength_8b; //数据位8
USART_Init(USART1, &UT_Init);

5.串口1使能。

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

二、相关的函数和寄存器介绍

1、数据接收

1.1.状态寄存器

其中的位5:RXNE就是可以接收完成标志位,当数据接收完成时,RXNE会自动置1,我们可以通过读取该位的状态,来判断是否接收到数据,直接读取该位的方法是

  u8 Receive_Data = 0;

 Receive_Flag = USART1->SR &  << ;        //若接收到数据则标志位=1

上面的方法就是直接读取位5的值,接收完成则Receive_Flag=1。

其实我们更喜欢用一个函数来读取RXNE的状态:

USART_GetFlagStatus(USART1, USART_IT_RXNE)

该函数不仅可以读取RXNE的状态,还可以读取其他位的状态,若标志置1,该函数则会返回SET,即1。

我们可以看看该函数是如何实现的呢?

FlagStatus USART_GetFlagStatus(USART_TypeDef* USARTx, uint16_t USART_FLAG)
{
FlagStatus bitstatus = RESET;
/* Check the parameters */
assert_param(IS_USART_ALL_PERIPH(USARTx));
assert_param(IS_USART_FLAG(USART_FLAG)); /* The CTS flag is not available for UART4 and UART5 */
if (USART_FLAG == USART_FLAG_CTS)
{
assert_param(IS_USART_1236_PERIPH(USARTx));
} if ((USARTx->SR & USART_FLAG) != (uint16_t)RESET)
{
bitstatus = SET;
}
else
{
bitstatus = RESET;
}
return bitstatus;
}

可以看出该函数也是通过读取寄存器相应的位来返回相应的状态。

还有另一个函数和这个很相似:

USART_GetITStatus(USART1, USART_IT_RXNE);

这个函数也是读取RXNE的状态,不同的是当我们打开了串口中断,接收到数据时,这个函数才会返回1.

也就是说,无论串口中断是打开,接受完成后RXNE标志位都会置1,即USART_GetFlagStatus(USART1, USART_IT_RXNE)返回1,如果开启了中断,那么USART_GetFlagStatus(USART1, USART_IT_RXNE)和USART_GetITStatus(USART1, USART_IT_RXNE)都会置1。

当RXNE置1,即接收到数据后,我们要及时的读取接收到的数据,相应的函数是Receive_Data = USART_ReceiveData(USART1);

我们也可以直接读取USART1->DR寄存器,如Receive_Data = USART1->DR;

1.2.数据寄存器

当读取完成后,RXNE将自动清零,如果没有读取数据,则需要软件清零,可以使用函数USART_ClearFlag(USART1,USART_IT_RXNE);来进行手动清零,或通过将状态寄存器位5-RXNE置为0:USART1->SR &= ~(1 << 5);这一点是很重要的。

2.数据发送

将数据进行处理后,我们需要回复一些数据,我们可以通过函数USART_SendData(USART1,data);来向外发送数据,当然我们也可以像读取数据那样,将数据写入DR寄存器来实现 USART1->DR=data。其实我们更多的是通过重定向printf()函数来向外格式化输出数据,如printf("Receive Succsed: %d \r\n", Data);这个更方便一些。

三、程序实现

那么我们怎么来实现电脑发送0x01来控制LED反转呢?

1.串口配置函数

void USART1_Config()
{
GPIO_InitTypeDef IO_Init;
USART_InitTypeDef UT_Init; RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE); //使能端口时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE); //使能串口时钟 GPIO_PinAFConfig(GPIOA, GPIO_PinSource9, GPIO_AF_USART1);//端口映射到串口
GPIO_PinAFConfig(GPIOA, GPIO_PinSource10, GPIO_AF_USART1); IO_Init.GPIO_Mode = GPIO_Mode_AF; //设置为复用模式
IO_Init.GPIO_OType = GPIO_OType_PP;
IO_Init.GPIO_Pin = GPIO_Pin_9 | GPIO_Pin_10;
IO_Init.GPIO_PuPd = GPIO_PuPd_UP;
IO_Init.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &IO_Init); UT_Init.USART_BaudRate = ; //波特率
UT_Init.USART_HardwareFlowControl = USART_HardwareFlowControl_None; //无硬件控制流
UT_Init.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; //发送接收
UT_Init.USART_Parity = USART_Parity_No; //无奇偶校验
UT_Init.USART_StopBits = USART_StopBits_1; //停止位1
UT_Init.USART_WordLength = USART_WordLength_8b; //数据位8
USART_Init(USART1, &UT_Init); USART_Cmd(USART1, ENABLE); //串口使能
}

2.主函数

主函数怎么写呢?其实很简单,就是对接收的数据进行一下判断就OK了,如果接收到的数据==0x01,那么LED=!LED,是不是很简单呢?

int main(void)
{
LED_Init();
delay_init();
USART1_Config();
LED0_OFF;
LED1_OFF;
while()
{
u8 Receive_Data = ;
if(USART_GetITStatus(USART1, USART_IT_RXNE)) //若接收到数据
{
Receive_Data = USART_ReceiveData(USART1); //读取数据后,RXNE标志位自动清零;
printf("Receive Succsed: %2x \r\n", Receive_Data);
if(Receive_Data == 0x01)
LED0 = !LED0;
// USART_ClearFlag(USART1,USART_IT_RXNE); //手动清零
}
else
{
LED1 = !LED1;
printf("Running....\r\n");
delay_ms();
}
}
}

这里使用了串口发送“Running....”和LED1闪烁来指示程序正在运行,这种方法很常用,可以指示程序的运行状态。

本文章没有使用串口中断来进行控制,下一篇文章将会介绍如何使用串口中断来控制LED。

参考资料:

STM32F4xx中文参考手册

STM32F4开发指南-库函数版本_V1.1

USART_百度百科

以上是我学习过程的一些个人理解,有不对或不准确的地方,欢迎各位大神指正。

2017年4月16日21:58:46

欢迎大家关注我的个人博客 http://www.wangchaochao.top/
微信扫码关注我的公众号

不定期更新个人学习笔记和技术总结,欢迎大家互相学习交流!

STM32学习笔记(二)——串口控制LED的更多相关文章

  1. stm32学习笔记----双串口同时打开时的printf()问题

    stm32学习笔记----双串口同时打开时的printf()问题 最近因为要使用串口2外接PN532芯片实现通信,另一方面,要使用串口1来将一些提示信息输出到上位机,于是重定义了printf(),使其 ...

  2. STM32学习笔记(二) 基于STM32-GPIO的流水灯实现

    学会了如何新建一个工程模板,下面就要开始动手实践了.像c/c++中经典的入门代码"hello world"一样,流水灯作为最简单的硬件设备在单片机领域也是入门首推.如果你已经有了一 ...

  3. stm32学习笔记之串口通信

    在基础实验成功的基础上,对串口的调试方法进行实践.硬件代码顺利完成之后,对日后调试需要用到的printf重定义进行调试,固定在自己的库函数中. b) 初始化函数定义: void USART_Confi ...

  4. STM32学习笔记——USART串口

    转载自:http://www.cnblogs.com/microxiami/p/3752715.html 一.USART简介 通用同步异步收发器(USART)提供了一种灵活的方法与使用工业标准NRZ异 ...

  5. STM32学习笔记——USART串口(向原子哥和火哥学习)

    一.USART简介 通用同步异步收发器(USART)提供了一种灵活的方法与使用工业标准NRZ异步串行数据格式的外部设备之间进行全双工数据交换.USART利用分数波特率发生器提供宽范围的波特率选择. S ...

  6. Arduino学习笔记二:修改LED点灯程序

    看了开源社区的LED控制程序,开始上手代码编写,修改,下载以及调试,原文地址:http://www.arduino.cn/thread-1072-1-1.html,这个帖子写的比较通俗易懂. 自己移植 ...

  7. STM32学习笔记(四)——串口控制LED(中断方式)

    目录: 一.时钟使能,包括GPIO的时钟和串口的时钟使能 二.设置引脚复用映射 三.GPIO的初始化配置,注意要设置为复用模式 四.串口参数初始化配置 五.中断分组和中断优先级配置 六.设置串口中断类 ...

  8. STM32学习笔记——点亮LED

    STM32学习笔记——点亮LED 本人学习STM32是直接通过操作stm32的寄存器,使用的开发板是野火ISO-V2版本: 先简单的介绍一下stm32的GPIO: stm32的GPIO有多种模式: 1 ...

  9. stm32学习笔记——外部中断的使用

    stm32学习笔记——外部中断的使用 基本概念 stm32中,每一个GPIO都可以触发一个外部中断,但是,GPIO的中断是以组为一个单位的,同组间的外部中断同一时间只能使用一个.比如说,PA0,PB0 ...

  10. STM32学习笔记——OLED屏

    STM32学习笔记--OLED屏 OLED屏的特点: 1.  模块有单色和双色可选,单色为纯蓝色,双色为黄蓝双色(本人选用双色): 2.  显示尺寸为0.96寸 3.  分辨率为128*64 4.   ...

随机推荐

  1. 学好UI你必须要掌握这些技术

    转自:http://blog.sina.com.cn/s/blog_15da22ed10102x0gx.html ui设计现在已经是设计行业中的瞩目之星,无论在PC端.移动端还是游戏上都是大放异彩. ...

  2. Sass使用小技巧

    1.任何可以用作css属性值的赋值都可以用作sass变量值.如果变量值与属性不匹配,sass会当作普通字符串来处理. $family: "microsoft yahei", Ari ...

  3. 你真的会 snapshot 吗? - 每天5分钟玩转 OpenStack(163)

    ​这是 OpenStack 实施经验分享系列的第 13 篇. instance snapshot 操作可用于备份或者将 instance 保存为新的 image.如果在生产系统中执行 snapshot ...

  4. Java集合ArrayList源码解读

    最近在回顾数据结构,想到JDK这样好的代码资源不利用有点可惜,这是第一篇,花了心思.篇幅有点长,希望想看的朋友认真看下去,提出宝贵的意见.  :) 内部原理 ArrayList 的3个字段 priva ...

  5. 使用Nginx+CppCMS构建高效Web应用服务器

    使用Nginx+CppCMS构建高效Web应用服务器 1:Why当前,越来越多的网站使用了各种框架,大部分框架使用了脚本语言.半编译语言等.比如Java.Python.Php.C#.NET等.这些框架 ...

  6. RabbitMQ-从基础到实战(1)— Hello RabbitMQ

    转载请注明出处 1.简介 本篇博文介绍了在windows平台下安装RabbitMQ Server端,并用JAVA代码实现收发消息 2.安装RabbitMQ RabbitMQ是用Erlang开发的,所以 ...

  7. [复习]java中hashCode的作用

    1.HashCode的官方文档定义 (1)hashcode方法返回该对象的哈希码值.支持该方法是为哈希表提供一些优点,例如java.util.HashTable提供的哈希表. (2)hashCode的 ...

  8. 【树莓派】h2数据库操作相关

    之前在树莓派上面操作时候,遇到一些业务方面的bug,和团队中的同事经过多次尝试,但就是难以重现用户现场的问题. 但是问题却实实在在地发生,虽然并不是必然可重现的bug,但是也比较闹心: 发生了问题,也 ...

  9. Java字节流在Android中的使用

    引言:项目开发有时会使用上传文件到服务器,再从服务器取数据显示到本地这一过程:或者输入一段文字,再把文字显示出来.这个过程都用到了IO流. IO流分为字符流(Reader\Writer)和字节流(In ...

  10. 利用echo命令实现倒计时的功能

    echo -e:启用反斜线控制字符的转换        -E:关闭反斜线控制字符的转换(预设如此)        -n:取消行末之换行符号(与 -e 选项下的 \c 字符同意 -e参数下的控制参数 \ ...