实验要求:
@[TOC]实验一:通过stm32通过串口发送数据到电脑的串口组手。
@[TOC]实验二:电脑发送串口数据到单片机,单片机再回传到电脑。
预备知识
实验一:
My_Usart.c
#include "My_usart.h"
#include "sys.h"
//第一步GPIO的复用
void My_usart_init(int boud_rate)
{
GPIO_InitTypeDef GPIO_Initstructure;//GPIO初始化结构体
USART_InitTypeDef USART_Initstructure;//串口初始化结构体
//(1)第一步,使能串口时钟和对应GPIO口的时钟
//GPIOA 和 USART1 时钟使能
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE); //使能 GPIOA 时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE );//使能 USART1
// (2)设置引脚复用器映射:调用 GPIO_PinAFConfig 函数。
GPIO_PinAFConfig(GPIOA,GPIO_PinSource9,GPIO_AF_USART1);//PA9 复用为 USART1
GPIO_PinAFConfig(GPIOA,GPIO_PinSource10,GPIO_AF_USART1);//PA10 复用为 USART1
// (3)GPIO 初始化设置:要设置模式为复用功能。
GPIO_Initstructure.GPIO_Pin = GPIO_Pin_9|GPIO_Pin_10;//GPIO口A9和A10
GPIO_Initstructure.GPIO_Mode = GPIO_Mode_AF;//复用功能
GPIO_Initstructure.GPIO_PuPd = GPIO_PuPd_UP;//上拉
GPIO_Initstructure.GPIO_Speed = GPIO_Fast_Speed;//速度 50MHz
GPIO_Initstructure.GPIO_OType = GPIO_OType_PP;//推挽复用输出
GPIO_Init(GPIOA,&GPIO_Initstructure); //初始化 PA9,PA10
//(4)USART 初始化设置
USART_Initstructure.USART_BaudRate =boud_rate;//波特率
USART_Initstructure.USART_WordLength = USART_WordLength_8b;//字长为八位数据格式
USART_Initstructure.USART_StopBits =USART_StopBits_1 ;//一个停止位
USART_Initstructure.USART_Parity = USART_Parity_No ;//无奇偶校验
USART_Initstructure.USART_Mode =USART_Mode_Rx|USART_Mode_Tx ;//接收和发送模式
USART_Initstructure.USART_HardwareFlowControl =USART_HardwareFlowControl_None ;//异步不需要硬件流控制
USART_Init(USART1,&USART_Initstructure);//初始化串口
//配置中断
NVIC_Configuration();
//使能串口中断
USART_Cmd(USART1, ENABLE);
}
void NVIC_Configuration()
{
NVIC_InitTypeDef NVIC_InitStructure;//中断结构体
//Usart1 NVIC 配置
NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;//串口1中断通道
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3;//抢占优先级3
NVIC_InitStructure.NVIC_IRQChannelSubPriority =3; //子优先级3
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能
NVIC_Init(&NVIC_InitStructure); //根据指定的参数初始化VIC寄存器、
//使能串口接收中断
USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);
}
//发送单个字符函数
void Usart_SendByte(USART_TypeDef *pUSART,uint8_t ch)
{
USART_SendData(USART1,ch);
while( USART_GetFlagStatus(pUSART,USART_FLAG_TXE)==RESET);
//USART_FLAG_TXE为发送寄存器空标志位,USART_FLAG_TC为发送移位寄存器标志位
//USART_IT_TXE和USART_IT_TC中断标志位
}
/*字符串发送函数*/
void Usart_Sendstring(USART_TypeDef *pUSART,char *str)
{
unsigned int i=0;
do
{
Usart_SendByte(pUSART,*(str+i));
i++;
} while( *(str+i) != '\0' );
while( USART_GetFlagStatus(pUSART,USART_IT_TC)==RESET)//获取中断状态标志位.等待发送完
{}
}
mian.c
#include "stm32f4xx.h"
#include "key.h"
#include "delay.h"
#include "led.h"
#include "sys.h"
#include "stm32f4xx_it.h"
#include "My_usart.h"
void main(void)
{
delay_init(168); //延时初始化
Led_Init();//led初始化函数
My_usart_init(115200);
//Usart_SendByte(USART1,'a');
Usart_Sendstring(USART1,"马伟的串口实验\0");
while(1);
}
实验二:
其中主要代码与实验一一样,加如下代码
//发送单个字符函数
void Usart_SendByte(USART_TypeDef *pUSART,uint8_t ch)
{
USART_SendData(USART1,ch);
while( USART_GetFlagStatus(pUSART,USART_FLAG_TXE)==RESET);
//USART_FLAG_TXE为发送寄存器空标志位,USART_FLAG_TC为发送移位寄存器标志位
//USART_IT_TXE和USART_IT_TC中断标志位
}
中断函数,我个人写在stm32f4xx_it.c里面
void USART1_IRQHandler()
{
uint8_t ucTemp;
//当 RDR 移位寄存器的内容已传输到 USART_DR 寄存器时,该位由硬件置 1
if (USART_GetITStatus(USART1,USART_IT_RXNE)!=RESET)//获取中断标志位,为1
{
ucTemp = USART_ReceiveData( USART1 );
USART_SendData(USART1,ucTemp);
}
}
mian.c
void main(void)
{
delay_init(168); //延时初始化
Led_Init();//led初始化函数
My_usart_init(115200);
//Usart_SendByte(USART1,'a');
//Usart_Sendstring(USART1,"马伟的串口实验\0");
printf("马伟的实验");
while(1);
}