Bootstrap

stm32+AT指令+ESP8266接入华为云物联网平台并完成属性上报与命令响应

(示例:stm32f103c8t6+esp01s串口AT指令模式)

流程简介:

一、 MQTT透传AT固件烧录

      本期教程使用的是stm32+AT指令控制esp01s接入华为云联网平台完成数据上报与命令处理,在先前如果使用Arduino开发过后的esp8266可能已经无法使用AT指令,或者出厂固件不支持全部的MQTT功能,这里我们首先对esp8266进行MQTT固件烧录,如果平时一直在使用原厂固件开发,可跳过第一步,若AT指令无效再按第一步操作。

1. 官网下载烧录工具和固件

烧录工具链接:https://docs.ai-thinker.com/tools

固件链接:

https://docs.ai-thinker.com/%E5%9B%BA%E4%BB%B6%E6%B1%87%E6%80%BB

大家根据自己需要选择相应的版本,我使用的是esp01s,下载的是1471号

2. 打开固件烧录工具

连接usb转ttl模块,点击START,然后保持esp01s的IO0拉低后重新上电

 

打开串口助手,按下复位,输入命令”AT”,注意要取消勾选十六进制发送,勾选发送新行,然后检查串口打印工作状况,下图表示正常:

3. 串口调试

我们即将用到的AT指令可以从官方手册中学习:

https://docs.ai-thinker.com/%E5%9B%BA%E4%BB%B6%E6%B1%87%E6%80%BB

1) AT+CWMODE=1

设置模块为STA模式

2) AT+CWJAP="填写wifi名称","填写WiFi密码"

连接WiFi,需略微等待片刻,等显示连接后再进行下面的操作

3) AT+MQTTUSERCFG=0,1,"NULL","填写用户名","填写密码",0,0,""

设置MQTT的登陆用户名与密码

用户名与密码为华为云MQTT三元组的相关信息,可以在下面的网站生成:https://iot-tool.obs-website.cn-north-4.myhuaweicloud.com/

具体可以参考之前的文章

4) AT+MQTTCLIENTID=0,"填写ClientID"

设置MQTT的ClientID,ClientID华为云MQTT三元组的相关信息,在上一步骤中获取的三元组中查看

5) AT+MQTTCONN=0,"填写MQTT接入的地址",1883,1

设置MQTT接入地址,华为云物联网平台的地址,比如iot-mqtts.cn-north-4.myhuaweicloud.com,具体参考控制台首页的平台接入地址

6) AT+MQTTSUB=0,"订阅的主题tpoic",1

完成属性上报的话需要先订阅设备属性上报的主题,此处可填写为:

$oc/devices/填写设备ID/sys/properties/report

7) AT+MQTTPUB=0," 订阅的主题tpoic ","上报的json数据",0,0

订阅的主题:

$oc/devices/填写设备ID/sys/properties/report

上报的json数据:

{\"services\":[{\"service_id\":\"填写服务ID\"\,\"properties\":{\"填写设备属性\": 填写属性数据值}}]}

建议大家完成这步时,先使用MQTT.fx进行设备属性上报测试,确保数据无误正常上传,具体参数因个人的配置而异

在完成上面的每一步时,系统都会打印一个OK,正常效果如下:

    

    此时我们可以查看华为云物联网平台的设备属性以及消息记录:

8) AT+MQTTSUB=0,"订阅的主题topic ",1

此时我们在这填写的是平台下发命令的主题:$oc/devices/填写设备ID/sys/commands/#

9)  进入华为云物联网平台的->监控运维->在线调试,完成命令下发

服务ID与下发命令的创建方法参考以前的文章:

此时设备会接收到下发的命令:

接收到命令后我们需要对命令进行响应,即下一步的步骤

10) 特别注意!!!下面两步的操作时间建议接收到命令后的20s以内,超时后平台会认定为同步命令设备响应异常!

11) AT+MQTTSUB=0,"$oc/devices/填写设备ID /sys/commands/response/request_id=填写接收到的request_id ",1

订阅设备响应平台命令主题,其中接收到的request_id在上图接收到的数据中,我们需要把它拿出来放到新订阅的tpoic的对应位置里

12)  AT+MQTTPUB=0,"填写上面订阅的含request_id的完整主题","填写响应值,可以为空",0,0

