dns_sd.h 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298
  1. /** @file
  2. * @brief DNS Service Discovery
  3. */
  4. /*
  5. * Copyright (c) 2020 Friedt Professional Engineering Services, Inc
  6. *
  7. * SPDX-License-Identifier: Apache-2.0
  8. */
  9. #ifndef ZEPHYR_INCLUDE_NET_DNS_SD_H_
  10. #define ZEPHYR_INCLUDE_NET_DNS_SD_H_
  11. #include <stdint.h>
  12. #include <sys/byteorder.h>
  13. #ifdef __cplusplus
  14. extern "C" {
  15. #endif
  16. /**
  17. * @brief DNS Service Discovery
  18. *
  19. * @details This API enables services to be advertised via DNS. To
  20. * advvertise a service, system or application code should use
  21. * @ref DNS_SD_REGISTER_TCP_SERVICE or
  22. * @ref DNS_SD_REGISTER_UDP_SERVICE.
  23. *
  24. * @see <a href="https://tools.ietf.org/html/rfc6763">RFC 6763</a>
  25. *
  26. * @defgroup dns_sd DNS Service Discovery
  27. * @ingroup networking
  28. * @{
  29. */
  30. /** RFC 1034 Section 3.1 */
  31. #define DNS_SD_INSTANCE_MIN_SIZE 1
  32. /** RFC 1034 Section 3.1, RFC 6763 Section 7.2 */
  33. #define DNS_SD_INSTANCE_MAX_SIZE 63
  34. /** RFC 6763 Section 7.2 - inclusive of underscore */
  35. #define DNS_SD_SERVICE_MIN_SIZE 2
  36. /** RFC 6763 Section 7.2 - inclusive of underscore */
  37. #define DNS_SD_SERVICE_MAX_SIZE 16
  38. /** RFC 6763 Section 4.1.2 */
  39. #define DNS_SD_SERVICE_PREFIX '_'
  40. /** RFC 6763 Section 4.1.2 - either _tcp or _udp (case insensitive) */
  41. #define DNS_SD_PROTO_SIZE 4
  42. /** ICANN Rules for TLD naming */
  43. #define DNS_SD_DOMAIN_MIN_SIZE 2
  44. /** RFC 1034 Section 3.1, RFC 6763 Section 7.2 */
  45. #define DNS_SD_DOMAIN_MAX_SIZE 63
  46. /**
  47. * Minimum number of segments in a fully-qualified name
  48. *
  49. * This reqpresents FQN's of the form below
  50. * ```
  51. * <sn>._tcp.<domain>.
  52. * ```
  53. * Currently sub-types and service domains are unsupported and only the
  54. * "local" domain is supported. Specifically, that excludes the following:
  55. * ```
  56. * <sub>._sub.<sn>._tcp.<servicedomain>.<parentdomain>.
  57. * ```
  58. * @see <a href="https://datatracker.ietf.org/doc/html/rfc6763">RFC 6763</a>, Section 7.2.
  59. */
  60. #define DNS_SD_MIN_LABELS 3
  61. /**
  62. * Maximum number of segments in a fully-qualified name
  63. *
  64. * This reqpresents FQN's of the form below
  65. * ```
  66. * <instance>.<sn>._tcp.<domain>.
  67. * ```
  68. *
  69. * Currently sub-types and service domains are unsupported and only the
  70. * "local" domain is supported. Specifically, that excludes the following:
  71. * ```
  72. * <sub>._sub.<sn>._tcp.<servicedomain>.<parentdomain>.
  73. * ```
  74. * @see <a href="https://datatracker.ietf.org/doc/html/rfc6763">RFC 6763</a>, Section 7.2.
  75. */
  76. #define DNS_SD_MAX_LABELS 4
  77. /**
  78. * @brief Register a service for DNS Service Discovery
  79. *
  80. * This macro should be used for advanced use cases. Two simple use cases are
  81. * when a custom @p domain or a custom (non-standard) @p proto is required.
  82. *
  83. * Another use case is when the port number is not preassigned. That could
  84. * be for a number of reasons, but the most common use case would be for
  85. * ephemeral port usage - i.e. when the service is bound using port number 0.
  86. * In that case, Zephyr (like other OS's) will simply choose an unused port.
  87. * When using ephemeral ports, it can be helpful to assign @p port to the
  88. * @ref sockaddr_in.sin_port field of an IPv4 @ref sockaddr_in, or to the
  89. * @ref sockaddr_in6.sin6_port field of an IPv6 @ref sockaddr_in6.
  90. *
  91. * The service can be referenced using the @p id variable.
  92. *
  93. * @param id variable name for the DNS-SD service record
  94. * @param instance name of the service instance such as "My HTTP Server"
  95. * @param service name of the service, such as "_http"
  96. * @param proto protocol used by the service - either "_tcp" or "_udp"
  97. * @param domain the domain of the service, such as "local"
  98. * @param text information for the DNS TXT record
  99. * @param port a pointer to the port number that this service will use
  100. */
  101. #define DNS_SD_REGISTER_SERVICE(id, instance, service, proto, domain, \
  102. text, port) \
  103. static const STRUCT_SECTION_ITERABLE(dns_sd_rec, id) = { \
  104. instance, \
  105. service, \
  106. proto, \
  107. domain, \
  108. (const char *)text, \
  109. sizeof(text) - 1, \
  110. port \
  111. }
  112. /**
  113. * @brief Register a TCP service for DNS Service Discovery
  114. *
  115. * This macro can be used for service advertisement using DNS-SD.
  116. *
  117. * The service can be referenced using the @p id variable.
  118. *
  119. * Example (with TXT):
  120. * @code{c}
  121. * #include <net/dns_sd.h>
  122. * static const bar_txt[] = {
  123. * "\x06" "path=/"
  124. * "\x0f" "this=is the way"
  125. * "\x0e" "foo or=foo not"
  126. * "\x17" "this=has\0embedded\0nulls"
  127. * "\x04" "true"
  128. * };
  129. * // Possibly use an ephemeral port
  130. * // Possibly only assign bar_port when the service is running
  131. * static uint16_t bar_port;
  132. * DNS_SD_REGISTER_TCP_SERVICE(bar, CONFIG_NET_HOSTNAME,
  133. * "_bar", "local", bar_txt, &bar_port);
  134. * @endcode{c}
  135. *
  136. * TXT records begin with a single length byte (hex-encoded)
  137. * and contain key=value pairs. Thus, the length of the key-value pair
  138. * must not exceed 255 bytes. Care must be taken to ensure that the
  139. * encoded length value is correct.
  140. *
  141. * For additional rules on TXT encoding, see RFC 6763, Section 6.
  142. * @param id variable name for the DNS-SD service record
  143. * @param instance name of the service instance such as "My HTTP Server"
  144. * @param service name of the service, such as "_http"
  145. * @param domain the domain of the service, such as "local"
  146. * @param text information for the DNS TXT record
  147. * @param port the port number that this service will use
  148. *
  149. * @see <a href="https://tools.ietf.org/html/rfc6763">RFC 6763</a>
  150. */
  151. #define DNS_SD_REGISTER_TCP_SERVICE(id, instance, service, domain, text, \
  152. port) \
  153. static const uint16_t id ## _port = sys_cpu_to_be16(port); \
  154. DNS_SD_REGISTER_SERVICE(id, instance, service, "_tcp", domain, \
  155. text, &id ## _port)
  156. /**
  157. * @brief Register a UDP service for DNS Service Discovery
  158. *
  159. * This macro can be used for service advertisement using DNS-SD.
  160. *
  161. * The service can be referenced using the @p id variable.
  162. *
  163. * Example (no TXT):
  164. * @code{c}
  165. * #include <net/dns_sd.h>
  166. * #include <sys/byteorder.h>
  167. * static const foo_port = sys_cpu_to_be16(4242);
  168. * DNS_SD_REGISTER_UDP_SERVICE(foo, CONFIG_NET_HOSTNAME,
  169. * "_foo", DNS_SD_EMPTY_TXT, &foo_port);
  170. * @endcode{c}
  171. *
  172. * @param id variable name for the DNS-SD service record
  173. * @param instance name of the service instance such as "My TFTP Server"
  174. * @param service name of the service, such as "_tftp"
  175. * @param domain the domain of the service, such as "local" or "zephyrproject.org"
  176. * @param text information for the DNS TXT record
  177. * @param port a pointer to the port number that this service will use
  178. *
  179. * @see <a href="https://tools.ietf.org/html/rfc6763">RFC 6763</a>
  180. */
  181. #define DNS_SD_REGISTER_UDP_SERVICE(id, instance, service, domain, text, \
  182. port) \
  183. static const uint16_t id ## _port = sys_cpu_to_be16(port); \
  184. DNS_SD_REGISTER_SERVICE(id, instance, service, "_udp", domain, \
  185. text, &id ## _port)
  186. /** Empty DNS-SD TXT specifier */
  187. #define DNS_SD_EMPTY_TXT dns_sd_empty_txt
  188. /** @cond INTERNAL_HIDDEN */
  189. /**
  190. * @brief DNS Service Discovery record
  191. *
  192. * This structure used in the implementation of RFC 6763 and should not
  193. * need to be accessed directly from application code.
  194. *
  195. * The @a port pointer must be non-NULL. When the value in @a port
  196. * is non-zero, the service is advertized as being on that particular
  197. * port. When the value in @a port is zero, then the service is not
  198. * advertised.
  199. *
  200. * Thus, it is possible for multiple services to advertise on a
  201. * particular port if they hard-code the port.
  202. *
  203. * @internal
  204. *
  205. * @see <a href="https://tools.ietf.org/html/rfc6763">RFC 6763</a>
  206. */
  207. struct dns_sd_rec {
  208. /** <Instance> - e.g. "My HTTP Server" */
  209. const char *instance;
  210. /** Top half of the <Service> such as "_http" */
  211. const char *service;
  212. /** Bottom half of the <Service> "_tcp" or "_udp" */
  213. const char *proto;
  214. /** <Domain> such as "local" or "zephyrproject.org" */
  215. const char *domain;
  216. /** DNS TXT record */
  217. const char *text;
  218. /** Size (in bytes) of the DNS TXT record */
  219. size_t text_size;
  220. /** A pointer to the port number used by the service */
  221. const uint16_t *port;
  222. };
  223. /**
  224. * @brief Empty TXT specifier for DNS-SD
  225. *
  226. * @internal
  227. */
  228. extern const char dns_sd_empty_txt[1];
  229. /**
  230. * @brief Wildcard Port specifier for DNS-SD
  231. *
  232. * @internal
  233. */
  234. extern const uint16_t dns_sd_port_zero;
  235. /** @endcond */
  236. /**
  237. * @brief Obtain the size of DNS-SD TXT data
  238. *
  239. * @param rec the record to in question
  240. * @return the size of the text field
  241. */
  242. static inline size_t dns_sd_txt_size(const struct dns_sd_rec *rec)
  243. {
  244. return rec->text_size;
  245. }
  246. /**
  247. * @brief Check if @a rec is a DNS-SD Service Type Enumeration
  248. *
  249. * DNS-SD Service Type Enumeration is used by network tooling to
  250. * acquire a list of all mDNS-advertised services belonging to a
  251. * particular host on a particular domain.
  252. *
  253. * For example, for the domain '.local', the equivalent query
  254. * would be '_services._dns-sd._udp.local'.
  255. *
  256. * Currently, only the '.local' domain is supported.
  257. *
  258. * @see <a href="https://datatracker.ietf.org/doc/html/rfc6763#section-9">Service Type Enumeration, RFC 6763</a>.
  259. *
  260. * @param rec the record to in question
  261. * @return true if @a rec is a DNS-SD Service Type Enumeration
  262. */
  263. bool dns_sd_is_service_type_enumeration(const struct dns_sd_rec *rec);
  264. /**
  265. * @brief Create a wildcard filter for DNS-SD records
  266. *
  267. * @param filter a pointer to the filter to use
  268. */
  269. void dns_sd_create_wildcard_filter(struct dns_sd_rec *filter);
  270. /**
  271. * @}
  272. */
  273. #ifdef __cplusplus
  274. };
  275. #endif
  276. #endif /* ZEPHYR_INCLUDE_NET_DNS_SD_H_ */