dns_resolve.h 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471
  1. /** @file
  2. * @brief DNS resolving library
  3. *
  4. * An API for applications to resolve a DNS name.
  5. */
  6. /*
  7. * Copyright (c) 2017 Intel Corporation
  8. *
  9. * SPDX-License-Identifier: Apache-2.0
  10. */
  11. #ifndef ZEPHYR_INCLUDE_NET_DNS_RESOLVE_H_
  12. #define ZEPHYR_INCLUDE_NET_DNS_RESOLVE_H_
  13. #include <net/net_ip.h>
  14. #include <net/net_context.h>
  15. #ifdef __cplusplus
  16. extern "C" {
  17. #endif
  18. /**
  19. * @brief DNS resolving library
  20. * @defgroup dns_resolve DNS Resolve Library
  21. * @ingroup networking
  22. * @{
  23. */
  24. /**
  25. * DNS query type enum
  26. */
  27. enum dns_query_type {
  28. /** IPv4 query */
  29. DNS_QUERY_TYPE_A = 1,
  30. /** IPv6 query */
  31. DNS_QUERY_TYPE_AAAA = 28
  32. };
  33. /** Max size of the resolved name. */
  34. #ifndef DNS_MAX_NAME_SIZE
  35. #define DNS_MAX_NAME_SIZE 20
  36. #endif
  37. /** @cond INTERNAL_HIDDEN */
  38. /* Make sure that we can compile things even if CONFIG_DNS_RESOLVER
  39. * is not enabled.
  40. */
  41. #if !defined(CONFIG_DNS_RESOLVER_MAX_SERVERS)
  42. #define CONFIG_DNS_RESOLVER_MAX_SERVERS 1
  43. #endif
  44. #if !defined(CONFIG_DNS_NUM_CONCUR_QUERIES)
  45. #define CONFIG_DNS_NUM_CONCUR_QUERIES 1
  46. #endif
  47. /* If mDNS is enabled, then add some extra well known multicast servers to the
  48. * server list.
  49. */
  50. #if defined(CONFIG_MDNS_RESOLVER)
  51. #if defined(CONFIG_NET_IPV6) && defined(CONFIG_NET_IPV4)
  52. #define MDNS_SERVER_COUNT 2
  53. #else
  54. #define MDNS_SERVER_COUNT 1
  55. #endif /* CONFIG_NET_IPV6 && CONFIG_NET_IPV4 */
  56. #else
  57. #define MDNS_SERVER_COUNT 0
  58. #endif /* CONFIG_MDNS_RESOLVER */
  59. /* If LLMNR is enabled, then add some extra well known multicast servers to the
  60. * server list.
  61. */
  62. #if defined(CONFIG_LLMNR_RESOLVER)
  63. #if defined(CONFIG_NET_IPV6) && defined(CONFIG_NET_IPV4)
  64. #define LLMNR_SERVER_COUNT 2
  65. #else
  66. #define LLMNR_SERVER_COUNT 1
  67. #endif /* CONFIG_NET_IPV6 && CONFIG_NET_IPV4 */
  68. #else
  69. #define LLMNR_SERVER_COUNT 0
  70. #endif /* CONFIG_MDNS_RESOLVER */
  71. #define DNS_MAX_MCAST_SERVERS (MDNS_SERVER_COUNT + LLMNR_SERVER_COUNT)
  72. /** @endcond */
  73. /**
  74. * Address info struct is passed to callback that gets all the results.
  75. */
  76. struct dns_addrinfo {
  77. struct sockaddr ai_addr;
  78. socklen_t ai_addrlen;
  79. uint8_t ai_family;
  80. char ai_canonname[DNS_MAX_NAME_SIZE + 1];
  81. };
  82. /**
  83. * Status values for the callback.
  84. */
  85. enum dns_resolve_status {
  86. /** Invalid value for `ai_flags' field */
  87. DNS_EAI_BADFLAGS = -1,
  88. /** NAME or SERVICE is unknown */
  89. DNS_EAI_NONAME = -2,
  90. /** Temporary failure in name resolution */
  91. DNS_EAI_AGAIN = -3,
  92. /** Non-recoverable failure in name res */
  93. DNS_EAI_FAIL = -4,
  94. /** No address associated with NAME */
  95. DNS_EAI_NODATA = -5,
  96. /** `ai_family' not supported */
  97. DNS_EAI_FAMILY = -6,
  98. /** `ai_socktype' not supported */
  99. DNS_EAI_SOCKTYPE = -7,
  100. /** SRV not supported for `ai_socktype' */
  101. DNS_EAI_SERVICE = -8,
  102. /** Address family for NAME not supported */
  103. DNS_EAI_ADDRFAMILY = -9,
  104. /** Memory allocation failure */
  105. DNS_EAI_MEMORY = -10,
  106. /** System error returned in `errno' */
  107. DNS_EAI_SYSTEM = -11,
  108. /** Argument buffer overflow */
  109. DNS_EAI_OVERFLOW = -12,
  110. /** Processing request in progress */
  111. DNS_EAI_INPROGRESS = -100,
  112. /** Request canceled */
  113. DNS_EAI_CANCELED = -101,
  114. /** Request not canceled */
  115. DNS_EAI_NOTCANCELED = -102,
  116. /** All requests done */
  117. DNS_EAI_ALLDONE = -103,
  118. /** IDN encoding failed */
  119. DNS_EAI_IDN_ENCODE = -105,
  120. };
  121. /**
  122. * @typedef dns_resolve_cb_t
  123. * @brief DNS resolve callback
  124. *
  125. * @details The DNS resolve callback is called after a successful
  126. * DNS resolving. The resolver can call this callback multiple times, one
  127. * for each resolved address.
  128. *
  129. * @param status The status of the query:
  130. * DNS_EAI_INPROGRESS returned for each resolved address
  131. * DNS_EAI_ALLDONE mark end of the resolving, info is set to NULL in
  132. * this case
  133. * DNS_EAI_CANCELED if the query was canceled manually or timeout happened
  134. * DNS_EAI_FAIL if the name cannot be resolved by the server
  135. * DNS_EAI_NODATA if there is no such name
  136. * other values means that an error happened.
  137. * @param info Query results are stored here.
  138. * @param user_data The user data given in dns_resolve_name() call.
  139. */
  140. typedef void (*dns_resolve_cb_t)(enum dns_resolve_status status,
  141. struct dns_addrinfo *info,
  142. void *user_data);
  143. enum dns_resolve_context_state {
  144. DNS_RESOLVE_CONTEXT_ACTIVE,
  145. DNS_RESOLVE_CONTEXT_DEACTIVATING,
  146. DNS_RESOLVE_CONTEXT_INACTIVE,
  147. };
  148. /**
  149. * DNS resolve context structure.
  150. */
  151. struct dns_resolve_context {
  152. struct {
  153. /** DNS server information */
  154. struct sockaddr dns_server;
  155. /** Connection to the DNS server */
  156. struct net_context *net_ctx;
  157. /** Is this server mDNS one */
  158. uint8_t is_mdns : 1;
  159. /** Is this server LLMNR one */
  160. uint8_t is_llmnr : 1;
  161. } servers[CONFIG_DNS_RESOLVER_MAX_SERVERS + DNS_MAX_MCAST_SERVERS];
  162. /** Prevent concurrent access */
  163. struct k_mutex lock;
  164. /** This timeout is also used when a buffer is required from the
  165. * buffer pools.
  166. */
  167. k_timeout_t buf_timeout;
  168. /** Result callbacks. We have multiple callbacks here so that it is
  169. * possible to do multiple queries at the same time.
  170. *
  171. * Contents of this structure can be inspected and changed only when
  172. * the lock is held.
  173. */
  174. struct dns_pending_query {
  175. /** Timeout timer */
  176. struct k_work_delayable timer;
  177. /** Back pointer to ctx, needed in timeout handler */
  178. struct dns_resolve_context *ctx;
  179. /** Result callback.
  180. *
  181. * A null value indicates the slot is not in use.
  182. */
  183. dns_resolve_cb_t cb;
  184. /** User data */
  185. void *user_data;
  186. /** TX timeout */
  187. k_timeout_t timeout;
  188. /** String containing the thing to resolve like www.example.com
  189. *
  190. * This is set to a non-null value when the query is started,
  191. * and is not used thereafter.
  192. *
  193. * If the query completed at a point where the work item was
  194. * still pending the pointer is cleared to indicate that the
  195. * query is complete, but release of the query slot will be
  196. * deferred until a request for a slot determines that the
  197. * work item has been released.
  198. */
  199. const char *query;
  200. /** Query type */
  201. enum dns_query_type query_type;
  202. /** DNS id of this query */
  203. uint16_t id;
  204. /** Hash of the DNS name + query type we are querying.
  205. * This hash is calculated so we can match the response that
  206. * we are receiving. This is needed mainly for mDNS which is
  207. * setting the DNS id to 0, which means that the id alone
  208. * cannot be used to find correct pending query.
  209. */
  210. uint16_t query_hash;
  211. } queries[CONFIG_DNS_NUM_CONCUR_QUERIES];
  212. /** Is this context in use */
  213. enum dns_resolve_context_state state;
  214. };
  215. /**
  216. * @brief Init DNS resolving context.
  217. *
  218. * @details This function sets the DNS server address and initializes the
  219. * DNS context that is used by the actual resolver. DNS server addresses
  220. * can be specified either in textual form, or as struct sockaddr (or both).
  221. * Note that the recommended way to resolve DNS names is to use
  222. * the dns_get_addr_info() API. In that case user does not need to
  223. * call dns_resolve_init() as the DNS servers are already setup by the system.
  224. *
  225. * @param ctx DNS context. If the context variable is allocated from
  226. * the stack, then the variable needs to be valid for the whole duration of
  227. * the resolving. Caller does not need to fill the variable beforehand or
  228. * edit the context afterwards.
  229. * @param dns_servers_str DNS server addresses using textual strings. The
  230. * array is NULL terminated. The port number can be given in the string.
  231. * Syntax for the server addresses with or without port numbers:
  232. * IPv4 : 10.0.9.1
  233. * IPv4 + port : 10.0.9.1:5353
  234. * IPv6 : 2001:db8::22:42
  235. * IPv6 + port : [2001:db8::22:42]:5353
  236. * @param dns_servers_sa DNS server addresses as struct sockaddr. The array
  237. * is NULL terminated. Port numbers are optional in struct sockaddr, the
  238. * default will be used if set to 0.
  239. *
  240. * @return 0 if ok, <0 if error.
  241. */
  242. int dns_resolve_init(struct dns_resolve_context *ctx,
  243. const char *dns_servers_str[],
  244. const struct sockaddr *dns_servers_sa[]);
  245. /**
  246. * @brief Close DNS resolving context.
  247. *
  248. * @details This releases DNS resolving context and marks the context unusable.
  249. * Caller must call the dns_resolve_init() again to make context usable.
  250. *
  251. * @param ctx DNS context
  252. *
  253. * @return 0 if ok, <0 if error.
  254. */
  255. int dns_resolve_close(struct dns_resolve_context *ctx);
  256. /**
  257. * @brief Reconfigure DNS resolving context.
  258. *
  259. * @details Reconfigures DNS context with new server list.
  260. *
  261. * @param ctx DNS context
  262. * @param servers_str DNS server addresses using textual strings. The
  263. * array is NULL terminated. The port number can be given in the string.
  264. * Syntax for the server addresses with or without port numbers:
  265. * IPv4 : 10.0.9.1
  266. * IPv4 + port : 10.0.9.1:5353
  267. * IPv6 : 2001:db8::22:42
  268. * IPv6 + port : [2001:db8::22:42]:5353
  269. * @param servers_sa DNS server addresses as struct sockaddr. The array
  270. * is NULL terminated. Port numbers are optional in struct sockaddr, the
  271. * default will be used if set to 0.
  272. *
  273. * @return 0 if ok, <0 if error.
  274. */
  275. int dns_resolve_reconfigure(struct dns_resolve_context *ctx,
  276. const char *servers_str[],
  277. const struct sockaddr *servers_sa[]);
  278. /**
  279. * @brief Cancel a pending DNS query.
  280. *
  281. * @details This releases DNS resources used by a pending query.
  282. *
  283. * @param ctx DNS context
  284. * @param dns_id DNS id of the pending query
  285. *
  286. * @return 0 if ok, <0 if error.
  287. */
  288. int dns_resolve_cancel(struct dns_resolve_context *ctx,
  289. uint16_t dns_id);
  290. /**
  291. * @brief Cancel a pending DNS query using id, name and type.
  292. *
  293. * @details This releases DNS resources used by a pending query.
  294. *
  295. * @param ctx DNS context
  296. * @param dns_id DNS id of the pending query
  297. * @param query_name Name of the resource we are trying to query (hostname)
  298. * @param query_type Type of the query (A or AAAA)
  299. *
  300. * @return 0 if ok, <0 if error.
  301. */
  302. int dns_resolve_cancel_with_name(struct dns_resolve_context *ctx,
  303. uint16_t dns_id,
  304. const char *query_name,
  305. enum dns_query_type query_type);
  306. /**
  307. * @brief Resolve DNS name.
  308. *
  309. * @details This function can be used to resolve e.g., IPv4 or IPv6 address.
  310. * Note that this is asynchronous call, the function will return immediately
  311. * and system will call the callback after resolving has finished or timeout
  312. * has occurred.
  313. * We might send the query to multiple servers (if there are more than one
  314. * server configured), but we only use the result of the first received
  315. * response.
  316. *
  317. * @param ctx DNS context
  318. * @param query What the caller wants to resolve.
  319. * @param type What kind of data the caller wants to get.
  320. * @param dns_id DNS id is returned to the caller. This is needed if one
  321. * wishes to cancel the query. This can be set to NULL if there is no need
  322. * to cancel the query.
  323. * @param cb Callback to call after the resolving has finished or timeout
  324. * has happened.
  325. * @param user_data The user data.
  326. * @param timeout The timeout value for the query. Possible values:
  327. * SYS_FOREVER_MS: the query is tried forever, user needs to cancel it
  328. * manually if it takes too long time to finish
  329. * >0: start the query and let the system timeout it after specified ms
  330. *
  331. * @return 0 if resolving was started ok, < 0 otherwise
  332. */
  333. int dns_resolve_name(struct dns_resolve_context *ctx,
  334. const char *query,
  335. enum dns_query_type type,
  336. uint16_t *dns_id,
  337. dns_resolve_cb_t cb,
  338. void *user_data,
  339. int32_t timeout);
  340. /**
  341. * @brief Get default DNS context.
  342. *
  343. * @details The system level DNS context uses DNS servers that are
  344. * defined in project config file. If no DNS servers are defined by the
  345. * user, then resolving DNS names using default DNS context will do nothing.
  346. * The configuration options are described in subsys/net/lib/dns/Kconfig file.
  347. *
  348. * @return Default DNS context.
  349. */
  350. struct dns_resolve_context *dns_resolve_get_default(void);
  351. /**
  352. * @brief Get IP address info from DNS.
  353. *
  354. * @details This function can be used to resolve e.g., IPv4 or IPv6 address.
  355. * Note that this is asynchronous call, the function will return immediately
  356. * and system will call the callback after resolving has finished or timeout
  357. * has occurred.
  358. * We might send the query to multiple servers (if there are more than one
  359. * server configured), but we only use the result of the first received
  360. * response.
  361. * This variant uses system wide DNS servers.
  362. *
  363. * @param query What the caller wants to resolve.
  364. * @param type What kind of data the caller wants to get.
  365. * @param dns_id DNS id is returned to the caller. This is needed if one
  366. * wishes to cancel the query. This can be set to NULL if there is no need
  367. * to cancel the query.
  368. * @param cb Callback to call after the resolving has finished or timeout
  369. * has happened.
  370. * @param user_data The user data.
  371. * @param timeout The timeout value for the connection. Possible values:
  372. * SYS_FOREVER_MS: the query is tried forever, user needs to cancel it
  373. * manually if it takes too long time to finish
  374. * >0: start the query and let the system timeout it after specified ms
  375. *
  376. * @return 0 if resolving was started ok, < 0 otherwise
  377. */
  378. static inline int dns_get_addr_info(const char *query,
  379. enum dns_query_type type,
  380. uint16_t *dns_id,
  381. dns_resolve_cb_t cb,
  382. void *user_data,
  383. int32_t timeout)
  384. {
  385. return dns_resolve_name(dns_resolve_get_default(),
  386. query,
  387. type,
  388. dns_id,
  389. cb,
  390. user_data,
  391. timeout);
  392. }
  393. /**
  394. * @brief Cancel a pending DNS query.
  395. *
  396. * @details This releases DNS resources used by a pending query.
  397. *
  398. * @param dns_id DNS id of the pending query
  399. *
  400. * @return 0 if ok, <0 if error.
  401. */
  402. static inline int dns_cancel_addr_info(uint16_t dns_id)
  403. {
  404. return dns_resolve_cancel(dns_resolve_get_default(), dns_id);
  405. }
  406. /**
  407. * @}
  408. */
  409. /** @cond INTERNAL_HIDDEN */
  410. /**
  411. * @brief Initialize DNS subsystem.
  412. */
  413. #if defined(CONFIG_DNS_RESOLVER)
  414. void dns_init_resolver(void);
  415. #else
  416. #define dns_init_resolver(...)
  417. #endif /* CONFIG_DNS_RESOLVER */
  418. /** @endcond */
  419. #ifdef __cplusplus
  420. }
  421. #endif
  422. #endif /* ZEPHYR_INCLUDE_NET_DNS_RESOLVE_H_ */