例如:AT+MQTTPUB=0,"$oc/devices/填写设备id/sys/commands/response/request_id=填写request_id ","",0,0

13) 完成上面的三步操作后,我们可以看到华为云物联网平台已经完成了命令的接收与响应

此时,我们的设备属性上报与设备命令的接收响应功能就已经完成了,接下来我们完成STM32代替串口助手完成对esp8266的上述控制。

二、 STM32编程

1. 打开STM32CubeMX创建工程

(1)配置串口与LED的GPIO

(2)打开串口并使能串口中断

2. 打开工程,编译,下载,检查无误

3. 串口收发

1) 添加头文件:

#include <stdio.h>
#include <string.h>

2) printf重定向

//支持printf函数串口发送
#if 1
#pragma import(__use_no_semihosting)
struct __FILE
{
    int handle;
};
FILE __stdout;
void _sys_exit(int x)//避免半主机模式
{
    x=x;
}
int fputc(int ch,FILE *f)
{
    while((USART1->SR&0X40)==0);//循环发送,直到发送完毕
    USART1->DR=(uint8_t)ch;
    return ch;
}
#endif
 

4. 发送测试

-------------------------------
-------------------------------
//主函数中插入
  while (1)
  {
    /* USER CODE END WHILE */
      printf("Hello esp8266\r\n");
      delay(4000);
    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}
​
void delay(int t)
{
  int i,j=0;
  for(i=0;i<5000;i++)
    for(j=0;j<t;j++);
}
 

5. 接收测试

​(1)main.c中添加下列声明

char R_data[1024];      //定义结束数据的缓冲区
uint8_t R_length=0;      //定义结束数据的长度
uint8_t R_i=0;        //由于遍历缓冲区数据
 

(2)重写串口接收中断回调函数

//重写串口接收中断回调函数
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
  char analysis_Str[256];
  if(huart->Instance == USART1)
  {
    HAL_UART_Receive_IT(&huart1,&aRxBuffer1,1);
    // save char from uart receive
    if(aRxBuffer1!='\n')
      R_data[R_length++]=aRxBuffer1;
                 else 
     for(R_i=0;R_i<R_length;R_i++)
        printf("%c",R_data[R_i]);
   }
}
 

(3)在串口初始化后使能串口中断

HAL_UART_Receive_IT(&huart1,&aRxBuffer1,1); //使能串口接收中断
 

运行测试:

6.  完善主要接入华为云物联网平台的相关代码

(1)接入华为云初始化

void HuaweiIot_init(void)
{
    uint8_t i=0;
    for(i=0;i<10;i++)
    {
        if(at_start_flag==1)
        {
            AT_write("AT\r\n");
            printf("AT+RST\r\n");
            delay(1000);
            AT_write("AT\r\n");
            AT_write("AT+CWMODE=1\r\n");
            AT_write("AT+CWJAP=\"wifi名称\",\"wifi密码\"\r\n");
            AT_write("AT+MQTTUSERCFG=0,1,\"NULL\",\"三元组之Username\",\"三元组之password\",0,0,\"\"\r\n");
            AT_write("AT+MQTTCLIENTID=0,\"三元组之ClientID\"\r\n");
            AT_write("AT+MQTTCONN=0,\"MQTT接入地址\",1883,1\r\n");
            AT_write("AT+MQTTSUB=0,\"$oc/devices/设备ID/sys/properties/report\",1\r\n");
            AT_write("AT+MQTTSUB=0,\"$oc/devices/设备ID/sys/commands/#\",1\r\n");
            AT_write("AT+MQTTSUB=0,\"$oc/devices/设备ID/sys/commands/response/#\",1\r\n");
            break;
        }    
        else
        {
            printf("等待模块就绪...\r\n");
            delay(1000);
        }            
    }
}
 

 (2) AT命令发送

void AT_write(char atstring[1024])//阻塞等待OK
{
    printf("%s",atstring);
    while(1)
    {
            if(R_data[0]=='O'&&R_data[1]=='K') break;
            else if(R_data[0]=='b'&&R_data[1]=='u'&&R_data[2]=='s'&&R_data[3]=='y')
            {
            }
            else delay(50);
    }
    R_data[0]=R_data[1]=0;
    delay(50);
}
 

(3) 属性上报

