serial.h 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. /*
  2. * Copyright Runtime.io 2018. All rights reserved.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief Utility functions used by the UART and shell mcumgr transports.
  9. *
  10. * Mcumgr packets sent over serial are fragmented into frames of 128 bytes or
  11. * fewer.
  12. *
  13. * The initial frame in a packet has the following format:
  14. * offset 0: 0x06 0x09
  15. * === Begin base64 encoding ===
  16. * offset 2: {16-bit packet-length}
  17. * offset ?: {body}
  18. * offset ?: {crc16} (if final frame)
  19. * === End base64 encoding ===
  20. * offset ?: 0x0a (newline)
  21. *
  22. * All subsequent frames have the following format:
  23. * offset 0: 0x04 0x14
  24. * === Begin base64 encoding ===
  25. * offset 2: {body}
  26. * offset ?: {crc16} (if final frame)
  27. * === End base64 encoding ===
  28. * offset ?: 0x0a (newline)
  29. *
  30. * All integers are represented in big-endian. The packet fields are described
  31. * below:
  32. *
  33. * | Field | Description |
  34. * | -------------- | ------------------------------------------------------- |
  35. * | 0x06 0x09 | Byte pair indicating the start of a packet. |
  36. * | 0x04 0x14 | Byte pair indicating the start of a continuation frame. |
  37. * | Packet length | The combined total length of the *unencoded* body. |
  38. * | Body | The actual SMP data (i.e., 8-byte header and CBOR |
  39. * | | key-value map). |
  40. * | CRC16 | A CRC16 of the *unencoded* body of the entire packet. |
  41. * | | This field is only present in the final frame of a |
  42. * | | packet. |
  43. * | Newline | A 0x0a byte; terminates a frame. |
  44. *
  45. * The packet is fully received when {packet-length} bytes of body has been
  46. * received.
  47. *
  48. * ## CRC details
  49. *
  50. * The CRC16 should be calculated with the following parameters:
  51. *
  52. * | Field | Value |
  53. * | ------------- | ------------- |
  54. * | Polynomial | 0x1021 |
  55. * | Initial Value | 0 |
  56. */
  57. #ifndef ZEPHYR_INCLUDE_MGMT_SERIAL_H_
  58. #define ZEPHYR_INCLUDE_MGMT_SERIAL_H_
  59. #include <zephyr/types.h>
  60. #ifdef __cplusplus
  61. extern "C" {
  62. #endif
  63. #define MCUMGR_SERIAL_HDR_PKT 0x0609
  64. #define MCUMGR_SERIAL_HDR_FRAG 0x0414
  65. #define MCUMGR_SERIAL_MAX_FRAME 128
  66. #define MCUMGR_SERIAL_HDR_PKT_1 (MCUMGR_SERIAL_HDR_PKT >> 8)
  67. #define MCUMGR_SERIAL_HDR_PKT_2 (MCUMGR_SERIAL_HDR_PKT & 0xff)
  68. #define MCUMGR_SERIAL_HDR_FRAG_1 (MCUMGR_SERIAL_HDR_FRAG >> 8)
  69. #define MCUMGR_SERIAL_HDR_FRAG_2 (MCUMGR_SERIAL_HDR_FRAG & 0xff)
  70. /**
  71. * @brief Maintains state for an incoming mcumgr request packet.
  72. */
  73. struct mcumgr_serial_rx_ctxt {
  74. /* Contains the partially- or fully-received mcumgr request. Data
  75. * stored in this buffer has already been base64-decoded.
  76. */
  77. struct net_buf *nb;
  78. /* Length of full packet, as read from header. */
  79. uint16_t pkt_len;
  80. };
  81. /** @typedef mcumgr_serial_tx_cb
  82. * @brief Transmits a chunk of raw response data.
  83. *
  84. * @param data The data to transmit.
  85. * @param len The number of bytes to transmit.
  86. * @param arg An optional argument.
  87. *
  88. * @return 0 on success; negative error code on failure.
  89. */
  90. typedef int (*mcumgr_serial_tx_cb)(const void *data, int len, void *arg);
  91. /**
  92. * @brief Processes an mcumgr request fragment received over a serial
  93. * transport.
  94. *
  95. * Processes an mcumgr request fragment received over a serial transport. If
  96. * the fragment is the end of a valid mcumgr request, this function returns a
  97. * net_buf containing the decoded request. It is the caller's responsibility
  98. * to free the net_buf after it has been processed.
  99. *
  100. * @param rx_ctxt The receive context associated with the serial
  101. * transport being used.
  102. * @param frag The incoming fragment to process.
  103. * @param frag_len The length of the fragment, in bytes.
  104. *
  105. * @return A net_buf containing the decoded request if a
  106. * complete and valid request has been
  107. * received.
  108. * NULL if the packet is incomplete or invalid.
  109. */
  110. struct net_buf *mcumgr_serial_process_frag(
  111. struct mcumgr_serial_rx_ctxt *rx_ctxt,
  112. const uint8_t *frag, int frag_len);
  113. /**
  114. * @brief Encodes and transmits an mcumgr packet over serial.
  115. *
  116. * @param data The mcumgr packet data to send.
  117. * @param len The length of the unencoded mcumgr packet.
  118. * @param cb A callback used to transmit raw bytes.
  119. * @param arg An optional argument to pass to the callback.
  120. *
  121. * @return 0 on success; negative error code on failure.
  122. */
  123. int mcumgr_serial_tx_pkt(const uint8_t *data, int len, mcumgr_serial_tx_cb cb,
  124. void *arg);
  125. #ifdef __cplusplus
  126. }
  127. #endif
  128. #endif