Annotation of libaitmqtt/src/sub.c, revision 1.3.12.3
1.1 misho 1: /*************************************************************************
2: * (C) 2011 AITNET ltd - Sofia/Bulgaria - <misho@aitbg.com>
3: * by Michael Pounov <misho@openbsd-bg.org>
4: *
5: * $Author: misho $
1.3.12.3! misho 6: * $Id: sub.c,v 1.3.12.2 2022/09/15 13:50:14 misho Exp $
1.1 misho 7: *
8: **************************************************************************
9: The ELWIX and AITNET software is distributed under the following
10: terms:
11:
12: All of the documentation and software included in the ELWIX and AITNET
13: Releases is copyrighted by ELWIX - Sofia/Bulgaria <info@elwix.org>
14:
1.3.12.1 misho 15: Copyright 2004 - 2022
1.1 misho 16: by Michael Pounov <misho@elwix.org>. All rights reserved.
17:
18: Redistribution and use in source and binary forms, with or without
19: modification, are permitted provided that the following conditions
20: are met:
21: 1. Redistributions of source code must retain the above copyright
22: notice, this list of conditions and the following disclaimer.
23: 2. Redistributions in binary form must reproduce the above copyright
24: notice, this list of conditions and the following disclaimer in the
25: documentation and/or other materials provided with the distribution.
26: 3. All advertising materials mentioning features or use of this software
27: must display the following acknowledgement:
28: This product includes software developed by Michael Pounov <misho@elwix.org>
29: ELWIX - Embedded LightWeight unIX and its contributors.
30: 4. Neither the name of AITNET nor the names of its contributors
31: may be used to endorse or promote products derived from this software
32: without specific prior written permission.
33:
34: THIS SOFTWARE IS PROVIDED BY AITNET AND CONTRIBUTORS ``AS IS'' AND
35: ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
36: IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
37: ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
38: FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
39: DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
40: OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
41: HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
42: LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
43: OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
44: SUCH DAMAGE.
45: */
46: #include "global.h"
47:
48:
49: /*
50: * mqtt_msgSUBSCRIBE() Create SUBSCRIBE message
51: *
52: * @Topics = MQTT subscription topics
53: * @msgID = MessageID
1.3.12.1 misho 54: * return: NULL error or allocated SUBSCRIBE message
1.1 misho 55: */
1.3.12.1 misho 56: mqtt_msg_t *
1.3.12.3! misho 57: mqtt_msgSUBSCRIBE(mqtt_subscr_t ** __restrict Topics, u_short msgID)
1.1 misho 58: {
1.3.12.1 misho 59: int len, siz;
1.3 misho 60: u_int n, *l;
1.1 misho 61: struct mqtthdr *hdr;
62: mqtthdr_var_t *topic;
63: mqtt_len_t *mid;
64: mqtt_subscr_t *t;
1.2 misho 65: void *data;
1.3.12.1 misho 66: mqtt_msg_t *msg = NULL;
1.1 misho 67:
1.3.12.1 misho 68: if (!Topics || !*Topics)
69: return NULL;
1.3.12.3! misho 70: if (!msgID) {
1.2 misho 71: mqtt_SetErr(EINVAL, "Invalid MessageID parameter must be >0");
1.3.12.1 misho 72: return NULL;
1.1 misho 73: }
74:
1.2 misho 75: /* calculate message size */
76: len = sizeof(mqtt_len_t); /* msgid */
1.3.12.1 misho 77: for (t = *Topics; t && t->sub_topic.msg_base; t++) /* subscribes & qos */
1.3.12.3! misho 78: len += sizeof(mqtt_len_t) + t->sub_topic.msg_len + 1;
1.2 misho 79:
80: /* calculate header size */
81: siz = sizeof(struct mqtthdr); /* mqtt fixed header */
82: n = mqtt_encodeLen(len); /* message size */
83: siz += mqtt_sizeLen(n) - 1; /* length size */
84:
1.3.12.1 misho 85: if (!(msg = mqtt_msgAlloc(siz + len)))
86: return NULL;
1.1 misho 87: else {
1.3.12.1 misho 88: data = msg->msg_base;
1.2 misho 89: hdr = (struct mqtthdr *) data;
1.1 misho 90: }
91:
1.2 misho 92: /* fixed header */
93: hdr->mqtt_msg.type = MQTT_TYPE_SUBSCRIBE;
1.3.12.3! misho 94: hdr->mqtt_msg.qos = MQTT_QOS_ACK;
1.3 misho 95: l = (u_int*) hdr->mqtt_len;
96: *l = n;
1.2 misho 97: data += siz;
98:
1.1 misho 99: /* variable header */
1.2 misho 100: mid = (mqtt_len_t*) data;
1.1 misho 101: mid->val = htons(msgID);
1.2 misho 102: data += sizeof(mqtt_len_t);
1.1 misho 103:
104: /* payload with subscriptions */
1.3.12.1 misho 105: for (t = *Topics; t && t->sub_topic.msg_base; t++) {
1.2 misho 106: topic = (mqtthdr_var_t*) data;
1.1 misho 107: topic->var_sb.val = htons(t->sub_topic.msg_len);
108: memcpy(topic->var_data, t->sub_topic.msg_base, ntohs(topic->var_sb.val));
1.2 misho 109: data += MQTTHDR_VAR_SIZEOF(topic);
1.3.12.3! misho 110: *((char*) data) = t->sub_qos;
! 111: data++;
1.1 misho 112: }
113:
1.3.12.1 misho 114: return msg;
1.1 misho 115: }
116:
117: /*
118: * mqtt_msgSUBACK() Create SUBACK message
119: *
120: * @Topics = MQTT subscription topics
121: * @msgID = MessageID
1.3.12.1 misho 122: * return: NULL error or allocated SUBACK message
1.1 misho 123: */
1.3.12.1 misho 124: mqtt_msg_t *
125: mqtt_msgSUBACK(mqtt_subscr_t ** __restrict Topics, u_short msgID)
1.1 misho 126: {
127: int siz = 0;
128: struct mqtthdr *hdr;
129: mqtt_len_t *v;
130: mqtt_subscr_t *t;
131: u_char *qos;
1.3.12.1 misho 132: mqtt_msg_t *msg = NULL;
1.1 misho 133:
1.3.12.1 misho 134: if (!Topics || !*Topics)
135: return NULL;
1.1 misho 136:
1.3.12.1 misho 137: if (!(msg = mqtt_msgAlloc(MQTTMSG_MAX)))
138: return NULL;
1.1 misho 139: else {
1.3.12.1 misho 140: hdr = (struct mqtthdr *) msg->msg_base;
141: siz = sizeof(struct mqtthdr);
142: v = (mqtt_len_t*) (msg->msg_base + siz);
1.1 misho 143: siz += sizeof(mqtt_len_t);
144: }
145:
146: /* MessageID */
147: v->val = htons(msgID);
148:
149: /* QoS payload from subscriptions */
1.3.12.1 misho 150: for (t = *Topics; t && t->sub_topic.msg_base; t++, siz++) {
151: qos = (msg->msg_base + siz);
1.3.12.2 misho 152: *qos = t->sub_qos;
1.1 misho 153: }
154:
155: /* fixed header */
156: hdr->mqtt_msg.type = MQTT_TYPE_SUBACK;
157: *hdr->mqtt_len = mqtt_encodeLen(siz - sizeof(struct mqtthdr));
158:
1.3.12.1 misho 159: return msg;
1.1 misho 160: }
161:
162: /*
163: * mqtt_msgUNSUBSCRIBE() Create UNSUBSCRIBE message
164: *
165: * @Topics = MQTT subscription topics
166: * @msgID = MessageID
167: * @Dup = Duplicate message
168: * @QOS = QoS
1.3.12.1 misho 169: * return: NULL error or allocated UNSUBSCRIBE message
1.1 misho 170: */
1.3.12.1 misho 171: mqtt_msg_t *
172: mqtt_msgUNSUBSCRIBE(mqtt_subscr_t ** __restrict Topics, u_short msgID,
173: u_char Dup, u_char QOS)
1.1 misho 174: {
1.2 misho 175: int len, siz = 0;
1.3 misho 176: u_int n, *l;
1.1 misho 177: struct mqtthdr *hdr;
178: mqtthdr_var_t *topic;
179: mqtt_len_t *mid;
180: mqtt_subscr_t *t;
1.2 misho 181: void *data;
1.3.12.1 misho 182: mqtt_msg_t *msg = NULL;
1.1 misho 183:
1.3.12.1 misho 184: if (!Topics || !*Topics)
185: return NULL;
1.1 misho 186: if (QOS > MQTT_QOS_EXACTLY) {
1.2 misho 187: mqtt_SetErr(EINVAL, "Invalid QoS parameter");
1.3.12.1 misho 188: return NULL;
1.1 misho 189: }
190: if (!msgID && QOS != MQTT_QOS_ONCE) {
1.2 misho 191: mqtt_SetErr(EINVAL, "Invalid MessageID parameter must be >0");
1.3.12.1 misho 192: return NULL;
1.1 misho 193: }
194:
1.2 misho 195: /* calculate message size */
196: len = sizeof(mqtt_len_t); /* msgid */
1.3.12.1 misho 197: for (t = *Topics; t && t->sub_topic.msg_base; t++) /* subscribes */
1.2 misho 198: len += sizeof(mqtt_len_t) + t->sub_topic.msg_len;
199:
200: /* calculate header size */
201: siz = sizeof(struct mqtthdr); /* mqtt fixed header */
202: n = mqtt_encodeLen(len); /* message size */
203: siz += mqtt_sizeLen(n) - 1; /* length size */
204:
1.3.12.1 misho 205: if (!(msg = mqtt_msgAlloc(siz + len)))
206: return NULL;
1.1 misho 207: else {
1.3.12.1 misho 208: data = msg->msg_base;
1.2 misho 209: hdr = (struct mqtthdr *) data;
1.1 misho 210: }
211:
1.2 misho 212: /* fixed header */
213: hdr->mqtt_msg.type = MQTT_TYPE_UNSUBSCRIBE;
214: hdr->mqtt_msg.qos = QOS;
215: hdr->mqtt_msg.dup = Dup ? 1 : 0;
216: hdr->mqtt_msg.retain = 0;
1.3 misho 217: l = (u_int*) hdr->mqtt_len;
218: *l = n;
1.2 misho 219: data += siz;
220:
1.1 misho 221: /* variable header */
1.3.12.1 misho 222: mid = (mqtt_len_t*) (msg->msg_base + siz);
1.1 misho 223: mid->val = htons(msgID);
1.2 misho 224: data += sizeof(mqtt_len_t);
1.1 misho 225:
226: /* payload with subscriptions */
1.3.12.1 misho 227: for (t = *Topics; t && t->sub_topic.msg_base; t++) {
1.2 misho 228: topic = (mqtthdr_var_t*) data;
1.1 misho 229: topic->var_sb.val = htons(t->sub_topic.msg_len);
230: memcpy(topic->var_data, t->sub_topic.msg_base, ntohs(topic->var_sb.val));
1.2 misho 231: data += MQTTHDR_VAR_SIZEOF(topic);
1.1 misho 232: }
233:
1.3.12.1 misho 234: return msg;
1.1 misho 235: }
236:
237: /*
238: * mqtt_msgUNSUBACK() Create UNSUBACK message
239: *
240: * @msgID = MessageID
1.3.12.1 misho 241: * return: NULL error or allocated UNSUBACK message
1.1 misho 242: */
1.3.12.1 misho 243: mqtt_msg_t *
244: mqtt_msgUNSUBACK(u_short msgID)
1.1 misho 245: {
246: struct mqtthdr *hdr;
247: mqtt_len_t *v;
1.3.12.1 misho 248: mqtt_msg_t *msg = NULL;
1.1 misho 249:
1.3.12.1 misho 250: if (!(msg = mqtt_msgAlloc(sizeof(struct mqtthdr) + sizeof(mqtt_len_t))))
251: return NULL;
1.1 misho 252: else {
1.3.12.1 misho 253: hdr = (struct mqtthdr *) msg->msg_base;
254: v = (mqtt_len_t*) (msg->msg_base + sizeof(struct mqtthdr));
1.1 misho 255: }
256:
257: /* fixed header */
258: hdr->mqtt_msg.type = MQTT_TYPE_UNSUBACK;
259: *hdr->mqtt_len = sizeof(mqtt_len_t);
260:
261: /* MessageID */
262: v->val = htons(msgID);
263:
1.3.12.1 misho 264: return msg;
1.1 misho 265: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>