概述
这个项目主要是将各类传感器的上报的数据通过WIFI模块上报值阿里云服务器,然后用其配套的云智能APP在手机上显示出来,便且可以控制I/O的开关。对于传感器上报的数据设置了阈值报警和自动处理操作,还可以语音控制相关的I/O的开关。所以用到了很多定时器、ADC、中断的操作。
主函数main.c
#include "stm32f10x.h"
#include "main.h"
#include "delay.h"
#include "usart1.h"
#include "usart2.h"
#include "timer1.h"
#include "timer2.h"
#include "timer3.h"
#include "timer4.h"
#include "wifi.h"
#include "led.h"
#include "mqtt.h"
#include "key.h"
#include "dht11.h"
#include "iic.h"
#include "oled.h"
#include "adc.h"
#include "Bujing.h"
#include "voice.h"
#include "TM.h"
int main(void)
{
u16 adcx;//烟雾浓度
float light;//光照强度
float temp1;//光照强度中间变量1
u8 temp2;//光照强度中间变量2
//u8 C_Ledflag=2;//手动开关灯与是否启动自动标志
//u8 No_Voice_Flag;
Delay_Init(); //延时功能初始化
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置NVIC中断分组2:2位抢占优先级,2位响应优先级
Usart1_Init(9600); //串口1功能初始化,波特率9600
Usart2_Init(115200); //串口2功能初始化,波特率115200
TIM4_Init(300,7200); //TIM4初始化,定时时间 300*7200*1000/72000000 = 30ms
LED_Init(); //LED初始化
KEY_Init(); //按键初始化
OLED_Init(); //oled初始化
KEY_Exti_Init();
Adc_Init(); //ADC1初始化
Adc2_Init(); //ADC2初始化
BJ_GPIO(); //步进电机初始化
Voice_Init(); //离线语音
//TM_GPIO_Init(); //天猫精灵控制风扇与窗户
while(DHT11_Init()) //DHT11初始化
{
u1_printf("DHT11_INIT error rn");
}
u1_printf("DHT11_INIT success rn");
//DHT11_Read_Data(&tempdata,&humidata); //读取温湿度值
// OLED_ShowNum(68,0,tempdata,2,16); //显示温度
// OLED_ShowNum(68,3,humidata,2,16); //显示湿度
WiFi_ResetIO_Init(); //初始化WiFi的复位IO
MQTT_Buff_Init(); //初始化接收,发送,命令数据的 缓冲区 以及各状态参数
AliIoT_Parameter_Init(); //初始化连接阿里云IoT平台MQTT服务器的参数
while(1) //主循环
{ OLED_Clear();
//离线语音控制
Voice_Conrol();
//语音控制风扇与窗户
Tm_Fan_Control();
Tm_Windows_Control();
//光照强度检测,小于60%开灯
light=Get_Adc2_Average(ADC_Channel_5,1,10);
temp1=(1-light/4095)*100;//百分比
temp2=temp1;
if(light>=2458)//光线较暗,自动开灯
{
OLED_ShowCHinese(0,0,21);//光
OLED_ShowCHinese(18,0,22);//照
OLED_ShowCHinese(36,0,23);//强
OLED_ShowCHinese(54,0,2);//度
OLED_ShowString(70,0,":");
OLED_ShowNum(78,0,temp2,2,16);
OLED_ShowString(106,0,"%");
if(C_Ledflag==1)//手动开灯
{
LED1_ON;
}
else if(C_Ledflag==0)//手动关灯
{
LED1_OFF;
}
else if(C_Ledflag==2)//LED自动模式
{
if(LED0_IN_STA==0)LED1_ON;
}
}
else//光线较亮,自动关灯
{
OLED_ShowCHinese(0,0,21);//光
OLED_ShowCHinese(18,0,22);//照
OLED_ShowCHinese(36,0,23);//强
OLED_ShowCHinese(54,0,2);//度
OLED_ShowString(70,0,":");
OLED_ShowNum(78,0,temp2,2,16);
OLED_ShowString(106,0,"%");
if(C_Ledflag==1)//手动开灯
{
LED1_ON;
}
else if(C_Ledflag==0)//手动关灯
{
LED1_OFF;
}
else if(C_Ledflag==2)//LED自动模式
{
if(LED0_IN_STA==1)LED1_OFF;
}
}
//烟雾浓度检测,达到20%报警
adcx=Get_Adc_Average(ADC_Channel_1,1,10);
adcx=adcx*(50000/4096);
if(adcx>=12000)
{
GPIO_SetBits(GPIOB,GPIO_Pin_8);//蜂鸣器报警
FAN01=1; //风扇打开
if(Window==1)
{
Window=BUJING_Cotrol(0,3,160);//开窗通风
}
OLED_ShowCHinese(0,3,18);//烟
OLED_ShowCHinese(18,3,19);//雾
OLED_ShowCHinese(36,3,20);//浓
OLED_ShowCHinese(54,3,2);//度
OLED_ShowString(70,3,":");
OLED_ShowCHinese(78,3,26);//超
OLED_ShowCHinese(96,3,27);//标
Delay_Ms(600);
}
else
{
GPIO_ResetBits(GPIOB,GPIO_Pin_8);
OLED_ShowCHinese(0,3,18);//烟
OLED_ShowCHinese(18,3,19);//雾
OLED_ShowCHinese(36,3,20);//浓
OLED_ShowCHinese(54,3,2);//度
OLED_ShowString(70,3,":");
OLED_ShowCHinese(78,3,24);//正
OLED_ShowCHinese(96,3,25);//常
Delay_Ms(600);
}
/*--------------------------------------------------------------------*/
/* Connect_flag=1同服务器建立了连接,我们可以发布数据和接收推送了 */
/*--------------------------------------------------------------------*/
if(Connect_flag==1){
/*-------------------------------------------------------------*/
/* 处理发送缓冲区数据 */
/*-------------------------------------------------------------*/
if(MQTT_TxDataOutPtr != MQTT_TxDataInPtr){ //if成立的话,说明发送缓冲区有数据了
//3种情况可进入if
//第1种:0x10 连接报文
//第2种:0x82 订阅报文,且ConnectPack_flag置位,表示连接报文成功
//第3种:SubcribePack_flag置位,说明连接和订阅均成功,其他报文可发
if((MQTT_TxDataOutPtr[2]==0x10)||((MQTT_TxDataOutPtr[2]==0x82)&&(ConnectPack_flag==1))||(SubcribePack_flag==1)){
u1_printf("发送数据:0x%xrn",MQTT_TxDataOutPtr[2]); //串口提示信息
MQTT_TxData(MQTT_TxDataOutPtr); //发送数据
MQTT_TxDataOutPtr += BUFF_UNIT; //指针下移
if(MQTT_TxDataOutPtr==MQTT_TxDataEndPtr) //如果指针到缓冲区尾部了
MQTT_TxDataOutPtr = MQTT_TxDataBuf[0]; //指针归位到缓冲区开头
}
}//处理发送缓冲区数据的else if分支结尾
/*-------------------------------------------------------------*/
/* 处理接收缓冲区数据 */
/*-------------------------------------------------------------*/
if(MQTT_RxDataOutPtr != MQTT_RxDataInPtr){ //if成立的话,说明接收缓冲区有数据了
u1_printf("接收到数据:");
/*-----------------------------------------------------*/
/* 处理CONNACK报文 */
/*-----------------------------------------------------*/
//if判断,如果第一个字节是0x20,表示收到的是CONNACK报文
//接着我们要判断第4个字节,看看CONNECT报文是否成功
if(MQTT_RxDataOutPtr[2]==0x20){
switch(MQTT_RxDataOutPtr[5]){
case 0x00 : u1_printf("CONNECT报文成功rn"); //串口输出信息
ConnectPack_flag = 1; //CONNECT报文成功,订阅报文可发
break; //跳出分支case 0x00
case 0x01 : u1_printf("连接已拒绝,不支持的协议版本,准备重启rn"); //串口输出信息
Connect_flag = 0; //Connect_flag置零,重启连接
break; //跳出分支case 0x01
case 0x02 : u1_printf("连接已拒绝,不合格的客户端标识符,准备重启rn"); //串口输出信息
Connect_flag = 0; //Connect_flag置零,重启连接
break; //跳出分支case 0x02
case 0x03 : u1_printf("连接已拒绝,服务端不可用,准备重启rn"); //串口输出信息
Connect_flag = 0; //Connect_flag置零,重启连接
break; //跳出分支case 0x03
case 0x04 : u1_printf("连接已拒绝,无效的用户名或密码,准备重启rn"); //串口输出信息
Connect_flag = 0; //Connect_flag置零,重启连接
break; //跳出分支case 0x04
case 0x05 : u1_printf("连接已拒绝,未授权,准备重启rn"); //串口输出信息
Connect_flag = 0; //Connect_flag置零,重启连接
break; //跳出分支case 0x05
default : u1_printf("连接已拒绝,未知状态,准备重启rn"); //串口输出信息
Connect_flag = 0; //Connect_flag置零,重启连接
break; //跳出分支case default
}
}
//if判断,第一个字节是0x90,表示收到的是SUBACK报文
//接着我们要判断订阅回复,看看是不是成功
else if(MQTT_RxDataOutPtr[2]==0x90){
switch(MQTT_RxDataOutPtr[6]){
case 0x00 :
case 0x01 : u1_printf("订阅成功rn"); //串口输出信息
SubcribePack_flag = 1; //SubcribePack_flag置1,表示订阅报文成功,其他报文可发送
Ping_flag = 0; //Ping_flag清零
TIM3_ENABLE_30S(); //启动30s的PING定时器
TIM2_ENABLE_30S(); //启动30s的上传数据的定时器
break; //跳出分支
default : u1_printf("订阅失败,准备重启rn"); //串口输出信息
Connect_flag = 0; //Connect_flag置零,重启连接
break; //跳出分支
}
}
//if判断,第一个字节是0xD0,表示收到的是PINGRESP报文
else if(MQTT_RxDataOutPtr[2]==0xD0){
u1_printf("PING报文回复rn"); //串口输出信息
if(Ping_flag==1){ //如果Ping_flag=1,表示第一次发送
Ping_flag = 0; //要清除Ping_flag标志
}else if(Ping_flag>1){ //如果Ping_flag>1,表示是多次发送了,而且是2s间隔的快速发送
Ping_flag = 0; //要清除Ping_flag标志
TIM3_ENABLE_30S(); //PING定时器重回30s的时间
}
}
//if判断,如果第一个字节是0x30,表示收到的是服务器发来的推送数据
//我们要提取控制命令
else if((MQTT_RxDataOutPtr[2]==0x30)){
u1_printf("服务器等级0推送rn"); //串口输出信息
MQTT_DealPushdata_Qs0(MQTT_RxDataOutPtr); //处理等级0推送数据
}
MQTT_RxDataOutPtr += BUFF_UNIT; //指针下移
if(MQTT_RxDataOutPtr==MQTT_RxDataEndPtr) //如果指针到缓冲区尾部了
MQTT_RxDataOutPtr = MQTT_RxDataBuf[0]; //指针归位到缓冲区开头
}//处理接收缓冲区数据的else if分支结尾
/*-------------------------------------------------------------*/
/* 处理命令缓冲区数据 */
/*-------------------------------------------------------------*/
if(MQTT_CMDOutPtr != MQTT_CMDInPtr){ //if成立的话,说明命令缓冲区有数据了
u1_printf("命令:%srn",&MQTT_CMDOutPtr[2]); //串口输出信息
if(strstr((char *)MQTT_CMDOutPtr+2,""params":{"powerstate":1}"))
{ //如果搜索到"params":{"PowerSwitch":1}说明服务器下发打开开关1
LED1_ON; //打开LED1
C_Ledflag=1; //手动开灯标志
GPIO_ResetBits(GPIOE, GPIO_Pin_5);
LED1_State(); //判断开关状态,并发布给服务器
}
else if(strstr((char *)MQTT_CMDOutPtr+2,""params":{"powerstate":0}"))
{ //如果搜索到"params":{"PowerSwitch":0}说明服务器下发关闭开关1
C_Ledflag=0; //手动关灯标志
LED1_OFF; //关闭LED1
GPIO_SetBits(GPIOE, GPIO_Pin_5);
LED1_State(); //判断开关状态,并发布给服务器
}
else if(strstr((char *)MQTT_CMDOutPtr+2,""params":{"fanSwitch":1}"))
{ //如果搜索到"params":{"PowerSwitch":0}说明服务器下发关闭开关1
FAN_ON; //打开风扇
FAN_State(); //判断开关状态,并发布给服务器
}
else if(strstr((char *)MQTT_CMDOutPtr+2,""params":{"fanSwitch":0}"))
{ //如果搜索到"params":{"PowerSwitch":0}说明服务器下发关闭开关1
FAN_OFF; //关闭风扇
FAN_State(); //判断开关状态,并发布给服务器
}
else if(strstr((char *)MQTT_CMDOutPtr+2,""params":{"Switch_control":0}"))
{ //如果搜索到"params":{"PowerSwitch":0}说明服务器下发关闭开关1
Window=BUJING_Cotrol(1,3,160); //关闭窗户
Window_State(); //判断开关状态,并发布给服务器
}
else if(strstr((char *)MQTT_CMDOutPtr+2,""params":{"Switch_control":1}"))
{ //如果搜索到"params":{"PowerSwitch":0}说明服务器下发关闭开关1
Window=BUJING_Cotrol(0,3,160); //打开窗户
Window_State(); //判断开关状态,并发布给服务器
}
MQTT_CMDOutPtr += BUFF_UNIT; //指针下移
if(MQTT_CMDOutPtr==MQTT_CMDEndPtr) //如果指针到缓冲区尾部了
MQTT_CMDOutPtr = MQTT_CMDBuf[0]; //指针归位到缓冲区开头
}//处理命令缓冲区数据的else if分支结尾
}//Connect_flag=1的if分支的结尾
/*--------------------------------------------------------------------*/
/* Connect_flag=0同服务器断开了连接,我们要重启连接服务器 */
/*--------------------------------------------------------------------*/
else{
u1_printf("需要连接服务器rn"); //串口输出信息
TIM_Cmd(TIM4,DISABLE); //关闭TIM4
TIM_Cmd(TIM3,DISABLE); //关闭TIM3
WiFi_RxCounter=0; //WiFi接收数据量变量清零
memset(WiFi_RX_BUF,0,WiFi_RXBUFF_SIZE); //清空WiFi接收缓冲区
if(WiFi_Connect_IoTServer()==0){ //如果WiFi连接云服务器函数返回0,表示正确,进入if
u1_printf("建立TCP连接成功rn"); //串口输出信息
Connect_flag = 1; //Connect_flag置1,表示连接成功
WiFi_RxCounter=0; //WiFi接收数据量变量清零
memset(WiFi_RX_BUF,0,WiFi_RXBUFF_SIZE); //清空WiFi接收缓冲区
MQTT_Buff_ReInit(); //重新初始化发送缓冲区
}
}
}
}
/*-------------------------------------------------*/
/*函数名:采集温湿度,并发布给服务器 */
/*参 数:无 */
/*返回值:无 */
/*-------------------------------------------------*/
void TempHumi_State(void)
{
u8 tempdata,humidata;
char temp[256];
DHT11_Read_Data(&tempdata,&humidata); //读取温湿度值
OLED_ShowNum(68,0,tempdata,2,16); //显示温度
OLED_ShowNum(68,3,humidata,2,16); //显示湿度
u1_printf("温度:%d 湿度:%drn",tempdata,humidata);
sprintf(temp,"{"method":"thing.event.property.post","id":"203302322","params":{"CuTemperature":%2d,"CurrentHumidity":%2d},"version":"1.0.0"}",tempdata,humidata); //构建回复湿度温度数据
MQTT_PublishQs0(P_TOPIC_NAME,temp,strlen(temp)); //添加数据,发布给服务器
}
/*-------------------------------------------------*/
/*函数名:控制LED,并发布给服务器 */
/*参 数:无 */
/*返回值:无 */
/*-------------------------------------------------*/
void LED1_State(void)
{
char temp[TBUFF_UNIT]; //定义一个临时缓冲区
if(LED0_IN_STA) sprintf(temp,"{"method":"thing.event.property.post","id":"203302322","params":{"powerstate":1},"version":"1.0.0"}"); //如果LED1是高电平,说明是熄灭状态,需要回复关闭状态给服务器
else sprintf(temp,"{"method":"thing.event.property.post","id":"203302322","params":{"powerstate":0},"version":"1.0.0"}"); //如果LED1是低电平,说明是点亮状态,需要回复打开状态给服务器
MQTT_PublishQs0(P_TOPIC_NAME,temp,strlen(temp)); //添加数据,发布给服务器
}
/*-------------------------------------------------*/
/*函数名:控制电风扇FAN,并发布给服务器 */
/*参 数:无 */
/*返回值:无 */
/*-------------------------------------------------*/
void FAN_State(void)
{
char temp1[256]; //定义一个临时缓冲区
if(FAN_IN_STA) sprintf(temp1,"{"method":"thing.event.property.post","id":"203302323","params":{"fanSwitch":1},"version":"1.0.0"}"); //如果PE8是高电平,说明是风扇是开启状态,需要回复开启状态给服务器
else sprintf(temp1,"{"method":"thing.event.property.post","id":"203302323","params":{"fanSwitch":0},"version":"1.0.0"}"); //如果PE8是低电平,说明是关闭状态,需要回复关闭状态给服务器
MQTT_PublishQs0(P_TOPIC_NAME,temp1,strlen(temp1)); //添加数据,发布给服务器
}
/*-------------------------------------------------*/
/*函数名:检测烟雾浓度是否正常,并发布给服务器 */
/*参 数:无 */
/*返回值:无 */
/*-------------------------------------------------*/
void MQ2_State(void)
{
u16 MQ2;
char temp2[256];
MQ2=Get_Adc_Average(ADC_Channel_1,1,10);
MQ2=MQ2*100/4095;
u1_printf("烟雾浓度:%d rn",MQ2);
sprintf(temp2,"{"method":"thing.event.property.post","id":"203302324","params":{"smokeSensorValue":%d},"version":"1.0.0"}",MQ2); //构建回复湿度温度数据
MQTT_PublishQs0(P_TOPIC_NAME,temp2,strlen(temp2)); //添加数据,发布给服务器
}
/*-------------------------------------------------*/
/*函数名:检测关照强度,并发布给服务器 */
/*参 数:无 */
/*返回值:无 */
/*-------------------------------------------------*/
void Light_State(void)
{
float Light;
char temp3[256];
Light=Get_Adc2_Average(ADC_Channel_5,1,10);
Light=(1-Light/4095)*100;//百分比;
u1_printf("光照强度:%.0f rn",Light);
sprintf(temp3,"{"method":"thing.event.property.post","id":"203302325","params":{"MeasuredIlluminance":%.0f},"version":"1.0.0"}",Light); //构建回复湿度温度数据
MQTT_PublishQs0(P_TOPIC_NAME,temp3,strlen(temp3)); //添加数据,发布给服务器
}
/*-------------------------------------------------*/
/*函数名:检测关照强度,并发布给服务器 */
/*参 数:无 */
/*返回值:无 */
/*-------------------------------------------------*/
void Window_State(void)
{
extern u8 Window;
char temp4[256]; //定义一个临时缓冲区
if(Window==0) sprintf(temp4,"{"method":"thing.event.property.post","id":"203302326","params":{"Switch_control":1},"version":"1.0.0"}"); //窗户是开启状态,需要回复开启状态给服务器
else sprintf(temp4,"{"method":"thing.event.property.post","id":"203302326","params":{"Switch_control":0},"version":"1.0.0"}"); //窗户是关闭状态,需要回复关闭状态给服务器
MQTT_PublishQs0(P_TOPIC_NAME,temp4,strlen(temp4)); //添加数据,发布给服务器
}
wifi.h文件
#ifndef __WIFI_H
#define __WIFI_H
#include "usart2.h" //包含需要的头文件
#define RESET_IO(x) GPIO_WriteBit(GPIOA, GPIO_Pin_4, (BitAction)x) //PA4控制WiFi的复位
#define WiFi_printf u2_printf //串口2控制 WiFi
#define WiFi_RxCounter Usart2_RxCounter //串口2控制 WiFi
#define WiFi_RX_BUF Usart2_RxBuff //串口2控制 WiFi
#define WiFi_RXBUFF_SIZE USART2_RXBUFF_SIZE //串口2控制 WiFi
#define SSID "DHT11" //路由器SSID名称
#define PASS "1234567890" //路由器密码
void WiFi_ResetIO_Init(void);
char WiFi_SendCmd(char *cmd, int timeout);
char WiFi_Reset(int timeout);
char WiFi_JoinAP(int timeout);
char WiFi_Connect_Server(int timeout);
char WiFi_Smartconfig(int timeout);
char WiFi_WaitAP(int timeout);
char WiFi_Connect_IoTServer(void);
#endif
wifi.c文件
这里使用到了ESP8266模块,将数据发到阿里云服务器,使用到AT指令qi统一
#include "stm32f10x.h" //包含需要的头文件
#include "wifi.h" //包含需要的头文件
#include "iic.h" //包含需要的头文件
#include "24c02.h" //包含需要的头文件
#include "delay.h" //包含需要的头文件
#include "usart1.h" //包含需要的头文件
#include "led.h" //包含需要的头文件
#include "mqtt.h" //包含需要的头文件
#include "key.h" //包含需要的头文件
char wifi_mode = 0; //联网模式 0:SSID和密码写在程序里 1:Smartconfig方式用APP发送
/*-------------------------------------------------*/
/*函数名:初始化WiFi的复位IO */
/*参 数:无 */
/*返回值:无 */
/*-------------------------------------------------*/
void WiFi_ResetIO_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure; //定义一个设置IO端口参数的结构体
RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOA , ENABLE); //使能PA端口时钟
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4; //准备设置PA4
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //速率50Mhz
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推免输出方式
GPIO_Init(GPIOA, &GPIO_InitStructure); //设置PA4
RESET_IO(1); //复位IO拉高电平
}
/*-------------------------------------------------*/
/*函数名:WiFi发送设置指令 */
/*参 数:cmd:指令 */
/*参 数:timeout:超时时间(100ms的倍数) */
/*返回值:0:正确 其他:错误 */
/*-------------------------------------------------*/
char WiFi_SendCmd(char *cmd, int timeout)
{
WiFi_RxCounter=0; //WiFi接收数据量变量清零
memset(WiFi_RX_BUF,0,WiFi_RXBUFF_SIZE); //清空WiFi接收缓冲区
WiFi_printf("%srn",cmd); //发送指令
while(timeout--){ //等待超时时间到0
Delay_Ms(100); //延时100ms
if(strstr(WiFi_RX_BUF,"OK")) //如果接收到OK表示指令成功
break; //主动跳出while循环
u1_printf("%d ",timeout); //串口输出现在的超时时间
}
u1_printf("rn"); //串口输出信息
if(timeout<=0)return 1; //如果timeout<=0,说明超时时间到了,也没能收到OK,返回1
else return 0; //反之,表示正确,说明收到OK,通过break主动跳出while
}
/*-------------------------------------------------*/
/*函数名:WiFi加入路由器指令 */
/*参 数:timeout:超时时间(1s的倍数) */
/*返回值:0:正确 其他:错误 */
/*-------------------------------------------------*/
char WiFi_JoinAP(int timeout)
{
WiFi_RxCounter=0; //WiFi接收数据量变量清零
memset(WiFi_RX_BUF,0,WiFi_RXBUFF_SIZE); //清空WiFi接收缓冲区
WiFi_printf("AT+CWJAP="%s","%s"rn",SSID,PASS); //发送指令
while(timeout--){ //等待超时时间到0
Delay_Ms(1000); //延时1s
if(strstr(WiFi_RX_BUF,"nrnOK")) //如果接收到WIFI GOT IP表示成功,但是正点原子输出形式是 WIFI GOT IPnrnOK,老版本输出形式是WIFI GOT IPrnrnOK,为了通用,此处这样设置
break; //主动跳出while循环
u1_printf("%d ",timeout); //串口输出现在的超时时间
}
u1_printf("rn"); //串口输出信息
if(timeout<=0)return 1; //如果timeout<=0,说明超时时间到了,也没能收到WIFI GOT IP,返回1
return 0; //正确,返回0
}
/*-------------------------------------------------*/
/*函数名:连接TCP服务器,并进入透传模式 */
/*参 数:timeout: 超时时间(100ms的倍数) */
/*返回值:0:正确 其他:错误 */
/*-------------------------------------------------*/
char WiFi_Connect_Server(int timeout)
{
WiFi_RxCounter=0; //WiFi接收数据量变量清零
memset(WiFi_RX_BUF,0,WiFi_RXBUFF_SIZE); //清空WiFi接收缓冲区
WiFi_printf("AT+CIPSTART="TCP","%s",%drn",ServerIP,ServerPort);//发送连接服务器指令
while(timeout--){ //等待超时与否
Delay_Ms(100); //延时100ms
if(strstr(WiFi_RX_BUF ,"CONNECT")) //如果接受到CONNECT表示连接成功
break; //跳出while循环
if(strstr(WiFi_RX_BUF ,"CLOSED")) //如果接受到CLOSED表示服务器未开启
return 1; //服务器未开启返回1
if(strstr(WiFi_RX_BUF ,"ALREADY CONNECTED"))//如果接受到ALREADY CONNECTED已经建立连接
return 2; //已经建立连接返回2
u1_printf("%d ",timeout); //串口输出现在的超时时间
}
u1_printf("rn"); //串口输出信息
if(timeout<=0)return 3; //超时错误,返回3
else //连接成功,准备进入透传
{
u1_printf("连接服务器成功,准备进入透传rn"); //串口显示信息
WiFi_RxCounter=0; //WiFi接收数据量变量清零
memset(WiFi_RX_BUF,0,WiFi_RXBUFF_SIZE); //清空WiFi接收缓冲区
WiFi_printf("AT+CIPSENDrn"); //发送进入透传指令
while(timeout--){ //等待超时与否
Delay_Ms(100); //延时100ms
if(strstr(WiFi_RX_BUF,"rnOKrnrn>")) //如果成立表示进入透传成功
break; //跳出while循环
u1_printf("%d ",timeout); //串口输出现在的超时时间
}
if(timeout<=0)return 4; //透传超时错误,返回4
}
return 0; //成功返回0
}
/*-------------------------------------------------*/
/*函数名:WiFi连接服务器 */
/*参 数:无 */
/*返回值:0:正确 其他:错误 */
/*-------------------------------------------------*/
char WiFi_Connect_IoTServer(void)
{
u1_printf("准备复位模块rn"); //串口提示数据
if(WiFi_Reset(50)){ //复位,100ms超时单位,总计5s超时时间
u1_printf("复位失败,准备重启rn"); //返回非0值,进入if,串口提示数据
return 1; //返回1
}else u1_printf("复位成功rn"); //串口提示数据
u1_printf("准备设置STA模式rn"); //串口提示数据
if(WiFi_SendCmd("AT+CWMODE=1",50)){ //设置STA模式,100ms超时单位,总计5s超时时间
u1_printf("设置STA模式失败,准备重启rn"); //返回非0值,进入if,串口提示数据
return 2; //返回2
}else u1_printf("设置STA模式成功rn"); //串口提示数据
if(wifi_mode==0){ //如果联网模式=0:SSID和密码写在程序里
u1_printf("准备取消自动连接rn"); //串口提示数据
if(WiFi_SendCmd("AT+CWAUTOCONN=0",50)){ //取消自动连接,100ms超时单位,总计5s超时时间
u1_printf("取消自动连接失败,准备重启rn"); //返回非0值,进入if,串口提示数据
return 3; //返回3
}else u1_printf("取消自动连接成功rn"); //串口提示数据
u1_printf("准备连接路由器rn"); //串口提示数据
if(WiFi_JoinAP(30)){ //连接路由器,1s超时单位,总计30s超时时间
u1_printf("连接路由器失败,准备重启rn"); //返回非0值,进入if,串口提示数据
return 4; //返回4
}else u1_printf("连接路由器成功rn"); //串口提示数据
}else{ //如果联网模式=1:Smartconfig方式,用APP发送
if(0){ //如果此时K2是按下的
u1_printf("准备设置自动连接rn"); //串口提示数据
if(WiFi_SendCmd("AT+CWAUTOCONN=1",50)){ //设置自动连接,100ms超时单位,总计5s超时时间
u1_printf("设置自动连接失败,准备重启rn"); //返回非0值,进入if,串口提示数据
return 3; //返回3
}else u1_printf("设置自动连接成功rn"); //串口提示数据
u1_printf("准备开启Smartconfigrn"); //串口提示数据
if(WiFi_SendCmd("AT+CWSTARTSMART",50)){ //开启Smartconfig,100ms超时单位,总计5s超时时间
u1_printf("开启Smartconfig失败,准备重启rn");//返回非0值,进入if,串口提示数据
return 4; //返回4
}else u1_printf("开启Smartconfig成功rn"); //串口提示数据
u1_printf("请使用APP软件传输密码rn"); //串口提示数据
if(WiFi_Smartconfig(60)){ //APP软件传输密码,1s超时单位,总计60s超时时间
u1_printf("传输密码失败,准备重启rn"); //返回非0值,进入if,串口提示数据
return 5; //返回5
}else u1_printf("传输密码成功rn"); //串口提示数据
u1_printf("准备关闭Smartconfigrn"); //串口提示数据
if(WiFi_SendCmd("AT+CWSTOPSMART",50)){ //关闭Smartconfig,100ms超时单位,总计5s超时时间
u1_printf("关闭Smartconfig失败,准备重启rn");//返回非0值,进入if,串口提示数据
return 6; //返回6
}else u1_printf("关闭Smartconfig成功rn"); //串口提示数据
}else{ //反之,此时K2是没有按下
u1_printf("等待连接路由器rn"); //串口提示数据
if(WiFi_WaitAP(30)){ //等待连接路由器,1s超时单位,总计30s超时时间
u1_printf("连接路由器失败,准备重启rn"); //返回非0值,进入if,串口提示数据
return 7; //返回7
}else u1_printf("连接路由器成功rn"); //串口提示数据
}
}
u1_printf("准备设置透传rn"); //串口提示数据
if(WiFi_SendCmd("AT+CIPMODE=1",50)){ //设置透传,100ms超时单位,总计5s超时时间
u1_printf("设置透传失败,准备重启rn"); //返回非0值,进入if,串口提示数据
return 8; //返回8
}else u1_printf("设置透传成功rn"); //串口提示数据
u1_printf("准备关闭多路连接rn"); //串口提示数据
if(WiFi_SendCmd("AT+CIPMUX=0",50)){ //关闭多路连接,100ms超时单位,总计5s超时时间
u1_printf("关闭多路连接失败,准备重启rn"); //返回非0值,进入if,串口提示数据
return 9; //返回9
}else u1_printf("关闭多路连接成功rn"); //串口提示数据
u1_printf("准备连接服务器rn"); //串口提示数据
if(WiFi_Connect_Server(100)){ //连接服务器,100ms超时单位,总计10s超时时间
u1_printf("连接服务器失败,准备重启rn"); //返回非0值,进入if,串口提示数据
return 10; //返回10
}else u1_printf("连接服务器成功rn"); //串口提示数据
return 0; //正确返回0
}
由于是基于MQTT协议的
mqtt.c文件
void MQTT_Buff_Init(void)
{
MQTT_RxDataInPtr=MQTT_RxDataBuf[0]; //指向发送缓冲区存放数据的指针归位
MQTT_RxDataOutPtr=MQTT_RxDataInPtr; //指向发送缓冲区读取数据的指针归位
MQTT_RxDataEndPtr=MQTT_RxDataBuf[R_NUM-1]; //指向发送缓冲区结束的指针归位
MQTT_TxDataInPtr=MQTT_TxDataBuf[0]; //指向发送缓冲区存放数据的指针归位
MQTT_TxDataOutPtr=MQTT_TxDataInPtr; //指向发送缓冲区读取数据的指针归位
MQTT_TxDataEndPtr=MQTT_TxDataBuf[T_NUM-1]; //指向发送缓冲区结束的指针归位
MQTT_CMDInPtr=MQTT_CMDBuf[0]; //指向命令缓冲区存放数据的指针归位
MQTT_CMDOutPtr=MQTT_CMDInPtr; //指向命令缓冲区读取数据的指针归位
MQTT_CMDEndPtr=MQTT_CMDBuf[C_NUM-1]; //指向命令缓冲区结束的指针归位
Connect_flag = 0; //各个参数清零
Ping_flag = ConnectPack_flag = SubcribePack_flag = 0; //各个参数清零
}
/*----------------------------------------------------------*/
/*函数名:重新初始化接收,发送,命令缓冲区 以及各状态参数 */
/*参 数:无 */
/*返回值:无 */
/*----------------------------------------------------------*/
void MQTT_Buff_ReInit(void)
{
unsigned char *MQTT_TxDatatempPtr; //指向发送缓冲区位置的临时指针
if(MQTT_TxDataOutPtr != MQTT_TxDataInPtr){ //if成立的话,说明发送缓冲区有数据了
MQTT_TxDataOutPtr = MQTT_TxDataInPtr; //OUT指针指向IN指针
if(MQTT_TxDataOutPtr==MQTT_TxDataBuf[0]){ //如果,现在OUT指针在缓冲区顶部,进入if
MQTT_TxDataOutPtr =MQTT_TxDataBuf[T_NUM-4];//重定位OUT指针
}else if(MQTT_TxDataOutPtr==MQTT_TxDataBuf[1]){//如果,现在OUT指针在缓冲区顶部下一个单元,进入if
MQTT_TxDataOutPtr =MQTT_TxDataBuf[T_NUM-3];//重定位OUT指针
}else if(MQTT_TxDataOutPtr==MQTT_TxDataBuf[2]){//如果,现在OUT指针在缓冲区顶部下两个单元,进入if
MQTT_TxDataOutPtr =MQTT_TxDataBuf[T_NUM-2];//重定位OUT指针
}else{
MQTT_TxDataOutPtr -= BUFF_UNIT; //OUT指针上移一个单元
MQTT_TxDataOutPtr -= BUFF_UNIT; //OUT指针上移一个单元
MQTT_TxDataOutPtr -= BUFF_UNIT; //OUT指针上移一个单元
}
MQTT_TxDatatempPtr = MQTT_TxDataInPtr; //将当前IN指针的位置暂存在temp指针中
MQTT_TxDataInPtr = MQTT_TxDataOutPtr; //IN指针指向当前OUT指针
MQTT_ConectPack(); //发送缓冲区添加连接报文
MQTT_Subscribe(S_TOPIC_NAME,0); //发送缓冲区添加订阅topic,等级0
MQTT_TxDataInPtr = MQTT_TxDatatempPtr; //IN指针通过temp指针,返回原来的位置
}else{ //反之,说明发送缓冲区没有数据
MQTT_ConectPack(); //发送缓冲区添加连接报文
MQTT_Subscribe(S_TOPIC_NAME,0); //发送缓冲区添加订阅topic,等级0
}
Ping_flag = ConnectPack_flag = SubcribePack_flag = 0; //各个参数清零
}
/*----------------------------------------------------------*/
/*函数名:阿里云初始化参数,得到客户端ID,用户名和密码 */
/*参 数:无 */
/*返回值:无 */
/*----------------------------------------------------------*/
void AliIoT_Parameter_Init(void)
{
char temp[128]; //计算加密的时候,临时使用的缓冲区
memset(ClientID,128,0); //客户端ID的缓冲区全部清零
sprintf(ClientID,"%s|securemode=3,signmethod=hmacsha1|",DEVICENAME); //构建客户端ID,并存入缓冲区
ClientID_len = strlen(ClientID); //计算客户端ID的长度
memset(Username,128,0); //用户名的缓冲区全部清零
sprintf(Username,"%s&%s",DEVICENAME,PRODUCTKEY); //构建用户名,并存入缓冲区
Username_len = strlen(Username); //计算用户名的长度
memset(temp,128,0); //临时缓冲区全部清零
sprintf(temp,"clientId%sdeviceName%sproductKey%s",DEVICENAME,DEVICENAME,PRODUCTKEY); //构建加密时的明文
utils_hmac_sha1(temp,strlen(temp),Passward,DEVICESECRE,DEVICESECRE_LEN); //以DeviceSecret为秘钥对temp中的明文,进行hmacsha1加密,结果就是密码,并保存到缓冲区中
Passward_len = strlen(Passward); //计算用户名的长度
memset(ServerIP,128,0);
sprintf(ServerIP,"%s.iot-as-mqtt.cn-shanghai.aliyuncs.com",PRODUCTKEY); //构建服务器域名
ServerPort = 1883; //服务器端口号1883
u1_printf("服 务 器:%s:%drn",ServerIP,ServerPort); //串口输出调试信息
u1_printf("客户端ID:%srn",ClientID); //串口输出调试信息
u1_printf("用 户 名:%srn",Username); //串口输出调试信息
u1_printf("密 码:%srn",Passward); //串口输出调试信息
}
项目代码过多,文章代码只是部分,主要AT指令和MQTT协议的数据处理
最后
以上就是快乐项链为你收集整理的STM32+阿里云+传感器小项目的全部内容,希望文章能够帮你解决STM32+阿里云+传感器小项目所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
发表评论 取消回复