coap.h 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889
  1. /*
  2. * Copyright (c) 2018 Intel Corporation
  3. * Copyright (c) 2021 Nordic Semiconductor
  4. *
  5. * SPDX-License-Identifier: Apache-2.0
  6. */
  7. /**
  8. * @file
  9. *
  10. * @brief CoAP implementation for Zephyr.
  11. */
  12. #ifndef ZEPHYR_INCLUDE_NET_COAP_H_
  13. #define ZEPHYR_INCLUDE_NET_COAP_H_
  14. /**
  15. * @brief COAP library
  16. * @defgroup coap COAP Library
  17. * @ingroup networking
  18. * @{
  19. */
  20. #include <zephyr/types.h>
  21. #include <stddef.h>
  22. #include <stdbool.h>
  23. #include <net/net_ip.h>
  24. #include <sys/slist.h>
  25. #ifdef __cplusplus
  26. extern "C" {
  27. #endif
  28. /**
  29. * @brief Set of CoAP packet options we are aware of.
  30. *
  31. * Users may add options other than these to their packets, provided
  32. * they know how to format them correctly. The only restriction is
  33. * that all options must be added to a packet in numeric order.
  34. *
  35. * Refer to RFC 7252, section 12.2 for more information.
  36. */
  37. enum coap_option_num {
  38. COAP_OPTION_IF_MATCH = 1,
  39. COAP_OPTION_URI_HOST = 3,
  40. COAP_OPTION_ETAG = 4,
  41. COAP_OPTION_IF_NONE_MATCH = 5,
  42. COAP_OPTION_OBSERVE = 6,
  43. COAP_OPTION_URI_PORT = 7,
  44. COAP_OPTION_LOCATION_PATH = 8,
  45. COAP_OPTION_URI_PATH = 11,
  46. COAP_OPTION_CONTENT_FORMAT = 12,
  47. COAP_OPTION_MAX_AGE = 14,
  48. COAP_OPTION_URI_QUERY = 15,
  49. COAP_OPTION_ACCEPT = 17,
  50. COAP_OPTION_LOCATION_QUERY = 20,
  51. COAP_OPTION_BLOCK2 = 23,
  52. COAP_OPTION_BLOCK1 = 27,
  53. COAP_OPTION_SIZE2 = 28,
  54. COAP_OPTION_PROXY_URI = 35,
  55. COAP_OPTION_PROXY_SCHEME = 39,
  56. COAP_OPTION_SIZE1 = 60,
  57. };
  58. /**
  59. * @brief Available request methods.
  60. *
  61. * To be used when creating a request or a response.
  62. */
  63. enum coap_method {
  64. COAP_METHOD_GET = 1,
  65. COAP_METHOD_POST = 2,
  66. COAP_METHOD_PUT = 3,
  67. COAP_METHOD_DELETE = 4,
  68. };
  69. #define COAP_REQUEST_MASK 0x07
  70. #define COAP_VERSION_1 1U
  71. /**
  72. * @brief CoAP packets may be of one of these types.
  73. */
  74. enum coap_msgtype {
  75. /**
  76. * Confirmable message.
  77. *
  78. * The packet is a request or response the destination end-point must
  79. * acknowledge.
  80. */
  81. COAP_TYPE_CON = 0,
  82. /**
  83. * Non-confirmable message.
  84. *
  85. * The packet is a request or response that doesn't
  86. * require acknowledgements.
  87. */
  88. COAP_TYPE_NON_CON = 1,
  89. /**
  90. * Acknowledge.
  91. *
  92. * Response to a confirmable message.
  93. */
  94. COAP_TYPE_ACK = 2,
  95. /**
  96. * Reset.
  97. *
  98. * Rejecting a packet for any reason is done by sending a message
  99. * of this type.
  100. */
  101. COAP_TYPE_RESET = 3
  102. };
  103. #define coap_make_response_code(class, det) ((class << 5) | (det))
  104. /**
  105. * @brief Set of response codes available for a response packet.
  106. *
  107. * To be used when creating a response.
  108. */
  109. enum coap_response_code {
  110. COAP_RESPONSE_CODE_OK = coap_make_response_code(2, 0),
  111. COAP_RESPONSE_CODE_CREATED = coap_make_response_code(2, 1),
  112. COAP_RESPONSE_CODE_DELETED = coap_make_response_code(2, 2),
  113. COAP_RESPONSE_CODE_VALID = coap_make_response_code(2, 3),
  114. COAP_RESPONSE_CODE_CHANGED = coap_make_response_code(2, 4),
  115. COAP_RESPONSE_CODE_CONTENT = coap_make_response_code(2, 5),
  116. COAP_RESPONSE_CODE_CONTINUE = coap_make_response_code(2, 31),
  117. COAP_RESPONSE_CODE_BAD_REQUEST = coap_make_response_code(4, 0),
  118. COAP_RESPONSE_CODE_UNAUTHORIZED = coap_make_response_code(4, 1),
  119. COAP_RESPONSE_CODE_BAD_OPTION = coap_make_response_code(4, 2),
  120. COAP_RESPONSE_CODE_FORBIDDEN = coap_make_response_code(4, 3),
  121. COAP_RESPONSE_CODE_NOT_FOUND = coap_make_response_code(4, 4),
  122. COAP_RESPONSE_CODE_NOT_ALLOWED = coap_make_response_code(4, 5),
  123. COAP_RESPONSE_CODE_NOT_ACCEPTABLE = coap_make_response_code(4, 6),
  124. COAP_RESPONSE_CODE_INCOMPLETE = coap_make_response_code(4, 8),
  125. COAP_RESPONSE_CODE_PRECONDITION_FAILED = coap_make_response_code(4, 12),
  126. COAP_RESPONSE_CODE_REQUEST_TOO_LARGE = coap_make_response_code(4, 13),
  127. COAP_RESPONSE_CODE_UNSUPPORTED_CONTENT_FORMAT =
  128. coap_make_response_code(4, 15),
  129. COAP_RESPONSE_CODE_INTERNAL_ERROR = coap_make_response_code(5, 0),
  130. COAP_RESPONSE_CODE_NOT_IMPLEMENTED = coap_make_response_code(5, 1),
  131. COAP_RESPONSE_CODE_BAD_GATEWAY = coap_make_response_code(5, 2),
  132. COAP_RESPONSE_CODE_SERVICE_UNAVAILABLE = coap_make_response_code(5, 3),
  133. COAP_RESPONSE_CODE_GATEWAY_TIMEOUT = coap_make_response_code(5, 4),
  134. COAP_RESPONSE_CODE_PROXYING_NOT_SUPPORTED =
  135. coap_make_response_code(5, 5)
  136. };
  137. #define COAP_CODE_EMPTY (0)
  138. #define COAP_TOKEN_MAX_LEN 8UL
  139. /**
  140. * @brief Set of Content-Format option values for CoAP.
  141. *
  142. * To be used when encoding or decoding a Content-Format option.
  143. */
  144. enum coap_content_format {
  145. COAP_CONTENT_FORMAT_TEXT_PLAIN = 0, /* charset=urf-8 */
  146. COAP_CONTENT_FORMAT_APP_LINK_FORMAT = 40,
  147. COAP_CONTENT_FORMAT_APP_XML = 41,
  148. COAP_CONTENT_FORMAT_APP_OCTET_STREAM = 42,
  149. COAP_CONTENT_FORMAT_APP_EXI = 47,
  150. COAP_CONTENT_FORMAT_APP_JSON = 50,
  151. COAP_CONTENT_FORMAT_APP_CBOR = 60,
  152. };
  153. /* block option helper */
  154. #define GET_BLOCK_NUM(v) ((v) >> 4)
  155. #define GET_BLOCK_SIZE(v) (((v) & 0x7))
  156. #define GET_MORE(v) (!!((v) & 0x08))
  157. struct coap_observer;
  158. struct coap_packet;
  159. struct coap_pending;
  160. struct coap_reply;
  161. struct coap_resource;
  162. /**
  163. * @typedef coap_method_t
  164. * @brief Type of the callback being called when a resource's method is
  165. * invoked by the remote entity.
  166. */
  167. typedef int (*coap_method_t)(struct coap_resource *resource,
  168. struct coap_packet *request,
  169. struct sockaddr *addr, socklen_t addr_len);
  170. /**
  171. * @typedef coap_notify_t
  172. * @brief Type of the callback being called when a resource's has observers
  173. * to be informed when an update happens.
  174. */
  175. typedef void (*coap_notify_t)(struct coap_resource *resource,
  176. struct coap_observer *observer);
  177. /**
  178. * @brief Description of CoAP resource.
  179. *
  180. * CoAP servers often want to register resources, so that clients can act on
  181. * them, by fetching their state or requesting updates to them.
  182. */
  183. struct coap_resource {
  184. /** Which function to be called for each CoAP method */
  185. coap_method_t get, post, put, del;
  186. coap_notify_t notify;
  187. const char * const *path;
  188. void *user_data;
  189. sys_slist_t observers;
  190. int age;
  191. };
  192. /**
  193. * @brief Represents a remote device that is observing a local resource.
  194. */
  195. struct coap_observer {
  196. sys_snode_t list;
  197. struct sockaddr addr;
  198. uint8_t token[8];
  199. uint8_t tkl;
  200. };
  201. /**
  202. * @brief Representation of a CoAP Packet.
  203. */
  204. struct coap_packet {
  205. uint8_t *data; /* User allocated buffer */
  206. uint16_t offset; /* CoAP lib maintains offset while adding data */
  207. uint16_t max_len; /* Max CoAP packet data length */
  208. uint8_t hdr_len; /* CoAP header length */
  209. uint16_t opt_len; /* Total options length (delta + len + value) */
  210. uint16_t delta; /* Used for delta calculation in CoAP packet */
  211. #if defined(CONFIG_COAP_KEEP_USER_DATA)
  212. void *user_data; /* Application specific user data */
  213. #endif
  214. };
  215. struct coap_option {
  216. uint16_t delta;
  217. #if defined(CONFIG_COAP_EXTENDED_OPTIONS_LEN)
  218. uint16_t len;
  219. uint8_t value[CONFIG_COAP_EXTENDED_OPTIONS_LEN_VALUE];
  220. #else
  221. uint8_t len;
  222. uint8_t value[12];
  223. #endif
  224. };
  225. /**
  226. * @typedef coap_reply_t
  227. * @brief Helper function to be called when a response matches the
  228. * a pending request.
  229. */
  230. typedef int (*coap_reply_t)(const struct coap_packet *response,
  231. struct coap_reply *reply,
  232. const struct sockaddr *from);
  233. #define COAP_DEFAULT_MAX_RETRANSMIT 4
  234. #define COAP_DEFAULT_ACK_RANDOM_FACTOR 1.5
  235. /**
  236. * @brief Represents a request awaiting for an acknowledgment (ACK).
  237. */
  238. struct coap_pending {
  239. struct sockaddr addr;
  240. uint32_t t0;
  241. uint32_t timeout;
  242. uint16_t id;
  243. uint8_t *data;
  244. uint16_t len;
  245. uint8_t retries;
  246. };
  247. /**
  248. * @brief Represents the handler for the reply of a request, it is
  249. * also used when observing resources.
  250. */
  251. struct coap_reply {
  252. coap_reply_t reply;
  253. void *user_data;
  254. int age;
  255. uint16_t id;
  256. uint8_t token[8];
  257. uint8_t tkl;
  258. };
  259. /**
  260. * @brief Returns the version present in a CoAP packet.
  261. *
  262. * @param cpkt CoAP packet representation
  263. *
  264. * @return the CoAP version in packet
  265. */
  266. uint8_t coap_header_get_version(const struct coap_packet *cpkt);
  267. /**
  268. * @brief Returns the type of the CoAP packet.
  269. *
  270. * @param cpkt CoAP packet representation
  271. *
  272. * @return the type of the packet
  273. */
  274. uint8_t coap_header_get_type(const struct coap_packet *cpkt);
  275. /**
  276. * @brief Returns the token (if any) in the CoAP packet.
  277. *
  278. * @param cpkt CoAP packet representation
  279. * @param token Where to store the token, must point to a buffer containing
  280. * at least COAP_TOKEN_MAX_LEN bytes
  281. *
  282. * @return Token length in the CoAP packet (0 - COAP_TOKEN_MAX_LEN).
  283. */
  284. uint8_t coap_header_get_token(const struct coap_packet *cpkt, uint8_t *token);
  285. /**
  286. * @brief Returns the code of the CoAP packet.
  287. *
  288. * @param cpkt CoAP packet representation
  289. *
  290. * @return the code present in the packet
  291. */
  292. uint8_t coap_header_get_code(const struct coap_packet *cpkt);
  293. /**
  294. * @brief Returns the message id associated with the CoAP packet.
  295. *
  296. * @param cpkt CoAP packet representation
  297. *
  298. * @return the message id present in the packet
  299. */
  300. uint16_t coap_header_get_id(const struct coap_packet *cpkt);
  301. /**
  302. * @brief Returns the data pointer and length of the CoAP packet.
  303. *
  304. * @param cpkt CoAP packet representation
  305. * @param len Total length of CoAP payload
  306. *
  307. * @return data pointer and length if payload exists
  308. * NULL pointer and length set to 0 in case there is no payload
  309. */
  310. const uint8_t *coap_packet_get_payload(const struct coap_packet *cpkt,
  311. uint16_t *len);
  312. /**
  313. * @brief Parses the CoAP packet in data, validating it and
  314. * initializing @a cpkt. @a data must remain valid while @a cpkt is used.
  315. *
  316. * @param cpkt Packet to be initialized from received @a data.
  317. * @param data Data containing a CoAP packet, its @a data pointer is
  318. * positioned on the start of the CoAP packet.
  319. * @param len Length of the data
  320. * @param options Parse options and cache its details.
  321. * @param opt_num Number of options
  322. *
  323. * @return 0 in case of success or negative in case of error.
  324. */
  325. int coap_packet_parse(struct coap_packet *cpkt, uint8_t *data, uint16_t len,
  326. struct coap_option *options, uint8_t opt_num);
  327. /**
  328. * @brief Creates a new CoAP Packet from input data.
  329. *
  330. * @param cpkt New packet to be initialized using the storage from @a data.
  331. * @param data Data that will contain a CoAP packet information
  332. * @param max_len Maximum allowable length of data
  333. * @param ver CoAP header version
  334. * @param type CoAP header type
  335. * @param token_len CoAP header token length
  336. * @param token CoAP header token
  337. * @param code CoAP header code
  338. * @param id CoAP header message id
  339. *
  340. * @return 0 in case of success or negative in case of error.
  341. */
  342. int coap_packet_init(struct coap_packet *cpkt, uint8_t *data, uint16_t max_len,
  343. uint8_t ver, uint8_t type, uint8_t token_len,
  344. const uint8_t *token, uint8_t code, uint16_t id);
  345. /**
  346. * @brief Create a new CoAP Acknowledgment message for given request.
  347. *
  348. * This function works like @ref coap_packet_init, filling CoAP header type,
  349. * CoAP header token, and CoAP header message id fields according to
  350. * acknowledgment rules.
  351. *
  352. * @param cpkt New packet to be initialized using the storage from @a data.
  353. * @param req CoAP request packet that is being acknowledged
  354. * @param data Data that will contain a CoAP packet information
  355. * @param max_len Maximum allowable length of data
  356. * @param code CoAP header code
  357. *
  358. * @return 0 in case of success or negative in case of error.
  359. */
  360. int coap_ack_init(struct coap_packet *cpkt, const struct coap_packet *req,
  361. uint8_t *data, uint16_t max_len, uint8_t code);
  362. /**
  363. * @brief Returns a randomly generated array of 8 bytes, that can be
  364. * used as a message's token.
  365. *
  366. * @return a 8-byte pseudo-random token.
  367. */
  368. uint8_t *coap_next_token(void);
  369. /**
  370. * @brief Helper to generate message ids
  371. *
  372. * @return a new message id
  373. */
  374. uint16_t coap_next_id(void);
  375. /**
  376. * @brief Return the values associated with the option of value @a
  377. * code.
  378. *
  379. * @param cpkt CoAP packet representation
  380. * @param code Option number to look for
  381. * @param options Array of #coap_option where to store the value
  382. * of the options found
  383. * @param veclen Number of elements in the options array
  384. *
  385. * @return The number of options found in packet matching code,
  386. * negative on error.
  387. */
  388. int coap_find_options(const struct coap_packet *cpkt, uint16_t code,
  389. struct coap_option *options, uint16_t veclen);
  390. /**
  391. * @brief Appends an option to the packet.
  392. *
  393. * Note: options must be added in numeric order of their codes. Otherwise
  394. * error will be returned.
  395. * TODO: Add support for placing options according to its delta value.
  396. *
  397. * @param cpkt Packet to be updated
  398. * @param code Option code to add to the packet, see #coap_option_num
  399. * @param value Pointer to the value of the option, will be copied to the
  400. * packet
  401. * @param len Size of the data to be added
  402. *
  403. * @return 0 in case of success or negative in case of error.
  404. */
  405. int coap_packet_append_option(struct coap_packet *cpkt, uint16_t code,
  406. const uint8_t *value, uint16_t len);
  407. /**
  408. * @brief Converts an option to its integer representation.
  409. *
  410. * Assumes that the number is encoded in the network byte order in the
  411. * option.
  412. *
  413. * @param option Pointer to the option value, retrieved by
  414. * coap_find_options()
  415. *
  416. * @return The integer representation of the option
  417. */
  418. unsigned int coap_option_value_to_int(const struct coap_option *option);
  419. /**
  420. * @brief Appends an integer value option to the packet.
  421. *
  422. * The option must be added in numeric order of their codes, and the
  423. * least amount of bytes will be used to encode the value.
  424. *
  425. * @param cpkt Packet to be updated
  426. * @param code Option code to add to the packet, see #coap_option_num
  427. * @param val Integer value to be added
  428. *
  429. * @return 0 in case of success or negative in case of error.
  430. */
  431. int coap_append_option_int(struct coap_packet *cpkt, uint16_t code,
  432. unsigned int val);
  433. /**
  434. * @brief Append payload marker to CoAP packet
  435. *
  436. * @param cpkt Packet to append the payload marker (0xFF)
  437. *
  438. * @return 0 in case of success or negative in case of error.
  439. */
  440. int coap_packet_append_payload_marker(struct coap_packet *cpkt);
  441. /**
  442. * @brief Append payload to CoAP packet
  443. *
  444. * @param cpkt Packet to append the payload
  445. * @param payload CoAP packet payload
  446. * @param payload_len CoAP packet payload len
  447. *
  448. * @return 0 in case of success or negative in case of error.
  449. */
  450. int coap_packet_append_payload(struct coap_packet *cpkt, const uint8_t *payload,
  451. uint16_t payload_len);
  452. /**
  453. * @brief When a request is received, call the appropriate methods of
  454. * the matching resources.
  455. *
  456. * @param cpkt Packet received
  457. * @param resources Array of known resources
  458. * @param options Parsed options from coap_packet_parse()
  459. * @param opt_num Number of options
  460. * @param addr Peer address
  461. * @param addr_len Peer address length
  462. *
  463. * @return 0 in case of success or negative in case of error.
  464. */
  465. int coap_handle_request(struct coap_packet *cpkt,
  466. struct coap_resource *resources,
  467. struct coap_option *options,
  468. uint8_t opt_num,
  469. struct sockaddr *addr, socklen_t addr_len);
  470. /**
  471. * Represents the size of each block that will be transferred using
  472. * block-wise transfers [RFC7959]:
  473. *
  474. * Each entry maps directly to the value that is used in the wire.
  475. *
  476. * https://tools.ietf.org/html/rfc7959
  477. */
  478. enum coap_block_size {
  479. COAP_BLOCK_16,
  480. COAP_BLOCK_32,
  481. COAP_BLOCK_64,
  482. COAP_BLOCK_128,
  483. COAP_BLOCK_256,
  484. COAP_BLOCK_512,
  485. COAP_BLOCK_1024,
  486. };
  487. /**
  488. * @brief Helper for converting the enumeration to the size expressed
  489. * in bytes.
  490. *
  491. * @param block_size The block size to be converted
  492. *
  493. * @return The size in bytes that the block_size represents
  494. */
  495. static inline uint16_t coap_block_size_to_bytes(
  496. enum coap_block_size block_size)
  497. {
  498. return (1 << (block_size + 4));
  499. }
  500. /**
  501. * @brief Represents the current state of a block-wise transaction.
  502. */
  503. struct coap_block_context {
  504. size_t total_size;
  505. size_t current;
  506. enum coap_block_size block_size;
  507. };
  508. /**
  509. * @brief Initializes the context of a block-wise transfer.
  510. *
  511. * @param ctx The context to be initialized
  512. * @param block_size The size of the block
  513. * @param total_size The total size of the transfer, if known
  514. *
  515. * @return 0 in case of success or negative in case of error.
  516. */
  517. int coap_block_transfer_init(struct coap_block_context *ctx,
  518. enum coap_block_size block_size,
  519. size_t total_size);
  520. /**
  521. * @brief Append BLOCK1 option to the packet.
  522. *
  523. * @param cpkt Packet to be updated
  524. * @param ctx Block context from which to retrieve the
  525. * information for the Block1 option
  526. *
  527. * @return 0 in case of success or negative in case of error.
  528. */
  529. int coap_append_block1_option(struct coap_packet *cpkt,
  530. struct coap_block_context *ctx);
  531. /**
  532. * @brief Append BLOCK2 option to the packet.
  533. *
  534. * @param cpkt Packet to be updated
  535. * @param ctx Block context from which to retrieve the
  536. * information for the Block2 option
  537. *
  538. * @return 0 in case of success or negative in case of error.
  539. */
  540. int coap_append_block2_option(struct coap_packet *cpkt,
  541. struct coap_block_context *ctx);
  542. /**
  543. * @brief Append SIZE1 option to the packet.
  544. *
  545. * @param cpkt Packet to be updated
  546. * @param ctx Block context from which to retrieve the
  547. * information for the Size1 option
  548. *
  549. * @return 0 in case of success or negative in case of error.
  550. */
  551. int coap_append_size1_option(struct coap_packet *cpkt,
  552. struct coap_block_context *ctx);
  553. /**
  554. * @brief Append SIZE2 option to the packet.
  555. *
  556. * @param cpkt Packet to be updated
  557. * @param ctx Block context from which to retrieve the
  558. * information for the Size2 option
  559. *
  560. * @return 0 in case of success or negative in case of error.
  561. */
  562. int coap_append_size2_option(struct coap_packet *cpkt,
  563. struct coap_block_context *ctx);
  564. /**
  565. * @brief Get the integer representation of a CoAP option.
  566. *
  567. * @param cpkt Packet to be inspected
  568. * @param code CoAP option code
  569. *
  570. * @return Integer value >= 0 in case of success or negative in case
  571. * of error.
  572. */
  573. int coap_get_option_int(const struct coap_packet *cpkt, uint16_t code);
  574. /**
  575. * @brief Retrieves BLOCK{1,2} and SIZE{1,2} from @a cpkt and updates
  576. * @a ctx accordingly.
  577. *
  578. * @param cpkt Packet in which to look for block-wise transfers options
  579. * @param ctx Block context to be updated
  580. *
  581. * @return 0 in case of success or negative in case of error.
  582. */
  583. int coap_update_from_block(const struct coap_packet *cpkt,
  584. struct coap_block_context *ctx);
  585. /**
  586. * @brief Updates @a ctx so after this is called the current entry
  587. * indicates the correct offset in the body of data being
  588. * transferred.
  589. *
  590. * @param cpkt Packet in which to look for block-wise transfers options
  591. * @param ctx Block context to be updated
  592. *
  593. * @return The offset in the block-wise transfer, 0 if the transfer
  594. * has finished.
  595. */
  596. size_t coap_next_block(const struct coap_packet *cpkt,
  597. struct coap_block_context *ctx);
  598. /**
  599. * @brief Indicates that the remote device referenced by @a addr, with
  600. * @a request, wants to observe a resource.
  601. *
  602. * @param observer Observer to be initialized
  603. * @param request Request on which the observer will be based
  604. * @param addr Address of the remote device
  605. */
  606. void coap_observer_init(struct coap_observer *observer,
  607. const struct coap_packet *request,
  608. const struct sockaddr *addr);
  609. /**
  610. * @brief After the observer is initialized, associate the observer
  611. * with an resource.
  612. *
  613. * @param resource Resource to add an observer
  614. * @param observer Observer to be added
  615. *
  616. * @return true if this is the first observer added to this resource.
  617. */
  618. bool coap_register_observer(struct coap_resource *resource,
  619. struct coap_observer *observer);
  620. /**
  621. * @brief Remove this observer from the list of registered observers
  622. * of that resource.
  623. *
  624. * @param resource Resource in which to remove the observer
  625. * @param observer Observer to be removed
  626. */
  627. void coap_remove_observer(struct coap_resource *resource,
  628. struct coap_observer *observer);
  629. /**
  630. * @brief Returns the observer that matches address @a addr.
  631. *
  632. * @param observers Pointer to the array of observers
  633. * @param len Size of the array of observers
  634. * @param addr Address of the endpoint observing a resource
  635. *
  636. * @return A pointer to a observer if a match is found, NULL
  637. * otherwise.
  638. */
  639. struct coap_observer *coap_find_observer_by_addr(
  640. struct coap_observer *observers, size_t len,
  641. const struct sockaddr *addr);
  642. /**
  643. * @brief Returns the next available observer representation.
  644. *
  645. * @param observers Pointer to the array of observers
  646. * @param len Size of the array of observers
  647. *
  648. * @return A pointer to a observer if there's an available observer,
  649. * NULL otherwise.
  650. */
  651. struct coap_observer *coap_observer_next_unused(
  652. struct coap_observer *observers, size_t len);
  653. /**
  654. * @brief Indicates that a reply is expected for @a request.
  655. *
  656. * @param reply Reply structure to be initialized
  657. * @param request Request from which @a reply will be based
  658. */
  659. void coap_reply_init(struct coap_reply *reply,
  660. const struct coap_packet *request);
  661. /**
  662. * @brief Initialize a pending request with a request.
  663. *
  664. * The request's fields are copied into the pending struct, so @a
  665. * request doesn't have to live for as long as the pending struct
  666. * lives, but "data" that needs to live for at least that long.
  667. *
  668. * @param pending Structure representing the waiting for a
  669. * confirmation message, initialized with data from @a request
  670. * @param request Message waiting for confirmation
  671. * @param addr Address to send the retransmission
  672. * @param retries Maximum number of retransmissions of the message.
  673. *
  674. * @return 0 in case of success or negative in case of error.
  675. */
  676. int coap_pending_init(struct coap_pending *pending,
  677. const struct coap_packet *request,
  678. const struct sockaddr *addr,
  679. uint8_t retries);
  680. /**
  681. * @brief Returns the next available pending struct, that can be used
  682. * to track the retransmission status of a request.
  683. *
  684. * @param pendings Pointer to the array of #coap_pending structures
  685. * @param len Size of the array of #coap_pending structures
  686. *
  687. * @return pointer to a free #coap_pending structure, NULL in case
  688. * none could be found.
  689. */
  690. struct coap_pending *coap_pending_next_unused(
  691. struct coap_pending *pendings, size_t len);
  692. /**
  693. * @brief Returns the next available reply struct, so it can be used
  694. * to track replies and notifications received.
  695. *
  696. * @param replies Pointer to the array of #coap_reply structures
  697. * @param len Size of the array of #coap_reply structures
  698. *
  699. * @return pointer to a free #coap_reply structure, NULL in case
  700. * none could be found.
  701. */
  702. struct coap_reply *coap_reply_next_unused(
  703. struct coap_reply *replies, size_t len);
  704. /**
  705. * @brief After a response is received, returns if there is any
  706. * matching pending request exits. User has to clear all pending
  707. * retransmissions related to that response by calling
  708. * coap_pending_clear().
  709. *
  710. * @param response The received response
  711. * @param pendings Pointer to the array of #coap_reply structures
  712. * @param len Size of the array of #coap_reply structures
  713. *
  714. * @return pointer to the associated #coap_pending structure, NULL in
  715. * case none could be found.
  716. */
  717. struct coap_pending *coap_pending_received(
  718. const struct coap_packet *response,
  719. struct coap_pending *pendings, size_t len);
  720. /**
  721. * @brief After a response is received, call coap_reply_t handler
  722. * registered in #coap_reply structure
  723. *
  724. * @param response A response received
  725. * @param from Address from which the response was received
  726. * @param replies Pointer to the array of #coap_reply structures
  727. * @param len Size of the array of #coap_reply structures
  728. *
  729. * @return Pointer to the reply matching the packet received, NULL if
  730. * none could be found.
  731. */
  732. struct coap_reply *coap_response_received(
  733. const struct coap_packet *response,
  734. const struct sockaddr *from,
  735. struct coap_reply *replies, size_t len);
  736. /**
  737. * @brief Returns the next pending about to expire, pending->timeout
  738. * informs how many ms to next expiration.
  739. *
  740. * @param pendings Pointer to the array of #coap_pending structures
  741. * @param len Size of the array of #coap_pending structures
  742. *
  743. * @return The next #coap_pending to expire, NULL if none is about to
  744. * expire.
  745. */
  746. struct coap_pending *coap_pending_next_to_expire(
  747. struct coap_pending *pendings, size_t len);
  748. /**
  749. * @brief After a request is sent, user may want to cycle the pending
  750. * retransmission so the timeout is updated.
  751. *
  752. * @param pending Pending representation to have its timeout updated
  753. *
  754. * @return false if this is the last retransmission.
  755. */
  756. bool coap_pending_cycle(struct coap_pending *pending);
  757. /**
  758. * @brief Cancels the pending retransmission, so it again becomes
  759. * available.
  760. *
  761. * @param pending Pending representation to be canceled
  762. */
  763. void coap_pending_clear(struct coap_pending *pending);
  764. /**
  765. * @brief Cancels all pending retransmissions, so they become
  766. * available again.
  767. *
  768. * @param pendings Pointer to the array of #coap_pending structures
  769. * @param len Size of the array of #coap_pending structures
  770. */
  771. void coap_pendings_clear(struct coap_pending *pendings, size_t len);
  772. /**
  773. * @brief Cancels awaiting for this reply, so it becomes available
  774. * again. User responsibility to free the memory associated with data.
  775. *
  776. * @param reply The reply to be canceled
  777. */
  778. void coap_reply_clear(struct coap_reply *reply);
  779. /**
  780. * @brief Cancels all replies, so they become available again.
  781. *
  782. * @param replies Pointer to the array of #coap_reply structures
  783. * @param len Size of the array of #coap_reply structures
  784. */
  785. void coap_replies_clear(struct coap_reply *replies, size_t len);
  786. /**
  787. * @brief Indicates that this resource was updated and that the @a
  788. * notify callback should be called for every registered observer.
  789. *
  790. * @param resource Resource that was updated
  791. *
  792. * @return 0 in case of success or negative in case of error.
  793. */
  794. int coap_resource_notify(struct coap_resource *resource);
  795. /**
  796. * @brief Returns if this request is enabling observing a resource.
  797. *
  798. * @param request Request to be checked
  799. *
  800. * @return True if the request is enabling observing a resource, False
  801. * otherwise
  802. */
  803. bool coap_request_is_observe(const struct coap_packet *request);
  804. #ifdef __cplusplus
  805. }
  806. #endif
  807. /**
  808. * @}
  809. */
  810. #endif /* ZEPHYR_INCLUDE_NET_COAP_H_ */