Contiki-NG
coap.h
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2013, Institute for Pervasive Computing, ETH Zurich
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  * notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  * notice, this list of conditions and the following disclaimer in the
12  * documentation and/or other materials provided with the distribution.
13  * 3. Neither the name of the Institute nor the names of its contributors
14  * may be used to endorse or promote products derived from this software
15  * without specific prior written permission.
16  *
17  * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
18  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20  * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
21  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27  * SUCH DAMAGE.
28  *
29  * This file is part of the Contiki operating system.
30  */
31 
32 /**
33  * \file
34  * An implementation of the Constrained Application Protocol (RFC 7252).
35  * \author
36  * Matthias Kovatsch <kovatsch@inf.ethz.ch>
37  */
38 
39 /**
40  * \addtogroup apps
41  * @{
42  *
43  * \defgroup coap An implementation of CoAP
44  * @{
45  *
46  * This is an implementation of the Constrained Application Protocol
47  */
48 
49 #ifndef COAP_H_
50 #define COAP_H_
51 
52 #include <stddef.h> /* for size_t */
53 #include "coap-constants.h"
54 #include "coap-conf.h"
55 #include "coap-transport.h"
56 
57 /**
58  * \brief The max size of the CoAP messages
59  */
60 #define COAP_MAX_PACKET_SIZE (COAP_MAX_HEADER_SIZE + COAP_MAX_CHUNK_SIZE)
61 
62 /* COAP_MAX_CHUNK_SIZE can be different from 2^x so we need to get next lower 2^x for COAP_MAX_BLOCK_SIZE */
63 #ifndef COAP_MAX_BLOCK_SIZE
64 #define COAP_MAX_BLOCK_SIZE (COAP_MAX_CHUNK_SIZE < 32 ? 16 : \
65  (COAP_MAX_CHUNK_SIZE < 64 ? 32 : \
66  (COAP_MAX_CHUNK_SIZE < 128 ? 64 : \
67  (COAP_MAX_CHUNK_SIZE < 256 ? 128 : \
68  (COAP_MAX_CHUNK_SIZE < 512 ? 256 : \
69  (COAP_MAX_CHUNK_SIZE < 1024 ? 512 : \
70  (COAP_MAX_CHUNK_SIZE < 2048 ? 1024 : 2048)))))))
71 #endif /* COAP_MAX_BLOCK_SIZE */
72 
73 /* bitmap for set options */
74 #define COAP_OPTION_MAP_SIZE (sizeof(uint8_t) * 8)
75 
76 /* parsed message struct */
77 typedef struct {
78  uint8_t *buffer; /* pointer to CoAP header / incoming message buffer / memory to serialize message */
79 
80  uint8_t version;
81  coap_message_type_t type;
82  uint8_t code;
83  uint16_t mid;
84 
85  uint8_t token_len;
86  uint8_t token[COAP_TOKEN_LEN];
87 
88  uint8_t options[COAP_OPTION_SIZE1 / COAP_OPTION_MAP_SIZE + 1]; /* bitmap to check if option is set */
89 
90  uint16_t content_format; /* parse options once and store; allows setting options in random order */
91  uint32_t max_age;
92  uint8_t etag_len;
93  uint8_t etag[COAP_ETAG_LEN];
94  size_t proxy_uri_len;
95  const char *proxy_uri;
96  size_t proxy_scheme_len;
97  const char *proxy_scheme;
98  size_t uri_host_len;
99  const char *uri_host;
100  size_t location_path_len;
101  const char *location_path;
102  uint16_t uri_port;
103  size_t location_query_len;
104  const char *location_query;
105  size_t uri_path_len;
106  const char *uri_path;
107  int32_t observe;
108  uint16_t accept;
109  uint8_t if_match_len;
110  uint8_t if_match[COAP_ETAG_LEN];
111  uint32_t block2_num;
112  uint8_t block2_more;
113  uint16_t block2_size;
114  uint32_t block2_offset;
115  uint32_t block1_num;
116  uint8_t block1_more;
117  uint16_t block1_size;
118  uint32_t block1_offset;
119  uint32_t size2;
120  uint32_t size1;
121  size_t uri_query_len;
122  const char *uri_query;
123  uint8_t if_none_match;
124 
125  const coap_endpoint_t *src_ep;
126 
127  uint16_t payload_len;
128  uint8_t *payload;
129 } coap_message_t;
130 
131 static inline int
132 coap_set_option(coap_message_t *message, unsigned int opt)
133 {
134  if(opt > COAP_OPTION_SIZE1) {
135  return 0;
136  }
137  message->options[opt / COAP_OPTION_MAP_SIZE] |= 1 << (opt % COAP_OPTION_MAP_SIZE);
138  return 1;
139 }
140 
141 static inline int
142 coap_is_option(const coap_message_t *message, unsigned int opt)
143 {
144  return (opt <= COAP_OPTION_SIZE1) &&
145  (message->options[opt / COAP_OPTION_MAP_SIZE] & (1 << (opt % COAP_OPTION_MAP_SIZE))) != 0;
146 }
147 
148 /* option format serialization */
149 #define COAP_SERIALIZE_INT_OPTION(number, field, text) \
150  if(coap_is_option(coap_pkt, number)) { \
151  LOG_DBG(text " [%u]\n", (unsigned int)coap_pkt->field); \
152  option += coap_serialize_int_option(number, current_number, option, coap_pkt->field); \
153  current_number = number; \
154  }
155 #define COAP_SERIALIZE_BYTE_OPTION(number, field, text) \
156  if(coap_is_option(coap_pkt, number)) { \
157  LOG_DBG(text " %u [0x%02X%02X%02X%02X%02X%02X%02X%02X]\n", (unsigned int)coap_pkt->field##_len, \
158  coap_pkt->field[0], \
159  coap_pkt->field[1], \
160  coap_pkt->field[2], \
161  coap_pkt->field[3], \
162  coap_pkt->field[4], \
163  coap_pkt->field[5], \
164  coap_pkt->field[6], \
165  coap_pkt->field[7] \
166  ); /* FIXME always prints 8 bytes */ \
167  option += coap_serialize_array_option(number, current_number, option, coap_pkt->field, coap_pkt->field##_len, '\0'); \
168  current_number = number; \
169  }
170 #define COAP_SERIALIZE_STRING_OPTION(number, field, splitter, text) \
171  if(coap_is_option(coap_pkt, number)) { \
172  LOG_DBG(text " ["); \
173  LOG_DBG_COAP_STRING(coap_pkt->field, coap_pkt->field##_len); \
174  LOG_DBG_("]\n"); \
175  option += coap_serialize_array_option(number, current_number, option, (uint8_t *)coap_pkt->field, coap_pkt->field##_len, splitter); \
176  current_number = number; \
177  }
178 #define COAP_SERIALIZE_BLOCK_OPTION(number, field, text) \
179  if(coap_is_option(coap_pkt, number)) { \
180  LOG_DBG(text " [%lu%s (%u B/blk)]\n", (unsigned long)coap_pkt->field##_num, coap_pkt->field##_more ? "+" : "", coap_pkt->field##_size); \
181  uint32_t block = coap_pkt->field##_num << 4; \
182  if(coap_pkt->field##_more) { block |= 0x8; } \
183  block |= 0xF & coap_log_2(coap_pkt->field##_size / 16); \
184  LOG_DBG(text " encoded: 0x%lX\n", (unsigned long)block); \
185  option += coap_serialize_int_option(number, current_number, option, block); \
186  current_number = number; \
187  }
188 
189 /* to store error code and human-readable payload */
190 extern coap_status_t coap_status_code;
191 extern const char *coap_error_message;
192 
193 void coap_init_connection(void);
194 uint16_t coap_get_mid(void);
195 
196 void coap_init_message(coap_message_t *message, coap_message_type_t type,
197  uint8_t code, uint16_t mid);
198 size_t coap_serialize_message(coap_message_t *message, uint8_t *buffer);
199 coap_status_t coap_parse_message(coap_message_t *request, uint8_t *data,
200  uint16_t data_len);
201 
202 int coap_get_query_variable(coap_message_t *message, const char *name,
203  const char **output);
204 int coap_get_post_variable(coap_message_t *message, const char *name,
205  const char **output);
206 
207 static inline coap_resource_flags_t
208 coap_get_method_type(coap_message_t *message)
209 {
210  return (coap_resource_flags_t)(1 << (message->code - 1));
211 }
212 
213 static inline const coap_endpoint_t *
214 coap_get_src_endpoint(coap_message_t *request)
215 {
216  return request->src_ep;
217 }
218 
219 static inline void
220 coap_set_src_endpoint(coap_message_t *request, const coap_endpoint_t *ep)
221 {
222  request->src_ep = ep;
223 }
224 /*---------------------------------------------------------------------------*/
225 
226 int coap_set_status_code(coap_message_t *message, unsigned int code);
227 
228 int coap_set_token(coap_message_t *message, const uint8_t *token,
229  size_t token_len);
230 
231 int coap_get_header_content_format(coap_message_t *message, unsigned int *format);
232 int coap_set_header_content_format(coap_message_t *message, unsigned int format);
233 
234 int coap_get_header_accept(coap_message_t *message, unsigned int *accept);
235 int coap_set_header_accept(coap_message_t *message, unsigned int accept);
236 
237 int coap_get_header_max_age(coap_message_t *message, uint32_t *age);
238 int coap_set_header_max_age(coap_message_t *message, uint32_t age);
239 
240 int coap_get_header_etag(coap_message_t *message, const uint8_t **etag);
241 int coap_set_header_etag(coap_message_t *message, const uint8_t *etag,
242  size_t etag_len);
243 
244 int coap_get_header_if_match(coap_message_t *message, const uint8_t **etag);
245 int coap_set_header_if_match(coap_message_t *message, const uint8_t *etag,
246  size_t etag_len);
247 
248 int coap_get_header_if_none_match(coap_message_t *message);
249 int coap_set_header_if_none_match(coap_message_t *message);
250 
251 /* in-place string might not be 0-terminated. */
252 int coap_get_header_proxy_uri(coap_message_t *message, const char **uri);
253 int coap_set_header_proxy_uri(coap_message_t *message, const char *uri);
254 
255 /* in-place string might not be 0-terminated. */
256 int coap_get_header_proxy_scheme(coap_message_t *message, const char **scheme);
257 int coap_set_header_proxy_scheme(coap_message_t *message, const char *scheme);
258 
259 /* in-place string might not be 0-terminated. */
260 int coap_get_header_uri_host(coap_message_t *message, const char **host);
261 int coap_set_header_uri_host(coap_message_t *message, const char *host);
262 
263 /* in-place string might not be 0-terminated. */
264 int coap_get_header_uri_path(coap_message_t *message, const char **path);
265 int coap_set_header_uri_path(coap_message_t *message, const char *path);
266 
267 /* in-place string might not be 0-terminated. */
268 int coap_get_header_uri_query(coap_message_t *message, const char **query);
269 int coap_set_header_uri_query(coap_message_t *message, const char *query);
270 
271 /* in-place string might not be 0-terminated. */
272 int coap_get_header_location_path(coap_message_t *message, const char **path);
273 /* also splits optional query into Location-Query option. */
274 int coap_set_header_location_path(coap_message_t *message, const char *path);
275 
276 /* in-place string might not be 0-terminated. */
277 int coap_get_header_location_query(coap_message_t *message, const char **query);
278 int coap_set_header_location_query(coap_message_t *message, const char *query);
279 
280 int coap_get_header_observe(coap_message_t *message, uint32_t *observe);
281 int coap_set_header_observe(coap_message_t *message, uint32_t observe);
282 
283 int coap_get_header_block2(coap_message_t *message, uint32_t *num, uint8_t *more,
284  uint16_t *size, uint32_t *offset);
285 int coap_set_header_block2(coap_message_t *message, uint32_t num, uint8_t more,
286  uint16_t size);
287 
288 int coap_get_header_block1(coap_message_t *message, uint32_t *num, uint8_t *more,
289  uint16_t *size, uint32_t *offset);
290 int coap_set_header_block1(coap_message_t *message, uint32_t num, uint8_t more,
291  uint16_t size);
292 
293 int coap_get_header_size2(coap_message_t *message, uint32_t *size);
294 int coap_set_header_size2(coap_message_t *message, uint32_t size);
295 
296 int coap_get_header_size1(coap_message_t *message, uint32_t *size);
297 int coap_set_header_size1(coap_message_t *message, uint32_t size);
298 
299 int coap_get_payload(coap_message_t *message, const uint8_t **payload);
300 int coap_set_payload(coap_message_t *message, const void *payload, size_t length);
301 
302 #endif /* COAP_H_ */
303 /** @} */
304 /** @} */
static uint8_t accept(uint8_t in)
Definition: mpl.c:1391
static uint8_t output(const linkaddr_t *localdest)
Take an IP packet and format it to be sent on an 802.15.4 network using 6lowpan.
Definition: sicslowpan.c:1565
Collection of default configuration values.
coap_resource_flags_t
Resource flags for allowed methods and special functionalities.
Collection of constants specified in the CoAP standard.
API for CoAP transport