#include "delay.h"
#include "uart5.h"
#include "stdarg.h"
#include "stdio.h"
#include "string.h"
/*串口5驱动代码,库函数版本 */
//串口发送缓存区
__align(8) u8 UART5_TX_BUF[UART5_MAX_SEND_LEN]; //发送缓冲,最大UART5_MAX_SEND_LEN字节
#ifdef UART5_RX_EN //如果使能了接收
//串口接收缓存区
u8 UART5_RX_BUF[UART5_MAX_RECV_LEN]; //接收缓冲,最大UART5_MAX_RECV_LEN个字节.
//通过判断接收连续2个字符之间的时间差不大于10ms来决定是不是一次连续的数据.
//如果2个字符接收间隔超过10ms,则认为不是1次连续数据.也就是超过10ms没有接收到
//任何数据,则表示此次接收完毕.
//接收到的数据状态
//[15]:0,没有接收到数据;1,接收到了一批数据.
//[14:0]:接收到的数据长度
u16 UART5_RX_STA=0;
void UART5_IRQHandler(void)
{
u8 res;
if(USART_GetITStatus(UART5, USART_IT_RXNE) != RESET)//接收到数据,(接收到的数据必须是0x0d 0x0a结尾)
{
res =USART_ReceiveData(UART5);//读取到收到的数据
if(UART5_RX_STA<UART5_MAX_RECV_LEN) //还可以接收数据
{
TIM_SetCounter(TIM3,0);//计数器清空
if(UART5_RX_STA==0)TIM3_Set(1); //使能定时器3的中断
UART5_RX_BUF[UART5_RX_STA++]=res; //记录接收到的值
}else
{
UART5_RX_STA|=1<<15; //强制标记接收完成
}
}
}
//void UART5_IRQHandler(void) //串口5中断服务程序
//{
// u8 res;
// if(USART_GetITStatus(UART5, USART_IT_TXE) != RESET)
// {
// USART_ClearITPendingBit(UART5, USART_IT_TXE);
// }
// if(USART_GetITStatus(UART5, USART_IT_RXNE) != RESET) //接收中断(接收到的数据必须是0x0d 0x0a结尾)
// {
// res = USART_ReceiveData(UART5);//(UART5->DR); //读取接收到的数据
// }
// }
//初始化IO PC10 PC11 串口4
//pclk1CLK1时钟频率(Mhz)
//bound:波特率
void UART5_Init(u32 bound)
{
//0.GPIO端口设置
GPIO_InitTypeDef GPIO_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;
USART_InitTypeDef UART_InitStructure;
//1.串口时钟使能,复用端口
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC|RCC_APB2Periph_GPIOD,ENABLE); // GPIOC时钟 |RCC_APB2Periph_AFIO
//RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO ,ENABLE);
RCC_APB1PeriphClockCmd(RCC_APB1Periph_UART5,ENABLE);
//UART5_TX  C.12
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12; //PC.12
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出
GPIO_Init(GPIOC, &GPIO_InitStructure); //初始化PC.12
//UART5_RX  D.2
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;//PD.2
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入
GPIO_Init(GPIOD, &GPIO_InitStructure); //初始化PD.2
//2.复位串口5
USART_DeInit(UART5);
//3.串口参数初始化
//UART5->BRR=(36*1000000)/(bound);// 波特率设置
UART_InitStructure.USART_BaudRate = bound;
UART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为8位数据格式
UART_InitStructure.USART_StopBits = USART_StopBits_1;//一个停止位
UART_InitStructure.USART_Parity = USART_Parity_No;//无奇偶校验位
UART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//无硬件数据流控制
UART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; //收发模式
//初始化串口 5
USART_Init(UART5, &UART_InitStructure);
//使能串口
USART_Cmd(UART5, ENABLE);
#ifdef UART5_RX_EN //如果使能了接收
USART_ITConfig(UART5, USART_IT_RXNE, ENABLE);//若接收数据寄存器满,开启中断
NVIC_InitStructure.NVIC_IRQChannel = UART5_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; //抢占优先级1
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; //子优先级0
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能
NVIC_Init(&NVIC_InitStructure); //根据指定的参数初始化VIC寄存器
TIM3_Init(99,7199); //10ms中断
UART5_RX_STA=0; //清零
TIM3_Set(0); //关闭定时器4
#endif
}
//定时器4中断服务程序
void TIM3_IRQHandler(void)
{
if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET)//是更新中断
{
UART5_RX_STA|=1<<15; //标记接收完成
TIM_ClearITPendingBit(TIM3, TIM_IT_Update ); //清除TIMx更新中断标志
TIM3_Set(0); //关闭TIM3
}
}
//设置TIM3的开关
//sta:0,关闭;1,开启;
void TIM3_Set(u8 sta)
{
if(sta)
{
TIM_SetCounter(TIM3,0);//计数器清空
TIM_Cmd(TIM3, ENABLE); //使能TIMx
}else TIM_Cmd(TIM3, DISABLE);//关闭定时器4
}
//通用定时器中断初始化
//这里始终选择为APB1的2倍,而APB1为36M
//arr:自动重装值。
//psc:时钟预分频数
void TIM3_Init(u16 arr,u16 psc)
{
NVIC_InitTypeDef NVIC_InitStructure;
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //时钟使能//TIM3时钟使能
//定时器TIM3初始化
TIM_TimeBaseStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值
TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置用来作为TIMx时钟频率除数的预分频值
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; //设置时钟分割:TDTS = Tck_tim ???
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数模式
TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); //根据指定的参数初始化TIMx的时间基数单位
TIM_ITConfig(TIM3,TIM_IT_Update,ENABLE ); //使能指定的TIM3中断,允许更新中断
// NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn;
// NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1 ;//抢占优先级3
// NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2; //子优先级3
// NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能
// NVIC_Init(&NVIC_InitStructure); //根据指定的参数初始化VIC寄存器
//
NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn; //TIM3中断
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //先占优先级0级
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; //从优先级3级
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道被使能
NVIC_Init(&NVIC_InitStructure); //根据NVIC_InitStruct中指定的参数初始化外设NVIC寄存器
}
#endif
/**************************************************
UART5发送一个字节
**************************************************/
void UART5_Tx_Char(u8 i)
{
USART_SendData(UART5,i); //发往WIFI模块
while(USART_GetFlagStatus(UART5, USART_FLAG_TXE)==RESET);
//printf("tset:\r\n%s\r\n",i);
}
/*******************************************
UART5发送字符串
*******************************************/
void UART5_Tx_Str(u8 *pt, u8 length)
{
u8 i;
for( i=0;i<length;i++ ){
UART5_Tx_Char(*pt++);
}
//delay_ms(10);
}
ESP_01.c
#include "delay.h"
#include "usart.h"
#include "uart5.h"
#include "ESP_01.h"
#include "string.h"
#include "math.h"
/*ESP_01 Wifi驱动代码*/
/********************************************************************************
*功能:通过AT指令检测ESP_01Wifi模块是否已经连接 *
*初始化ESP_01模块 *
*返回值:0,成功;1,失败. *
********************************************************************************/
u8 ESP_01_Init(void)
{
u8 retry=10,t;
u8 temp=1;
u8 len;
u8 reclen = 0;
u8 AT[] = "AT\r\n"; //+GMR
UART5_Init(9600);//初始化串口5为:115200,波特率.
while(retry--)
{
delay_ms(10);
/*发送AT测试指令到WIFI,查看是否接收OK*/
UART5_Tx_Str(AT, sizeof(AT)/sizeof(AT[0]));
for(t=0;t<10;t++) //最长等待50ms,来接收WIFI模块的回应
{
if(UART5_RX_STA&0X8000)
break;
delay_ms(5);
}
if(UART5_RX_STA&0x8000) //接收到一次数据了
{
temp=UART5_RX_STA&0X7FFF; //得到数据长度
/******************************test_uart4****************************/
len=UART5_RX_STA&0x3fff;//得到此次接收到的数据长度
reclen=UART5_RX_STA&0X7FFF; //得到数据长度
UART5_RX_BUF[reclen]=0; //加入结束符
printf("test2:%s\r\n",UART5_RX_BUF);//发送回应数据到串口1
/*********************************************************************/
UART5_RX_STA=0;
if((temp==0x0C)&&UART5_RX_BUF[7]=='O'&&UART5_RX_BUF[8]=='K')
{
temp=0;//接收到OK响应
break;
}
}
}
if(retry==0)temp=1; //检测失败
return temp;
}
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
main.c
#include "stm32f10x.h"
#include "delay.h"
#include "sys.h"
#include "usart.h"
#include "uart4.h"
#include "uart5.h"
#include "led.h"
#include "spp_ca.h"
#include "ESP_01.h"
#include "string.h"
u32 i=0; // 定义延时参量
int main(void)
{
u8 t;
u8 len;
u16 times=0;
u8 reclen=0;
SystemInit(); //系统时钟配置函数通过不同的定义 来选择不同的主频 这里设置是72m
LED_Init(); //初始化与LED连接的硬件接口
delay_init(); //延时函数初始化
NVIC_Configuration();// 设置中断优先级分组
uart_init(9600); //串口初始化为9600
//while (SPP_CA_Init());//初始化蓝牙模块,先判断STM32和蓝牙是否连接上
while (ESP_01_Init());//初始化WIFI模块,先判断STM32和WIFI是否连接上
while(1)
{
if(USART_RX_STA&0x8000)//通过USART1接收数据(即PC->USART1->UART5)
{
len=USART_RX_STA&0x3fff;//得到此次接收到的数据长度
printf("\r\nSend_data2:\r\n");
for(t=0;t<len;t++)
{
USART1->DR=USART_RX_BUF[t];
while((USART1->SR&0X40)==0);//等待发送结束
}
//UART5_Tx_Str(USART_RX_BUF, strlen(USART_RX_BUF));//发送--UART5--WIFI
UART5_Tx_Str(USART_RX_BUF, sizeof(USART_RX_BUF)/sizeof(USART_RX_BUF[0]));//发送--UART5--WIFI
/*最长等待500ms,来接收WIFI模块的回应 */
for(t=0;t<10;t++)
{
if(UART5_RX_STA&0X8000)break;
delay_ms(50);
}
for(t=0;t<len;t++) USART_RX_BUF[t] = 0;//这里USART1发到UART5上去后,500ms后就开始清零USART1
if(UART5_RX_STA&0x8000)//通过UART4接收数据
{
len=UART5_RX_STA&0x3fff;//得到此次接收到的数据长度
reclen=UART5_RX_STA&0X7FFF; //得到数据长度
UART5_RX_BUF[reclen]=0; //加入结束符
printf("\r\n%s\r\n",UART5_RX_BUF);//发送回应数据到串口1
UART5_RX_STA = 0;
delay_ms(500);
}
else
{
times++;
if(times%200==0)
printf("Wifi module data returned failure!\r\n");
}
USART_RX_STA=0;
}
}
return 0;
}