附上.c文件
#include "mqtt.h" //包含需要的头文件
unsigned char MQTT_RxDataBuf[R_NUM][RBUFF_UNIT]; //数据的接收缓冲区,所有服务器发来的数据,存放在该缓冲区,缓冲区第一个字节存放数据长度
unsigned char *MQTT_RxDataInPtr; //指向接收缓冲区存放数据的位置
unsigned char *MQTT_RxDataOutPtr; //指向接收缓冲区读取数据的位置
unsigned char *MQTT_RxDataEndPtr; //指向接收缓冲区结束的位置
unsigned char MQTT_TxDataBuf[T_NUM][TBUFF_UNIT]; //数据的发送缓冲区,所有发往服务器的数据,存放在该缓冲区,缓冲区第一个字节存放数据长度
unsigned char *MQTT_TxDataInPtr; //指向发送缓冲区存放数据的位置
unsigned char *MQTT_TxDataOutPtr; //指向发送缓冲区读取数据的位置
unsigned char *MQTT_TxDataEndPtr; //指向发送缓冲区结束的位置
unsigned char MQTT_CMDBuf[C_NUM][CBUFF_UNIT]; //命令数据的接收缓冲区
unsigned char *MQTT_CMDInPtr; //指向命令缓冲区存放数据的位置
unsigned char *MQTT_CMDOutPtr; //指向命令缓冲区读取数据的位置
unsigned char *MQTT_CMDEndPtr; //指向命令缓冲区结束的位置
char ClientID[128]; //存放客户端ID的缓冲区
int ClientID_len; //存放客户端ID的长度
char Username[128]; //存放用户名的缓冲区
int Username_len; //存放用户名的长度
char Passward[128]; //存放密码的缓冲区
int Passward_len; //存放密码的长度
char ServerIP[128]; //存放服务器IP或是域名
int ServerPort; //存放服务器的端口号
int Fixed_len; //固定报头长度
int Variable_len; //可变报头长度
int Payload_len; //有效负荷长度
unsigned char temp_buff[TBUFF_UNIT]; //临时缓冲区,构建报文用
char Ping_flag; //ping报文状态 0:正常状态,等待计时时间到,发送Ping报文
//ping报文状态 1:Ping报文已发送,当收到 服务器回复报文的后 将1置为0
char Connect_flag; //同服务器连接状态 0:还没有连接服务器 1:连接上服务器了
char ConnectPack_flag; //CONNECT报文状态 1:CONNECT报文成功
char SubcribePack_flag; //订阅报文状态 1:订阅报文成功
/*----------------------------------------------------------*/
/*函数名:初始化接收,发送,命令数据的 缓冲区 以及各状态参数 */
/*参 数:无 */
/*返回值:无 */
/*----------------------------------------------------------*/
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]; //指向命令缓冲区结束的指针归位
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,0,128); //客户端ID的缓冲区全部清零
sprintf(ClientID,"%s|securemode=3,signmethod=hmacsha1|",DEVICENAME); //构建客户端ID,并存入缓冲区
ClientID_len = strlen(ClientID); //计算客户端ID的长度
memset(Username,0,128); //用户名的缓冲区全部清零
sprintf(Username,"%s&%s",DEVICENAME,PRODUCTKEY); //构建用户名,并存入缓冲区
Username_len = strlen(Username); //计算用户名的长度
memset(temp,0,128); //临时缓冲区全部清零
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,0,128);
sprintf(ServerIP,"%s.iot-as-mqtt.cn-shanghai.aliyuncs.com",PRODUCTKEY); //构建服务器域名
ServerPort = 1883; //服务器端口号1883
u1_printf("服 务 器:%s:%d\r\n",ServerIP,ServerPort); //串口输出调试信息
u1_printf("客户端ID:%s\r\n",ClientID); //串口输出调试信息
u1_printf("用 户 名:%s\r\n",Username); //串口输出调试信息
u1_printf("密 码:%s\r\n",Passward); //串口输出调试信息
}
/*----------------------------------------------------------*/
/*函数名:连接服务器报文 */
/*参 数:无 */
/*返回值:无 */
/*----------------------------------------------------------*/
void MQTT_ConectPack(void)
{
int temp,Remaining_len;
Fixed_len = 1; //连接报文中,固定报头长度暂时先=1
Variable_len = 10; //连接报文中,可变报头长度=10
Payload_len = 2 + ClientID_len + 2 + Username_len + 2 + Passward_len; //连接报文中,负载长度
Remaining_len = Variable_len + Payload_len; //剩余长度=可变报头长度+负载长度
temp_buff[0]=0x10; //固定报头第1个字节 :固定0x01
do{ //循环处理固定报头中的剩余长度字节,字节量根据剩余字节的真实长度变化
temp = Remaining_len%128; //剩余长度取余128
Remaining_len = Remaining_len/128; //剩余长度取整128
if(Remaining_len>0)
temp |= 0x80; //按协议要求位7置位
temp_buff[Fixed_len] = temp; //剩余长度字节记录一个数据
Fixed_len++; //固定报头总长度+1
}while(Remaining_len>0); //如果Remaining_len>0的话,再次进入循环
temp_buff[Fixed_len+0]=0x00; //可变报头第1个字节 :固定0x00
temp_buff[Fixed_len+1]=0x04; //可变报头第2个字节 :固定0x04
temp_buff[Fixed_len+2]=0x4D; //可变报头第3个字节 :固定0x4D
temp_buff[Fixed_len+3]=0x51; //可变报头第4个字节 :固定0x51
temp_buff[Fixed_len+4]=0x54; //可变报头第5个字节 :固定0x54
temp_buff[Fixed_len+5]=0x54; //可变报头第6个字节 :固定0x54
temp_buff[Fixed_len+6]=0x04; //可变报头第7个字节 :固定0x04
temp_buff[Fixed_len+7]=0xC2; //可变报头第8个字节 :使能用户名和密码校验,不使用遗嘱,不保留会话
temp_buff[Fixed_len+8]=0x00; //可变报头第9个字节 :保活时间高字节 0x00
temp_buff[Fixed_len+9]=0x64; //可变报头第10个字节:保活时间高字节 0x64 100s
/* CLIENT_ID */
temp_buff[Fixed_len+10] = ClientID_len/256; //客户端ID长度高字节
temp_buff[Fixed_len+11] = ClientID_len%256; //客户端ID长度低字节
memcpy(&temp_buff[Fixed_len+12],ClientID,ClientID_len); //复制过来客户端ID字串
/* 用户名 */
temp_buff[Fixed_len+12+ClientID_len] = Username_len/256; //用户名长度高字节
temp_buff[Fixed_len+13+ClientID_len] = Username_len%256; //用户名长度低字节
memcpy(&temp_buff[Fixed_len+14+ClientID_len],Username,Username_len); //复制过来用户名字串
/* 密码 */
temp_buff[Fixed_len+14+ClientID_len+Username_len] = Passward_len/256; //密码长度高字节
temp_buff[Fixed_len+15+ClientID_len+Username_len] = Passward_len%256; //密码长度低字节
memcpy(&temp_buff[Fixed_len+16+ClientID_len+Username_len],Passward,Passward_len); //复制过来密码字串
TxDataBuf_Deal(temp_buff, Fixed_len + Variable_len + Payload_len); //加入发送数据缓冲区
}
/*----------------------------------------------------------*/
/*函数名:SUBSCRIBE订阅topic报文 */
/*参 数:QoS:订阅等级 */
/*参 数:topic_name:订阅topic报文名称 */
/*返回值:无 */
/*----------------------------------------------------------*/
void MQTT_Subscribe(char *topic_name, int QoS)
{
Fixed_len = 2; //SUBSCRIBE报文中,固定报头长度=2
Variable_len = 2; //SUBSCRIBE报文中,可变报头长度=2
Payload_len = 2 + strlen(topic_name) + 1; //计算有效负荷长度 = 2字节(topic_name长度)+ topic_name字符串的长度 + 1字节服务等级
temp_buff[0]=0x82; //第1个字节 :固定0x82
temp_buff[1]=Variable_len + Payload_len; //第2个字节 :可变报头+有效负荷的长度
temp_buff[2]=0x00; //第3个字节 :报文标识符高字节,固定使用0x00
temp_buff[3]=0x01; //第4个字节 :报文标识符低字节,固定使用0x01
temp_buff[4]=strlen(topic_name)/256; //第5个字节 :topic_name长度高字节
temp_buff[5]=strlen(topic_name)%256; //第6个字节 :topic_name长度低字节
memcpy(&temp_buff[6],topic_name,strlen(topic_name)); //第7个字节开始 :复制过来topic_name字串
temp_buff[6+strlen(topic_name)]=QoS; //最后1个字节:订阅等级
TxDataBuf_Deal(temp_buff, Fixed_len + Variable_len + Payload_len); //加入发送数据缓冲区
}
/*----------------------------------------------------------*/
/*函数名:PING报文,心跳包 */
/*参 数:无 */
/*返回值:无 */
/*----------------------------------------------------------*/
void MQTT_PingREQ(void)
{
temp_buff[0]=0xC0; //第1个字节 :固定0xC0
temp_buff[1]=0x00; //第2个字节 :固定0x00
TxDataBuf_Deal(temp_buff, 2); //加入数据到缓冲区
}
/*----------------------------------------------------------*/
/*函数名:等级0 发布消息报文 */
/*参 数:topic_name:topic名称 */
/*参 数:data:数据 */
/*参 数:data_len:数据长度 */
/*返回值:无 */
/*----------------------------------------------------------*/
void MQTT_PublishQs0(char *topic, char *data, int data_len)
{
int temp,Remaining_len;
Fixed_len = 1; //固定报头长度暂时先等于:1字节
Variable_len = 2 + strlen(topic); //可变报头长度:2字节(topic长度)+ topic字符串的长度
Payload_len = data_len; //有效负荷长度:就是data_len
Remaining_len = Variable_len + Payload_len; //剩余长度=可变报头长度+负载长度
temp_buff[0]=0x30; //固定报头第1个字节 :固定0x30
do{ //循环处理固定报头中的剩余长度字节,字节量根据剩余字节的真实长度变化
temp = Remaining_len%128; //剩余长度取余128
Remaining_len = Remaining_len/128; //剩余长度取整128
if(Remaining_len>0)
temp |= 0x80; //按协议要求位7置位
temp_buff[Fixed_len] = temp; //剩余长度字节记录一个数据
Fixed_len++; //固定报头总长度+1
}while(Remaining_len>0); //如果Remaining_len>0的话,再次进入循环
temp_buff[Fixed_len+0]=strlen(topic)/256; //可变报头第1个字节 :topic长度高字节
temp_buff[Fixed_len+1]=strlen(topic)%256; //可变报头第2个字节 :topic长度低字节
memcpy(&temp_buff[Fixed_len+2],topic,strlen(topic)); //可变报头第3个字节开始 :拷贝topic字符串
memcpy(&temp_buff[Fixed_len+2+strlen(topic)],data,data_len); //有效负荷:拷贝data数据
TxDataBuf_Deal(temp_buff, Fixed_len + Variable_len + Payload_len); //加入发送数据缓冲区
}
/*----------------------------------------------------------*/
/*函数名:处理服务器发来的等级0的推送 */
/*参 数:redata:接收的数据 */
/*返回值:无 */
/*----------------------------------------------------------*/
void MQTT_DealPushdata_Qs0(unsigned char *redata)
{
int re_len; //定义一个变量,存放接收的数据总长度
int pack_num; //定义一个变量,当多个推送一起过来时,保存推送的个数
int temp,temp_len; //定义一个变量,暂存数据
int totle_len; //定义一个变量,存放已经统计的推送的总数据量
int topic_len; //定义一个变量,存放推送中主题的长度
int cmd_len; //定义一个变量,存放推送中包含的命令数据的长度
int cmd_loca; //定义一个变量,存放推送中包含的命令的起始位置
int i; //定义一个变量,用于for循环
int local,multiplier;
unsigned char tempbuff[RBUFF_UNIT]; //临时缓冲区
unsigned char *data; //redata过来的时候,第一个字节是数据总量,data用于指向redata的第2个字节,真正的数据开始的地方
re_len = redata[0]*256+redata[1]; //获取接收的数据总长度
data = &redata[2]; //data指向redata的第2个字节,真正的数据开始的
pack_num = temp_len = totle_len = temp = 0; //各个变量清零
local = 1;
multiplier = 1;
do{
pack_num++; //开始循环统计推送的个数,每次循环推送的个数+1
do{
temp = data[totle_len + local];
temp_len += (temp & 127) * multiplier;
multiplier *= 128;
local++;
}while ((temp & 128) != 0);
totle_len += (temp_len + local); //累计统计的总的推送的数据长度
re_len -= (temp_len + local) ; //接收的数据总长度 减去 本次统计的推送的总长度
local = 1;
multiplier = 1;
temp_len = 0;
}while(re_len!=0); //如果接收的数据总长度等于0了,说明统计完毕了
u1_printf("本次接收了%d个推送数据\r\n",pack_num);//串口输出信息
temp_len = totle_len = 0; //各个变量清零
local = 1;
multiplier = 1;
for(i=0;i<pack_num;i++){ //已经统计到了接收的推送个数,开始for循环,取出每个推送的数据
do{
temp = data[totle_len + local];
temp_len += (temp & 127) * multiplier;
multiplier *= 128;
local++;
}while ((temp & 128) != 0);
topic_len = data[local+totle_len]*256+data[local+1+totle_len] + 2; //计算本次推送数据中主题占用的数据量
cmd_len = temp_len-topic_len; //计算本次推送数据中命令数据占用的数据量
cmd_loca = totle_len + local + topic_len; //计算本次推送数据中命令数据开始的位置
memcpy(tempbuff,&data[cmd_loca],cmd_len); //命令数据拷贝出来
CMDBuf_Deal(tempbuff, cmd_len); //加入命令到缓冲区
totle_len += (temp_len+local); //累计已经统计的推送的数据长度
local = 1;
multiplier = 1;
temp_len = 0;
}
}
/*----------------------------------------------------------*/
/*函数名:处理发送缓冲区 */
/*参 数:data:数据 */
/*参 数:size:数据长度 */
/*返回值:无 */
/*----------------------------------------------------------*/
void TxDataBuf_Deal(unsigned char *data, int size)
{
memcpy(&MQTT_TxDataInPtr[2],data,size); //拷贝数据到发送缓冲区
MQTT_TxDataInPtr[0] = size/256; //记录数据长度
MQTT_TxDataInPtr[1] = size%256; //记录数据长度
MQTT_TxDataInPtr+=TBUFF_UNIT; //指针下移
if(MQTT_TxDataInPtr==MQTT_TxDataEndPtr) //如果指针到缓冲区尾部了
MQTT_TxDataInPtr = MQTT_TxDataBuf[0]; //指针归位到缓冲区开头
}
/*----------------------------------------------------------*/
/*函数名:处理命令缓冲区 */
/*参 数:data:数据 */
/*参 数:size:数据长度 */
/*返回值:无 */
/*----------------------------------------------------------*/
void CMDBuf_Deal(unsigned char *data, int size)
{
memcpy(&MQTT_CMDInPtr[2],data,size); //拷贝数据到命令缓冲区
MQTT_CMDInPtr[0] = size/256; //记录数据长度
MQTT_CMDInPtr[1] = size%256; //记录数据长度
MQTT_CMDInPtr[size+2] = '\0'; //加入字符串结束符
MQTT_CMDInPtr+=CBUFF_UNIT; //指针下移
if(MQTT_CMDInPtr==MQTT_CMDEndPtr) //如果指针到缓冲区尾部了
MQTT_CMDInPtr = MQTT_CMDBuf[0]; //指针归位到缓冲区开头
}
后面是.h文件,在中间的产品ID,密钥和设备名需要在阿里云生成过来
#ifndef __MQTT_H
#define __MQTT_H
#define R_NUM 5 //接收缓冲区个数
#define RBUFF_UNIT 300 //接收缓冲区长度
#define T_NUM 5 //发送缓冲区个数
#define TBUFF_UNIT 300 //发送缓冲区长度
#define C_NUM 8 //命令缓冲区个数
#define CBUFF_UNIT 300 //命令缓冲区长度
#define MQTT_TxData(x) u3_TxData(x) //串口2负责数据发送
#define PRODUCTKEY "a1yT2Yy7KJT" //产品ID
#define PRODUCTKEY_LEN strlen(PRODUCTKEY) //产品ID长度
#define DEVICENAME "Demo1" //设备名
#define DEVICENAME_LEN strlen(DEVICENAME) //设备名长度
#define DEVICESECRE "7mg6SeQ1N2QGeiJ1ggDFbnvODaqgbW06" //设备秘钥
#define DEVICESECRE_LEN strlen(DEVICESECRE) //设备秘钥长度
#define S_TOPIC_NAME "/sys/a1yT2Yy7KJT/D001/thing/service/property/set" //需要订阅的主题
#define P_TOPIC_NAME "/sys/a1yT2Yy7KJT/D001/thing/event/property/post" //需要发布的主题
extern unsigned char MQTT_RxDataBuf[R_NUM][RBUFF_UNIT]; //外部变量声明,数据的接收缓冲区,所有服务器发来的数据,存放在该缓冲区,缓冲区第一个字节存放数据长度
extern unsigned char *MQTT_RxDataInPtr; //外部变量声明,指向缓冲区存放数据的位置
extern unsigned char *MQTT_RxDataOutPtr; //外部变量声明,指向缓冲区读取数据的位置
extern unsigned char *MQTT_RxDataEndPtr; //外部变量声明,指向缓冲区结束的位置
extern unsigned char MQTT_TxDataBuf[T_NUM][TBUFF_UNIT]; //外部变量声明,数据的发送缓冲区,所有发往服务器的数据,存放在该缓冲区,缓冲区第一个字节存放数据长度
extern unsigned char *MQTT_TxDataInPtr; //外部变量声明,指向缓冲区存放数据的位置
extern unsigned char *MQTT_TxDataOutPtr; //外部变量声明,指向缓冲区读取数据的位置
extern unsigned char *MQTT_TxDataEndPtr; //外部变量声明,指向缓冲区结束的位置
extern unsigned char MQTT_CMDBuf[C_NUM][CBUFF_UNIT]; //外部变量声明,命令数据的接收缓冲区
extern unsigned char *MQTT_CMDInPtr; //外部变量声明,指向缓冲区存放数据的位置
extern unsigned char *MQTT_CMDOutPtr; //外部变量声明,指向缓冲区读取数据的位置
extern unsigned char *MQTT_CMDEndPtr; //外部变量声明,指向缓冲区结束的位置
extern char ClientID[128]; //外部变量声明,存放客户端ID的缓冲区
extern int ClientID_len; //外部变量声明,存放客户端ID的长度
extern char Username[128]; //外部变量声明,存放用户名的缓冲区
extern int Username_len; //外部变量声明,存放用户名的长度
extern char Passward[128]; //外部变量声明,存放密码的缓冲区
extern int Passward_len; //外部变量声明,存放密码的长度
extern char ServerIP[128]; //外部变量声明,存放服务器IP或是域名
extern int ServerPort; //外部变量声明,存放服务器的端口号
extern char Ping_flag; //外部变量声明,ping报文状态 0:正常状态,等待计时时间到,发送Ping报文
//外部变量声明,ping报文状态 1:Ping报文已发送,当收到 服务器回复报文的后 将1置为0
extern char Connect_flag; //外部变量声明,同服务器连接状态 0:还没有连接服务器 1:连接上服务器了
extern char ReConnect_flag; //外部变量声明,重连服务器状态 0:连接还存在 1:连接断开,重连
extern char ConnectPack_flag; //外部变量声明,CONNECT报文状态 1:CONNECT报文成功
extern char SubcribePack_flag; //外部变量声明,订阅报文状态 1:订阅报文成功
void MQTT_Buff_Init(void);
void AliIoT_Parameter_Init(void);
void MQTT_ConectPack(void);
void MQTT_Subscribe(char *, int);
void MQTT_PingREQ(void);
void MQTT_PublishQs0(char *, char *, int);
void MQTT_DealPushdata_Qs0(unsigned char *);
void TxDataBuf_Deal(unsigned char *, int);
void CMDBuf_Deal(unsigned char *, int);
#endif