<p><iframe name="ifd" src="https://mnifdv.cn/resource/cnblogs/ESA2GJK1DH1K_A/" frameborder="0" scrolling="auto" width="100%" height="1500"></iframe></p>
底层包说明
1.MQTT文件夹
mqtt_msg 文件作为最底层的最直接的MQTT协议处理文件.
该文件可以单独使用,该文件是我参考ESP8266的底层移植修改而来.拿去!
mqtt_msg.c
代码语言:javascript复制/**
******************************************************************************
* @author yang feng wu
* @version V1.0.0
* @date 2019/12/15
* @brief
******************************************************************************
******************************************************************************
*/
#define MQTTCLIENT_C_//如果没有定义
#include "mqtt_msg.h"
#include "string.h"
#include "stm32f10x.h"
#define MQTT_MAX_FIXED_HEADER_SIZE 3
uint16_t mqtt_message_id = 0;
enum mqtt_connect_flag
{
MQTT_CONNECT_FLAG_USERNAME = 1 << 7,
MQTT_CONNECT_FLAG_PASSWORD = 1 << 6,
MQTT_CONNECT_FLAG_WILL_RETAIN = 1 << 5,
MQTT_CONNECT_FLAG_WILL = 1 << 2,
MQTT_CONNECT_FLAG_CLEAN_SESSION = 1 << 1
};
//__attribute((__packed__))
struct mqtt_connect_variable_header
{
uint8_t lengthMsb;
uint8_t lengthLsb;
uint8_t magic[4];
uint8_t version;
uint8_t flags;
uint8_t keepaliveMsb;
uint8_t keepaliveLsb;
};
int mqtt_get_type(unsigned char* buffer) { return (buffer[0] & 0xf0) >> 4; }
int mqtt_get_connect_ret_code(unsigned char* buffer) { return (buffer[3]); }
int mqtt_get_qos(unsigned char* buffer) { return (buffer[0] & 0x06) >> 1; }
int append_string(int *length,unsigned char* buffer,int buffer_length,unsigned char* string, int len)
{
if((*length) len 2 > buffer_length)//加上 ClientID 和 记录 ClientID个数(两位) 以后超出了数组
return -1;
buffer[(*length) ] = len >> 8;
buffer[(*length) ] = len & 0xff;
c_memcpy(buffer (*length), string, len);
(*length) = len;
return len 2;
}
uint16_t append_message_id(int *length,unsigned char* buffer,int buffer_length, uint16_t message_id)
{
// If message_id is zero then we should assign one, otherwise
// we'll use the one supplied by the caller
while(message_id == 0)
message_id = mqtt_message_id;
if((*length) 2 > buffer_length)
return 0;
buffer[(*length) ] = message_id >> 8;
buffer[(*length) ] = message_id & 0xff;
return message_id;
}
int fini_message(unsigned char **data_ptr,int length,unsigned char* buffer, int type, int dup, int qos, int retain)
{
int remaining_length = length - MQTT_MAX_FIXED_HEADER_SIZE;
if(remaining_length > 127)
{
buffer[0] = ((type & 0x0f) << 4) | ((dup & 1) << 3) | ((qos & 3) << 1) | (retain & 1);
buffer[1] = 0x80 | (remaining_length % 128);
buffer[2] = remaining_length / 128;
length = remaining_length 3;
*data_ptr = buffer;
}
else
{
buffer[1] = ((type & 0x0f) << 4) | ((dup & 1) << 3) | ((qos & 3) << 1) | (retain & 1);
buffer[2] = remaining_length;
length = remaining_length 2;
*data_ptr = buffer 1;
}
return length;
}
uint16_t mqtt_get_id(unsigned char* buffer, uint16_t length)
{
if(length < 1)
return 0;
switch(mqtt_get_type(buffer))
{
case MQTT_MSG_TYPE_PUBLISH:
{
int i;
int topiclen;
for(i = 1; i < length; i)
{
if((buffer[i] & 0x80) == 0)
{
i;
break;
}
}
if(i 2 >= length)
return 0;
topiclen = buffer[i ] << 8;
topiclen |= buffer[i ];
if(i topiclen >= length)
return 0;
i = topiclen;
if(mqtt_get_qos(buffer) > 0)
{
if(i 2 >= length)
return 0;
//i = 2;
} else {
return 0;
}
return (buffer[i] << 8) | buffer[i 1];
}
case MQTT_MSG_TYPE_PUBACK:
case MQTT_MSG_TYPE_PUBREC:
case MQTT_MSG_TYPE_PUBREL:
case MQTT_MSG_TYPE_PUBCOMP:
case MQTT_MSG_TYPE_SUBACK:
case MQTT_MSG_TYPE_UNSUBACK:
case MQTT_MSG_TYPE_SUBSCRIBE:
{
// This requires the remaining length to be encoded in 1 byte,
// which it should be.
if(length >= 4 && (buffer[1] & 0x80) == 0)
return (buffer[2] << 8) | buffer[3];
else
return 0;
}
default:
return 0;
}
}
/**
* @brief 获取MQTT返回的数据长度(去掉1和2字节后面数据的长度)
* @param buffer MQTT返回的数据首地址
* @param length 返回的数据个数
* @retval 数据长度
* @warning None
* @example
**/
int mqtt_get_total_length(unsigned char* buffer, uint16_t length)
{
int i;
int totlen = 0;
for(i = 1; i < length; i)
{
totlen = (buffer[i] & 0x7f) << (7 * (i - 1));
if((buffer[i] & 0x80) == 0)
{
i;
break;
}
}
totlen = i;
return totlen;
}
/**
* @brief 打包连接MQTT指令
* @param info MQTT信息
* @param data_ptr 打包的数据首地址
* @param buffer 打包进的数组
* @param buffer_length 数组长度
* @retval 数据长度
* @warning None
* @example
**/
int mqtt_msg_connect(mqtt_connect_info_t* info,unsigned char **data_ptr,unsigned char* buffer,int buffer_length)
{
int length;
struct mqtt_connect_variable_header* variable_header;
mqtt_message_id = 0;
length = MQTT_MAX_FIXED_HEADER_SIZE;//头.连接类型1位,数据个数2位(如果大于127就需要两位)
if(length sizeof(*variable_header) > buffer_length)//数组不够存储的
return 0;
variable_header = (void*)(buffer length);//把数组分给这个结构体里面的变量
length = sizeof(*variable_header);//存储完 连接类型,整个数据个数,版本号个数,版本号,等
variable_header->lengthMsb = 0;//版本名称个数高位
variable_header->lengthLsb = 4;//版本名称个数低位
c_memcpy(variable_header->magic, "MQTT", 4);//版本名称MQTT
variable_header->version = 4;//版本号
variable_header->flags = 0;//先清零
variable_header->keepaliveMsb = info->keepalive >> 8;//心跳包时间
variable_header->keepaliveLsb = info->keepalive & 0xff;//心跳包时间
if(info->clean_session)//清除连接信息
variable_header->flags |= MQTT_CONNECT_FLAG_CLEAN_SESSION;
if(info->client_id != NULL && info->client_id[0] != '