void HuaweiIot_publish(void)
{
    char pubtemp[256];
    if(at_start_flag==1)
    {    
        sprintf(pubtemp,"AT+MQTTPUB=0,\"$oc/devices/设备ID/sys/properties/report\",\"{\\\"services\\\":[{\\\"service_id\\\":\\\"服务ID\\\"\\,\\\"properties\\\":{\\\"属性\\\": %d}}]}\",0,0\r\n",hometemp++);
        AT_write(pubtemp);
    }
}
 

(4)串口接收中断回调

//重写串口接收中断回调函数
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
    char analysis_Str[256];
    if(huart->Instance == USART1)
    {
        HAL_UART_Receive_IT(&huart1,&aRxBuffer1,1);
        // save char from uart receive
        if(aRxBuffer1!='\n')
            R_data[R_length++]=aRxBuffer1;
        else
        {
            if(R_length>=5)
            {
                    if(R_data[0]=='r'&&R_data[1]=='e'&&R_data[2]=='a'&&R_data[3]=='d'&&R_data[4]=='y')
                    {
                            at_start_flag=1;
                            printf("检测到ready\r\n");
                            memset(R_data,0,sizeof(R_data));//清空缓存区
                    }
            }
                //我的接收数据长度为201,下面采用JSON字符串硬解析的方式,具体下标请根据自己实际接收的参数处理
    //+MQTTSUBRECV:0,"$oc/devices/61fb2d7fde9933029be5ff9e_esp8266_test01/sys/commands/request_id=4152fb5d-e5ae-4b89-b39d-283ba59cf033",68,{"paras":{"led":1},"service_id":"Dev_data","command_name":"Control"}
            if(R_length>200)
            {
                    strncpy(analysis_Str, R_data, 12);//提取出“+MQTTSUBRECV”
                    if(strcmp(analysis_Str,"+MQTTSUBRECV")==0)
                    {
                            memset(analysis_Str,0,sizeof(analysis_Str));//清空缓存区
                            //printf("MQTT命令接收头正确\r\n");
                            strncpy(request_id, R_data+92, 36);        //提取出request_id
                            //printf("request_id=%s\r\n",request_id);
                            strncpy(analysis_Str, R_data+135, 5);    //提取出"paras"
                            //printf("paras=%s\r\n",analysis_Str);
                            if(strcmp(analysis_Str,"paras")==0)        //有效参数体
                            {
                                    memset(analysis_Str,0,sizeof(analysis_Str));//清空缓存区
                                    strncpy(analysis_Str, R_data+144,3);    //提取出"led"
                                    //printf("att is %s\r\n",analysis_Str);
                                    if(strcmp(analysis_Str,"led")==0)
                                    {
                                            //printf("led set %c",R_data[149]);
                                            memset(analysis_Str,0,sizeof(analysis_Str));//清空缓存区
                                            if(R_data[149]=='0')
                                            {
                                                    //printf("关灯\r\n");
                                                    printf("AT+MQTTPUB=0,\"$oc/devices/设备ID/sys/commands/response/request_id=%s\",\"\",0,0\r\n",request_id);
                                                    HAL_GPIO_WritePin(GPIOC, GPIO_PIN_13, GPIO_PIN_SET);
                                                    memset(analysis_Str,0,sizeof(analysis_Str));//清空缓存区    
                                            }
                                           else if(R_data[149]=='1')
                                            {
                                                    //printf("开灯\r\n");
                                                    printf("AT+MQTTPUB=0,\"$oc/devices/设备ID/sys/commands/response/request_id=%s\",\"\",0,0\r\n",request_id);
                                                    memset(analysis_Str,0,sizeof(analysis_Str));//清空缓存区        
                                                    HAL_GPIO_WritePin(GPIOC, GPIO_PIN_13, GPIO_PIN_RESET);
                                            }                                        
                                    }        
                            }
                    }
                    else
                    {
                        //printf("MQTT命令接收头异常:%s\r\n",analysis_Str);
                        memset(analysis_Str,0,sizeof(analysis_Str));//清空缓存区
                    }
            }
            //else printf("MQTT命令长度异常,长度:%d",R_length);            
            R_length=0;
        }
    }
}

最终效果:

上述教程完成基本功能的核心代码已经给出,如果需要keil项目文件,可公-众,号搜索“IOT趣制作”,回复关键字“8266at”即可。

 

;