|
version 1.1, 2012/01/26 13:07:33
|
version 1.3, 2012/06/28 11:06:17
|
|
Line 12 terms:
|
Line 12 terms:
|
| All of the documentation and software included in the ELWIX and AITNET |
All of the documentation and software included in the ELWIX and AITNET |
| Releases is copyrighted by ELWIX - Sofia/Bulgaria <info@elwix.org> |
Releases is copyrighted by ELWIX - Sofia/Bulgaria <info@elwix.org> |
| |
|
| Copyright 2004, 2005, 2006, 2007, 2008, 2009, 2010, 2011 | Copyright 2004, 2005, 2006, 2007, 2008, 2009, 2010, 2011, 2012 |
| by Michael Pounov <misho@elwix.org>. All rights reserved. |
by Michael Pounov <misho@elwix.org>. All rights reserved. |
| |
|
| Redistribution and use in source and binary forms, with or without |
Redistribution and use in source and binary forms, with or without |
|
Line 46 SUCH DAMAGE.
|
Line 46 SUCH DAMAGE.
|
| #include "global.h" |
#include "global.h" |
| |
|
| |
|
| /* ------------------------------------------------------------------- */ |
|
| |
|
| /* |
/* |
| * mqtt_msgSUBSCRIBE() Create SUBSCRIBE message |
* mqtt_msgSUBSCRIBE() Create SUBSCRIBE message |
| * |
* |
|
Line 62 int
|
Line 60 int
|
| mqtt_msgSUBSCRIBE(mqtt_msg_t * __restrict buf, mqtt_subscr_t * __restrict Topics, |
mqtt_msgSUBSCRIBE(mqtt_msg_t * __restrict buf, mqtt_subscr_t * __restrict Topics, |
| u_short msgID, u_char Dup, u_char QOS) |
u_short msgID, u_char Dup, u_char QOS) |
| { |
{ |
| int siz = 0; | int len, siz = 0; |
| | u_int n, *l; |
| struct mqtthdr *hdr; |
struct mqtthdr *hdr; |
| mqtthdr_var_t *topic; |
mqtthdr_var_t *topic; |
| mqtt_len_t *mid; |
mqtt_len_t *mid; |
| mqtt_subscr_t *t; |
mqtt_subscr_t *t; |
| u_char *qos; |
u_char *qos; |
| |
void *data; |
| |
|
| if (!buf || !Topics) |
if (!buf || !Topics) |
| return -1; |
return -1; |
| if (QOS > MQTT_QOS_EXACTLY) { |
if (QOS > MQTT_QOS_EXACTLY) { |
| mqtt_SetErr(EINVAL, "Error:: invalid QoS parameter"); | mqtt_SetErr(EINVAL, "Invalid QoS parameter"); |
| return -1; |
return -1; |
| } |
} |
| if (!msgID && QOS != MQTT_QOS_ONCE) { |
if (!msgID && QOS != MQTT_QOS_ONCE) { |
| mqtt_SetErr(EINVAL, "Error:: invalid MessageID parameter must be >0"); | mqtt_SetErr(EINVAL, "Invalid MessageID parameter must be >0"); |
| return -1; |
return -1; |
| } |
} |
| |
|
| if (mqtt_msgRealloc(buf, MQTTMSG_MAX) == -1) | /* calculate message size */ |
| | len = sizeof(mqtt_len_t); /* msgid */ |
| | for (t = Topics; t && t->sub_topic.msg_base; t++) /* subscribes & qos */ |
| | len += sizeof(mqtt_len_t) + t->sub_topic.msg_len + 1; |
| | |
| | /* calculate header size */ |
| | siz = sizeof(struct mqtthdr); /* mqtt fixed header */ |
| | n = mqtt_encodeLen(len); /* message size */ |
| | siz += mqtt_sizeLen(n) - 1; /* length size */ |
| | |
| | if (mqtt_msgRealloc(buf, siz + len) == -1) |
| return -1; |
return -1; |
| else { |
else { |
| hdr = (struct mqtthdr *) (buf->msg_base + siz); | data = buf->msg_base; |
| siz += sizeof(struct mqtthdr); | hdr = (struct mqtthdr *) data; |
| } |
} |
| |
|
| |
/* fixed header */ |
| |
MQTTHDR_MSGINIT(hdr); |
| |
hdr->mqtt_msg.type = MQTT_TYPE_SUBSCRIBE; |
| |
hdr->mqtt_msg.qos = QOS; |
| |
hdr->mqtt_msg.dup = Dup ? 1 : 0; |
| |
hdr->mqtt_msg.retain = 0; |
| |
l = (u_int*) hdr->mqtt_len; |
| |
*l = n; |
| |
data += siz; |
| |
|
| /* variable header */ |
/* variable header */ |
| mid = (mqtt_len_t*) (buf->msg_base + siz); | mid = (mqtt_len_t*) data; |
| mid->val = htons(msgID); |
mid->val = htons(msgID); |
| siz += sizeof(mqtt_len_t); | data += sizeof(mqtt_len_t); |
| |
|
| /* payload with subscriptions */ |
/* payload with subscriptions */ |
| for (t = Topics; t && t->sub_topic.msg_base; t++) { |
for (t = Topics; t && t->sub_topic.msg_base; t++) { |
| topic = (mqtthdr_var_t*) (buf->msg_base + siz); | topic = (mqtthdr_var_t*) data; |
| topic->var_sb.val = htons(t->sub_topic.msg_len); |
topic->var_sb.val = htons(t->sub_topic.msg_len); |
| memcpy(topic->var_data, t->sub_topic.msg_base, ntohs(topic->var_sb.val)); |
memcpy(topic->var_data, t->sub_topic.msg_base, ntohs(topic->var_sb.val)); |
| siz += MQTTHDR_VAR_SIZEOF(topic); | data += MQTTHDR_VAR_SIZEOF(topic); |
| qos = (buf->msg_base + siz); | qos = data++; |
| *qos = t->sub_ret; |
*qos = t->sub_ret; |
| siz++; |
|
| } |
} |
| |
|
| /* fixed header */ | return siz + len; |
| MQTTHDR_MSGINIT(hdr); | |
| hdr->mqtt_msg.type = MQTT_TYPE_SUBSCRIBE; | |
| hdr->mqtt_msg.qos = QOS; | |
| hdr->mqtt_msg.dup = Dup ? 1 : 0; | |
| hdr->mqtt_msg.retain = 0; | |
| *hdr->mqtt_len = mqtt_encodeLen(siz - sizeof(struct mqtthdr)); | |
| |
| mqtt_msgRealloc(buf, siz); | |
| return siz; | |
| } |
} |
| |
|
| /* |
/* |
|
Line 159 mqtt_msgSUBACK(mqtt_msg_t * __restrict buf, mqtt_subsc
|
Line 169 mqtt_msgSUBACK(mqtt_msg_t * __restrict buf, mqtt_subsc
|
| hdr->mqtt_msg.type = MQTT_TYPE_SUBACK; |
hdr->mqtt_msg.type = MQTT_TYPE_SUBACK; |
| *hdr->mqtt_len = mqtt_encodeLen(siz - sizeof(struct mqtthdr)); |
*hdr->mqtt_len = mqtt_encodeLen(siz - sizeof(struct mqtthdr)); |
| |
|
| mqtt_msgRealloc(buf, siz); |
|
| return siz; |
return siz; |
| } |
} |
| |
|
|
Line 177 int
|
Line 186 int
|
| mqtt_msgUNSUBSCRIBE(mqtt_msg_t * __restrict buf, mqtt_subscr_t * __restrict Topics, |
mqtt_msgUNSUBSCRIBE(mqtt_msg_t * __restrict buf, mqtt_subscr_t * __restrict Topics, |
| u_short msgID, u_char Dup, u_char QOS) |
u_short msgID, u_char Dup, u_char QOS) |
| { |
{ |
| int siz = 0; | int len, siz = 0; |
| | u_int n, *l; |
| struct mqtthdr *hdr; |
struct mqtthdr *hdr; |
| mqtthdr_var_t *topic; |
mqtthdr_var_t *topic; |
| mqtt_len_t *mid; |
mqtt_len_t *mid; |
| mqtt_subscr_t *t; |
mqtt_subscr_t *t; |
| |
void *data; |
| |
|
| if (!buf || !Topics) |
if (!buf || !Topics) |
| return -1; |
return -1; |
| if (QOS > MQTT_QOS_EXACTLY) { |
if (QOS > MQTT_QOS_EXACTLY) { |
| mqtt_SetErr(EINVAL, "Error:: invalid QoS parameter"); | mqtt_SetErr(EINVAL, "Invalid QoS parameter"); |
| return -1; |
return -1; |
| } |
} |
| if (!msgID && QOS != MQTT_QOS_ONCE) { |
if (!msgID && QOS != MQTT_QOS_ONCE) { |
| mqtt_SetErr(EINVAL, "Error:: invalid MessageID parameter must be >0"); | mqtt_SetErr(EINVAL, "Invalid MessageID parameter must be >0"); |
| return -1; |
return -1; |
| } |
} |
| |
|
| if (mqtt_msgRealloc(buf, MQTTMSG_MAX) == -1) | /* calculate message size */ |
| | len = sizeof(mqtt_len_t); /* msgid */ |
| | for (t = Topics; t && t->sub_topic.msg_base; t++) /* subscribes */ |
| | len += sizeof(mqtt_len_t) + t->sub_topic.msg_len; |
| | |
| | /* calculate header size */ |
| | siz = sizeof(struct mqtthdr); /* mqtt fixed header */ |
| | n = mqtt_encodeLen(len); /* message size */ |
| | siz += mqtt_sizeLen(n) - 1; /* length size */ |
| | |
| | if (mqtt_msgRealloc(buf, siz + len) == -1) |
| return -1; |
return -1; |
| else { |
else { |
| hdr = (struct mqtthdr *) (buf->msg_base + siz); | data = buf->msg_base; |
| siz += sizeof(struct mqtthdr); | hdr = (struct mqtthdr *) data; |
| } |
} |
| |
|
| |
/* fixed header */ |
| |
MQTTHDR_MSGINIT(hdr); |
| |
hdr->mqtt_msg.type = MQTT_TYPE_UNSUBSCRIBE; |
| |
hdr->mqtt_msg.qos = QOS; |
| |
hdr->mqtt_msg.dup = Dup ? 1 : 0; |
| |
hdr->mqtt_msg.retain = 0; |
| |
l = (u_int*) hdr->mqtt_len; |
| |
*l = n; |
| |
data += siz; |
| |
|
| /* variable header */ |
/* variable header */ |
| mid = (mqtt_len_t*) (buf->msg_base + siz); |
mid = (mqtt_len_t*) (buf->msg_base + siz); |
| mid->val = htons(msgID); |
mid->val = htons(msgID); |
| siz += sizeof(mqtt_len_t); | data += sizeof(mqtt_len_t); |
| |
|
| /* payload with subscriptions */ |
/* payload with subscriptions */ |
| for (t = Topics; t && t->sub_topic.msg_base; t++) { |
for (t = Topics; t && t->sub_topic.msg_base; t++) { |
| topic = (mqtthdr_var_t*) (buf->msg_base + siz); | topic = (mqtthdr_var_t*) data; |
| topic->var_sb.val = htons(t->sub_topic.msg_len); |
topic->var_sb.val = htons(t->sub_topic.msg_len); |
| memcpy(topic->var_data, t->sub_topic.msg_base, ntohs(topic->var_sb.val)); |
memcpy(topic->var_data, t->sub_topic.msg_base, ntohs(topic->var_sb.val)); |
| siz += MQTTHDR_VAR_SIZEOF(topic); | data += MQTTHDR_VAR_SIZEOF(topic); |
| } |
} |
| |
|
| /* fixed header */ | return siz + len; |
| MQTTHDR_MSGINIT(hdr); | |
| hdr->mqtt_msg.type = MQTT_TYPE_UNSUBSCRIBE; | |
| hdr->mqtt_msg.qos = QOS; | |
| hdr->mqtt_msg.dup = Dup ? 1 : 0; | |
| hdr->mqtt_msg.retain = 0; | |
| *hdr->mqtt_len = mqtt_encodeLen(siz - sizeof(struct mqtthdr)); | |
| |
| mqtt_msgRealloc(buf, siz); | |
| return siz; | |
| } |
} |
| |
|
| /* |
/* |
|
Line 272 mqtt_msgUNSUBACK(mqtt_msg_t * __restrict buf, u_short
|
Line 294 mqtt_msgUNSUBACK(mqtt_msg_t * __restrict buf, u_short
|
| * @buf = Message buffer |
* @buf = Message buffer |
| * @msgID = MessageID |
* @msgID = MessageID |
| * @subscr = Subscriptions, must be free after use with mqtt_subFree() |
* @subscr = Subscriptions, must be free after use with mqtt_subFree() |
| * return: NULL error or !=NULL MQTT fixed header | * return: -1 error or >-1 elements into subscr |
| */ |
*/ |
| struct mqtthdr * | int |
| mqtt_readSUBSCRIBE(mqtt_msg_t * __restrict buf, u_short *msgID, mqtt_subscr_t **subscr) |
mqtt_readSUBSCRIBE(mqtt_msg_t * __restrict buf, u_short *msgID, mqtt_subscr_t **subscr) |
| { |
{ |
| register int i; |
register int i; |
|
Line 286 mqtt_readSUBSCRIBE(mqtt_msg_t * __restrict buf, u_shor
|
Line 308 mqtt_readSUBSCRIBE(mqtt_msg_t * __restrict buf, u_shor
|
| caddr_t pos; |
caddr_t pos; |
| |
|
| if (!buf || !msgID || !subscr) |
if (!buf || !msgID || !subscr) |
| return NULL; | return -1; |
| |
|
| hdr = _mqtt_readHEADER(buf, MQTT_TYPE_SUBSCRIBE, &ret, &len); |
hdr = _mqtt_readHEADER(buf, MQTT_TYPE_SUBSCRIBE, &ret, &len); |
| if (!hdr) |
if (!hdr) |
| return NULL; | return -1; |
| pos = buf->msg_base + ret + 1; |
pos = buf->msg_base + ret + 1; |
| v = (mqtt_len_t*) pos; |
v = (mqtt_len_t*) pos; |
| |
|
| /* MessageID */ |
/* MessageID */ |
| len -= sizeof(mqtt_len_t); |
len -= sizeof(mqtt_len_t); |
| if (len < 0) { |
if (len < 0) { |
| mqtt_SetErr(EINVAL, "Error:: short message length %d", len); | mqtt_SetErr(EINVAL, "Short message length %d", len); |
| return NULL; | return -1; |
| } else { |
} else { |
| *msgID = ntohs(v->val); |
*msgID = ntohs(v->val); |
| pos += sizeof(mqtt_len_t); |
pos += sizeof(mqtt_len_t); |
|
Line 306 mqtt_readSUBSCRIBE(mqtt_msg_t * __restrict buf, u_shor
|
Line 328 mqtt_readSUBSCRIBE(mqtt_msg_t * __restrict buf, u_shor
|
| |
|
| subs = mqtt_subAlloc(0); |
subs = mqtt_subAlloc(0); |
| if (!subs) |
if (!subs) |
| return NULL; | return -1; |
| else |
else |
| *subscr = subs; |
*subscr = subs; |
| |
|
|
Line 316 mqtt_readSUBSCRIBE(mqtt_msg_t * __restrict buf, u_shor
|
Line 338 mqtt_readSUBSCRIBE(mqtt_msg_t * __restrict buf, u_shor
|
| len -= MQTTHDR_VAR_SIZEOF(var) + 1; |
len -= MQTTHDR_VAR_SIZEOF(var) + 1; |
| if (len < 0) { |
if (len < 0) { |
| mqtt_subFree(subscr); |
mqtt_subFree(subscr); |
| mqtt_SetErr(EINVAL, "Error:: short message length %d", len); | mqtt_SetErr(EINVAL, "Short message length %d", len); |
| return NULL; | return -1; |
| } |
} |
| subs = mqtt_subRealloc(subs, i + 1); | if (!mqtt_subRealloc(&subs, i + 1)) { |
| if (!subs) { | |
| mqtt_subFree(subscr); |
mqtt_subFree(subscr); |
| return NULL; | return -1; |
| } else |
} else |
| *subscr = subs; |
*subscr = subs; |
| |
|
| memset(&subs[i], 0, sizeof subs[i]); |
memset(&subs[i], 0, sizeof subs[i]); |
| subs[i].sub_topic.msg_len = ntohs(var->var_sb.val); |
subs[i].sub_topic.msg_len = ntohs(var->var_sb.val); |
| subs[i].sub_topic.msg_base = malloc(subs[i].sub_topic.msg_len); | subs[i].sub_topic.msg_base = malloc(subs[i].sub_topic.msg_len + 1); |
| if (!subs[i].sub_topic.msg_base) { |
if (!subs[i].sub_topic.msg_base) { |
| LOGERR; |
LOGERR; |
| mqtt_subFree(subscr); |
mqtt_subFree(subscr); |
| return NULL; | return -1; |
| } else | } else { |
| memcpy(subs[i].sub_topic.msg_base, var->var_data, subs[i].sub_topic.msg_len); |
memcpy(subs[i].sub_topic.msg_base, var->var_data, subs[i].sub_topic.msg_len); |
| |
((char*) subs[i].sub_topic.msg_base)[subs[i].sub_topic.msg_len] = 0; |
| |
} |
| pos += MQTTHDR_VAR_SIZEOF(var); |
pos += MQTTHDR_VAR_SIZEOF(var); |
| |
|
| subs[i].sub_ret = *pos; |
subs[i].sub_ret = *pos; |
| pos++; |
pos++; |
| } |
} |
| |
|
| return hdr; | return i; |
| } |
} |
| |
|
| /* |
/* |
|
Line 372 mqtt_readSUBACK(mqtt_msg_t * __restrict buf, u_short *
|
Line 395 mqtt_readSUBACK(mqtt_msg_t * __restrict buf, u_short *
|
| /* MessageID */ |
/* MessageID */ |
| len -= sizeof(mqtt_len_t); |
len -= sizeof(mqtt_len_t); |
| if (len < 0) { |
if (len < 0) { |
| mqtt_SetErr(EINVAL, "Error:: short message length %d", len); | mqtt_SetErr(EINVAL, "Short message length %d", len); |
| return -1; |
return -1; |
| } else { |
} else { |
| *msgID = ntohs(v->val); |
*msgID = ntohs(v->val); |
|
Line 396 mqtt_readSUBACK(mqtt_msg_t * __restrict buf, u_short *
|
Line 419 mqtt_readSUBACK(mqtt_msg_t * __restrict buf, u_short *
|
| * @buf = Message buffer |
* @buf = Message buffer |
| * @msgID = MessageID |
* @msgID = MessageID |
| * @subscr = Subscriptions, must be free after use with mqtt_subFree() |
* @subscr = Subscriptions, must be free after use with mqtt_subFree() |
| * return: NULL error or !=NULL MQTT fixed header | * return: -1 error or >-1 elements into subscr |
| */ |
*/ |
| struct mqtthdr * | int |
| mqtt_readUNSUBSCRIBE(mqtt_msg_t * __restrict buf, u_short *msgID, mqtt_subscr_t **subscr) |
mqtt_readUNSUBSCRIBE(mqtt_msg_t * __restrict buf, u_short *msgID, mqtt_subscr_t **subscr) |
| { |
{ |
| register int i; |
register int i; |
|
Line 410 mqtt_readUNSUBSCRIBE(mqtt_msg_t * __restrict buf, u_sh
|
Line 433 mqtt_readUNSUBSCRIBE(mqtt_msg_t * __restrict buf, u_sh
|
| caddr_t pos; |
caddr_t pos; |
| |
|
| if (!buf || !msgID || !subscr) |
if (!buf || !msgID || !subscr) |
| return NULL; | return -1; |
| |
|
| hdr = _mqtt_readHEADER(buf, MQTT_TYPE_UNSUBSCRIBE, &ret, &len); |
hdr = _mqtt_readHEADER(buf, MQTT_TYPE_UNSUBSCRIBE, &ret, &len); |
| if (!hdr) |
if (!hdr) |
| return NULL; | return -1; |
| pos = buf->msg_base + ret + 1; |
pos = buf->msg_base + ret + 1; |
| v = (mqtt_len_t*) pos; |
v = (mqtt_len_t*) pos; |
| |
|
| /* MessageID */ |
/* MessageID */ |
| len -= sizeof(mqtt_len_t); |
len -= sizeof(mqtt_len_t); |
| if (len < 0) { |
if (len < 0) { |
| mqtt_SetErr(EINVAL, "Error:: short message length %d", len); | mqtt_SetErr(EINVAL, "Short message length %d", len); |
| return NULL; | return -1; |
| } else { |
} else { |
| *msgID = ntohs(v->val); |
*msgID = ntohs(v->val); |
| pos += sizeof(mqtt_len_t); |
pos += sizeof(mqtt_len_t); |
|
Line 430 mqtt_readUNSUBSCRIBE(mqtt_msg_t * __restrict buf, u_sh
|
Line 453 mqtt_readUNSUBSCRIBE(mqtt_msg_t * __restrict buf, u_sh
|
| |
|
| subs = mqtt_subAlloc(0); |
subs = mqtt_subAlloc(0); |
| if (!subs) |
if (!subs) |
| return NULL; | return -1; |
| else |
else |
| *subscr = subs; |
*subscr = subs; |
| |
|
|
Line 440 mqtt_readUNSUBSCRIBE(mqtt_msg_t * __restrict buf, u_sh
|
Line 463 mqtt_readUNSUBSCRIBE(mqtt_msg_t * __restrict buf, u_sh
|
| len -= MQTTHDR_VAR_SIZEOF(var); |
len -= MQTTHDR_VAR_SIZEOF(var); |
| if (len < 0) { |
if (len < 0) { |
| mqtt_subFree(subscr); |
mqtt_subFree(subscr); |
| mqtt_SetErr(EINVAL, "Error:: short message length %d", len); | mqtt_SetErr(EINVAL, "Short message length %d", len); |
| return NULL; | return -1; |
| } |
} |
| subs = mqtt_subRealloc(subs, i + 1); | if (!mqtt_subRealloc(&subs, i + 1)) { |
| if (!subs) { | |
| mqtt_subFree(subscr); |
mqtt_subFree(subscr); |
| return NULL; | return -1; |
| } else |
} else |
| *subscr = subs; |
*subscr = subs; |
| |
|
| memset(&subs[i], 0, sizeof subs[i]); |
memset(&subs[i], 0, sizeof subs[i]); |
| subs[i].sub_topic.msg_len = ntohs(var->var_sb.val); |
subs[i].sub_topic.msg_len = ntohs(var->var_sb.val); |
| subs[i].sub_topic.msg_base = malloc(subs[i].sub_topic.msg_len); | subs[i].sub_topic.msg_base = malloc(subs[i].sub_topic.msg_len + 1); |
| if (!subs[i].sub_topic.msg_base) { |
if (!subs[i].sub_topic.msg_base) { |
| LOGERR; |
LOGERR; |
| mqtt_subFree(subscr); |
mqtt_subFree(subscr); |
| return NULL; | return -1; |
| } else | } else { |
| memcpy(subs[i].sub_topic.msg_base, var->var_data, subs[i].sub_topic.msg_len); |
memcpy(subs[i].sub_topic.msg_base, var->var_data, subs[i].sub_topic.msg_len); |
| |
((char*) subs[i].sub_topic.msg_base)[subs[i].sub_topic.msg_len] = 0; |
| |
} |
| pos += MQTTHDR_VAR_SIZEOF(var); |
pos += MQTTHDR_VAR_SIZEOF(var); |
| } |
} |
| |
|
| return hdr; | return i; |
| } |
} |
| |
|
| /* |
/* |
|
Line 483 mqtt_readUNSUBACK(mqtt_msg_t * __restrict buf)
|
Line 507 mqtt_readUNSUBACK(mqtt_msg_t * __restrict buf)
|
| if (!hdr) |
if (!hdr) |
| return (u_short) -1; |
return (u_short) -1; |
| if (len < sizeof(mqtt_len_t)) { |
if (len < sizeof(mqtt_len_t)) { |
| mqtt_SetErr(EINVAL, "Error:: short message length %d", len); | mqtt_SetErr(EINVAL, "Short message length %d", len); |
| return (u_short) -1; |
return (u_short) -1; |
| } else { |
} else { |
| pos = buf->msg_base + ret + 1; |
pos = buf->msg_base + ret + 1; |