mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
773 lines
23 KiB
C
773 lines
23 KiB
C
/*
|
|
* Copyright (C) 2021 Freie Universität Berlin
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
* directory for more details.
|
|
*/
|
|
|
|
/**
|
|
* @{
|
|
*
|
|
* @file
|
|
* @author Martine Lenders <m.lenders@fu-berlin.de>
|
|
*/
|
|
|
|
#include <assert.h>
|
|
#include <errno.h>
|
|
#include <stdbool.h>
|
|
|
|
#include "fmt.h"
|
|
#include "log.h"
|
|
#include "mutex.h"
|
|
#include "net/credman.h"
|
|
#include "net/gcoap.h"
|
|
#include "net/af.h"
|
|
#include "net/dns/cache.h"
|
|
#include "net/ipv4/addr.h"
|
|
#include "net/ipv6/addr.h"
|
|
#include "net/sock/dns.h"
|
|
#include "net/sock/udp.h"
|
|
#include "net/sock/util.h"
|
|
#include "random.h"
|
|
#include "uri_parser.h"
|
|
#include "ut_process.h"
|
|
|
|
#include "net/gcoap/dns.h"
|
|
|
|
#define ENABLE_DEBUG 0
|
|
#include "debug.h"
|
|
|
|
/**
|
|
* @brief Context for a DNS query-response-pair.
|
|
*/
|
|
typedef struct {
|
|
/**
|
|
* @brief Synchronization mutex to wait for response
|
|
*/
|
|
mutex_t resp_wait;
|
|
/**
|
|
* @brief The CoAP request packet
|
|
*
|
|
* Only needs to have coap_pkt_t::payload and coap_pkt_t::payload_len
|
|
* initialized.
|
|
*/
|
|
coap_pkt_t *pkt;
|
|
#if IS_USED(MODULE_DNS_CACHE) || defined(DOXYGEN)
|
|
/**
|
|
* @brief The queried hostname
|
|
*
|
|
* Only required for DNS caching and thus only available with module @ref net_dns_cache
|
|
*/
|
|
const char *domain_name;
|
|
#endif
|
|
void *dns_buf; /**< The buffer for the DNS message exchange */
|
|
void *addr_out; /**< Pointer to the resulting address */
|
|
/**
|
|
* @brief Status for the DNS message exchange
|
|
*
|
|
* - length of _req_ctx_t::addr_out in bytes on success
|
|
* - -EBADMSG, when receiving erroneous response or response containing
|
|
* - -EDESTADDRREQ, if CoAP response was received from an unexpected remote.
|
|
* - -EINVAL, when block-wise transfer can not be completed.
|
|
* - -ENOBUFS, if length of received CoAP body is greater than
|
|
* @ref CONFIG_DNS_MSG_LEN.
|
|
* - -ENOMSG, if CoAP response did not contain a DNS response.
|
|
* - -ETIMEDOUT, if CoAP request timed out.
|
|
*/
|
|
int res;
|
|
uint8_t dns_buf_len; /**< Length of _req_ctx_t::dns_buf */
|
|
int8_t family; /**< Address family to resolve */
|
|
/**
|
|
* @brief The current block number for block-wise transfer
|
|
*
|
|
* Leave unset on function call.
|
|
*/
|
|
uint8_t cur_blk_num;
|
|
#if IS_USED(MODULE_GCOAP_DTLS) || defined(DOXYGEN)
|
|
/**
|
|
* @brief Request tag to rule out potential request reordering attacks
|
|
*/
|
|
uint16_t req_tag;
|
|
#endif
|
|
} _req_ctx_t;
|
|
|
|
typedef struct {
|
|
credman_type_t type; /**< Type of the credential */
|
|
credman_tag_t tag; /**< Tag of the credential */
|
|
} _cred_t;
|
|
|
|
static mutex_t _client_mutex = MUTEX_INIT;
|
|
static char _uri[CONFIG_GCOAP_DNS_SERVER_URI_LEN];
|
|
static char _proxy[CONFIG_GCOAP_DNS_SERVER_URI_LEN];
|
|
static uri_parser_result_t _uri_comp;
|
|
static sock_udp_ep_t _remote;
|
|
#if IS_USED(MODULE_GCOAP_DTLS)
|
|
static _cred_t _creds[CONFIG_GCOAP_DNS_CREDS_MAX] = { 0 };
|
|
#endif
|
|
static uint16_t _req_tag;
|
|
|
|
static inline bool _dns_server_uri_isset(void);
|
|
#if IS_USED(MODULE_GCOAP_DTLS)
|
|
static void _remove_cred(sock_dtls_t *sock, _cred_t *cred);
|
|
#else
|
|
#define _remove_cred(sock, cred) (void)(sock); (void)(cred)
|
|
#endif
|
|
static inline bool _is_proxied(void);
|
|
static int _add_init_block2_opt(coap_pkt_t *pdu);
|
|
static int _add_remaining_options(coap_pkt_t *pdu, const char *proxy_uri, _req_ctx_t *context);
|
|
static int _dns_query(const char *domain_name, _req_ctx_t *req_ctx);
|
|
static ssize_t _send(const void *buf, size_t len, const sock_udp_ep_t *remote,
|
|
bool lock_resp_wait, _req_ctx_t *context, gcoap_socket_type_t tl_type);
|
|
|
|
int gcoap_dns_query(const char *domain_name, void *addr_out, int family)
|
|
{
|
|
int res;
|
|
|
|
if ((res = dns_cache_query(domain_name, addr_out, family)) > 0) {
|
|
return res;
|
|
}
|
|
|
|
static uint8_t coap_buf[CONFIG_GCOAP_DNS_PDU_BUF_SIZE];
|
|
static uint8_t dns_buf[CONFIG_DNS_MSG_LEN];
|
|
coap_pkt_t pdu;
|
|
_req_ctx_t req_ctx = {
|
|
.resp_wait = MUTEX_INIT,
|
|
.pkt = &pdu,
|
|
.dns_buf = dns_buf,
|
|
.addr_out = addr_out,
|
|
.family = family,
|
|
};
|
|
mutex_lock(&_client_mutex);
|
|
pdu.payload = coap_buf;
|
|
pdu.payload_len = sizeof(coap_buf);
|
|
res = _dns_query(domain_name, &req_ctx);
|
|
if (res > 0) {
|
|
/* wait for req_ctx.addr_out to be set */
|
|
mutex_lock(&req_ctx.resp_wait);
|
|
res = req_ctx.res;
|
|
}
|
|
mutex_unlock(&_client_mutex);
|
|
return res;
|
|
}
|
|
|
|
int gcoap_dns_server_uri_set(const char *uri)
|
|
{
|
|
int res;
|
|
|
|
if (!uri) {
|
|
mutex_lock(&_client_mutex);
|
|
_uri[0] = '\0';
|
|
mutex_unlock(&_client_mutex);
|
|
return 0;
|
|
}
|
|
if (IS_USED(MODULE_GCOAP_DTLS)) {
|
|
/* reinitialize request tag */
|
|
_req_tag = (uint16_t)random_uint32();
|
|
}
|
|
if ((strncmp(uri, "coap:", sizeof("coap:") - 1) != 0) &&
|
|
/* if gcoap_dtls is used, URIs not starting with coaps: are also invalid */
|
|
(!IS_USED(MODULE_GCOAP_DTLS) || (strncmp(uri, "coaps:", sizeof("coaps:") - 1) != 0))) {
|
|
return -EINVAL;
|
|
}
|
|
res = strlen(uri);
|
|
if ((((unsigned)res + 1) >= CONFIG_GCOAP_DNS_SERVER_URI_LEN)) {
|
|
return -ENOBUFS;
|
|
}
|
|
mutex_lock(&_client_mutex);
|
|
strcpy(_uri, uri);
|
|
if (uri_parser_process(&_uri_comp, _uri, res) < 0) {
|
|
res = -EINVAL;
|
|
_uri[0] = '\0';
|
|
}
|
|
mutex_unlock(&_client_mutex);
|
|
return res;
|
|
}
|
|
|
|
bool gcoap_dns_server_uri_is_set(void)
|
|
{
|
|
mutex_lock(&_client_mutex);
|
|
bool res = _dns_server_uri_isset();
|
|
mutex_unlock(&_client_mutex);
|
|
return res;
|
|
}
|
|
|
|
ssize_t gcoap_dns_server_uri_get(char *uri, size_t uri_len)
|
|
{
|
|
ssize_t res = 0;
|
|
mutex_lock(&_client_mutex);
|
|
if (_dns_server_uri_isset()) {
|
|
res = strlen(_uri);
|
|
if ((size_t)(res + 1) > uri_len) {
|
|
/* account for trailing \0 */
|
|
res = -ENOBUFS;
|
|
}
|
|
else {
|
|
strcpy(uri, _uri);
|
|
}
|
|
}
|
|
mutex_unlock(&_client_mutex);
|
|
return res;
|
|
}
|
|
|
|
void gcoap_dns_cred_reset(void)
|
|
{
|
|
#if IS_USED(MODULE_GCOAP_DTLS)
|
|
sock_dtls_t *sock = gcoap_get_sock_dtls();
|
|
for (unsigned i = 0; i < ARRAY_SIZE(_creds); i++) {
|
|
if (_creds[i].type != CREDMAN_TYPE_EMPTY) {
|
|
_remove_cred(sock, &_creds[i]);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
int gcoap_dns_cred_add(credman_credential_t *creds)
|
|
{
|
|
_cred_t *c = NULL;
|
|
|
|
if (!IS_USED(MODULE_GCOAP_DTLS)) {
|
|
return -ENOTSUP;
|
|
}
|
|
assert(creds != NULL);
|
|
#if IS_USED(MODULE_GCOAP_DTLS)
|
|
for (unsigned i = 0; i < ARRAY_SIZE(_creds); i++) {
|
|
if (!c &&
|
|
(((_creds[i].type == creds->type) && (_creds[i].tag == creds->tag)) ||
|
|
(_creds[i].type == CREDMAN_TYPE_EMPTY))) {
|
|
c = &_creds[i];
|
|
break;
|
|
}
|
|
}
|
|
#endif
|
|
if (c == NULL) {
|
|
DEBUG("gcoap_dns: no space to store credential tag\n");
|
|
return -ENOMEM;
|
|
}
|
|
int res = credman_add(creds);
|
|
if ((res < 0) && (res != CREDMAN_EXIST)) {
|
|
/* ignore duplicate credentials */
|
|
DEBUG("gcoap_dns: cannot add credential to system: %d\n", res);
|
|
return -EBADF;
|
|
}
|
|
if (res == CREDMAN_OK) {
|
|
#if IS_USED(MODULE_GCOAP_DTLS)
|
|
/* functions used in here are only available with module gcoap_dtls, so guard this
|
|
* section */
|
|
sock_dtls_t *gcoap_sock_dtls = gcoap_get_sock_dtls();
|
|
|
|
res = sock_dtls_add_credential(gcoap_sock_dtls, creds->tag);
|
|
if (res < 0) {
|
|
DEBUG("gcoap_dns: cannot add credential to GCoAPs DTLS sock\n");
|
|
return -EBADF;
|
|
}
|
|
#endif
|
|
c->tag = creds->tag;
|
|
c->type = creds->type;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void gcoap_dns_cred_remove(credman_tag_t tag, credman_type_t type)
|
|
{
|
|
#if IS_USED(MODULE_GCOAP_DTLS)
|
|
sock_dtls_t *sock = gcoap_get_sock_dtls();
|
|
for (unsigned i = 0; i < ARRAY_SIZE(_creds); i++) {
|
|
if ((_creds[i].tag == tag) && (_creds[i].type == type)) {
|
|
_remove_cred(sock, &_creds[i]);
|
|
break;
|
|
}
|
|
}
|
|
#else
|
|
(void)tag;
|
|
(void)type;
|
|
#endif
|
|
}
|
|
|
|
void gcoap_dns_server_proxy_reset(void)
|
|
{
|
|
if (IS_USED(MODULE_GCOAP_DNS_PROXIED)) {
|
|
mutex_lock(&_client_mutex);
|
|
_proxy[0] = '\0';
|
|
mutex_unlock(&_client_mutex);
|
|
}
|
|
}
|
|
|
|
int gcoap_dns_server_proxy_set(const char *proxy)
|
|
{
|
|
int res;
|
|
|
|
if (!IS_USED(MODULE_GCOAP_DNS_PROXIED)) {
|
|
return -ENOTSUP;
|
|
}
|
|
res = strlen(proxy);
|
|
|
|
if ((unsigned)(res + 1) >= CONFIG_GCOAP_DNS_SERVER_URI_LEN) {
|
|
return -EINVAL;
|
|
}
|
|
mutex_lock(&_client_mutex);
|
|
strcpy(_proxy, proxy);
|
|
mutex_unlock(&_client_mutex);
|
|
return res;
|
|
}
|
|
|
|
ssize_t gcoap_dns_server_proxy_get(char *proxy, size_t proxy_len)
|
|
{
|
|
ssize_t res = 0;
|
|
mutex_lock(&_client_mutex);
|
|
if (_dns_server_uri_isset()) {
|
|
res = strlen(_uri);
|
|
if (((size_t)res + 1) > proxy_len) {
|
|
/* account for trailing \0 */
|
|
res = -ENOBUFS;
|
|
}
|
|
else {
|
|
strcpy(proxy, _proxy);
|
|
}
|
|
}
|
|
mutex_unlock(&_client_mutex);
|
|
return res;
|
|
}
|
|
|
|
static inline bool _dns_server_uri_isset(void)
|
|
{
|
|
return _uri[0] != '\0';
|
|
}
|
|
|
|
#if IS_USED(MODULE_GCOAP_DTLS)
|
|
static void _remove_cred(sock_dtls_t *sock, _cred_t *cred)
|
|
{
|
|
sock_dtls_remove_credential(sock, cred->tag);
|
|
credman_delete(cred->tag, cred->type);
|
|
cred->type = CREDMAN_TYPE_EMPTY;
|
|
cred->tag = CREDMAN_TAG_EMPTY;
|
|
}
|
|
#endif
|
|
|
|
static inline bool _is_proxied(void)
|
|
{
|
|
return IS_USED(MODULE_GCOAP_DNS_PROXIED) && _proxy[0] != '\0';
|
|
}
|
|
|
|
static int _add_init_block2_opt(coap_pkt_t *pdu)
|
|
{
|
|
if (CONFIG_GCOAP_DNS_PDU_BUF_SIZE < CONFIG_DNS_MSG_LEN) {
|
|
/* If our largest DNS message fits in the DNS PDU BUF, there is no point in sending Block2
|
|
* in the initial message---a huge response might still overwhelm our PDU buffer, but if
|
|
* that happens we could not have processed it as a DNS message if it came in in fragments
|
|
* either. */
|
|
coap_block1_t block;
|
|
|
|
coap_block_object_init(&block, 0, CONFIG_GCOAP_DNS_BLOCK_SIZE, 0);
|
|
return coap_opt_add_block2_control(pdu, &block);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int _add_proxy_uri_opt(coap_pkt_t *pdu, const char *proxy_uri)
|
|
{
|
|
if (_is_proxied()) {
|
|
return coap_opt_add_proxy_uri(pdu, proxy_uri);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int _add_req_tag_opt(coap_pkt_t *pdu, _req_ctx_t *context)
|
|
{
|
|
#if IS_USED(MODULE_GCOAP_DTLS)
|
|
if (CONFIG_GCOAP_DNS_PDU_BUF_SIZE < CONFIG_DNS_MSG_LEN) {
|
|
return coap_opt_add_opaque(pdu, 292, (uint8_t *)&context->req_tag,
|
|
sizeof(context->req_tag));
|
|
}
|
|
#else
|
|
(void)pdu;
|
|
(void)context;
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
static int _add_remaining_options(coap_pkt_t *pdu, const char *proxy_uri, _req_ctx_t *context)
|
|
{
|
|
if (_add_proxy_uri_opt(pdu, proxy_uri) < 0) {
|
|
DEBUG("gcoap_dns: unable to add Proxy-URI option to request\n");
|
|
return -ENOBUFS;
|
|
}
|
|
/* add request tag to distinguish multiple blockwise requests on-the-air */
|
|
if (_add_req_tag_opt(pdu, context) < 0) {
|
|
DEBUG("gcoap_dns: unable to add Request-Tag option to request");
|
|
return -ENOBUFS;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static size_t _dns_msg_compose(void *dns_buf, const char *domain_name,
|
|
int family)
|
|
{
|
|
return dns_msg_compose_query(dns_buf, domain_name, 0, family);
|
|
}
|
|
|
|
static int _set_remote(const uri_parser_result_t *uri_comp,
|
|
sock_udp_ep_t *remote) {
|
|
assert(uri_comp->host != NULL);
|
|
memset(remote, 0, sizeof(*remote));
|
|
|
|
if (uri_comp->ipv6addr != NULL) {
|
|
if (ipv6_addr_from_buf((ipv6_addr_t *)remote->addr.ipv6,
|
|
uri_comp->ipv6addr,
|
|
uri_comp->ipv6addr_len) == NULL) {
|
|
DEBUG("gcoap_dns: unable to parse IPv6 address %*.s\n",
|
|
uri_comp->ipv6addr_len, uri_comp->ipv6addr);
|
|
return -EINVAL;
|
|
}
|
|
remote->family = AF_INET6;
|
|
if (uri_comp->zoneid != NULL) {
|
|
netif_t *netif = netif_get_by_name_buffer(uri_comp->zoneid,
|
|
uri_comp->zoneid_len);
|
|
if (netif == NULL) {
|
|
DEBUG("gcoap_dns: unable to interpret zoneid %*.s\n",
|
|
uri_comp->zoneid_len, uri_comp->zoneid);
|
|
return -ENOENT;
|
|
}
|
|
remote->netif = netif_get_id(netif);
|
|
}
|
|
else {
|
|
remote->netif = SOCK_ADDR_ANY_NETIF;
|
|
}
|
|
}
|
|
else if (ipv4_addr_from_buf((ipv4_addr_t *)remote->addr.ipv4,
|
|
uri_comp->host, uri_comp->host_len) != NULL) {
|
|
remote->family = AF_INET;
|
|
}
|
|
else if (IS_USED(MODULE_SOCK_DNS)) {
|
|
char hostname[uri_comp->host_len + 1];
|
|
|
|
strncpy(hostname, uri_comp->host, uri_comp->host_len);
|
|
hostname[uri_comp->host_len] = '\0';
|
|
int res = sock_dns_query(hostname, remote->addr.ipv6, AF_UNSPEC);
|
|
if (res < 0) {
|
|
DEBUG("gcoap_dns: unable to resolve hostname %s for remote\n",
|
|
hostname);
|
|
return -EHOSTUNREACH;
|
|
}
|
|
if (res == sizeof(remote->addr.ipv4)) {
|
|
remote->family = AF_INET;
|
|
}
|
|
else if (res == sizeof(remote->addr.ipv6)) {
|
|
remote->family = AF_INET6;
|
|
}
|
|
else {
|
|
DEBUG("gcoap_dns: Unable to determine address family of hostname %s.\n",
|
|
hostname);
|
|
return -EHOSTUNREACH;
|
|
}
|
|
}
|
|
else {
|
|
DEBUG("gcoap_dns: unable to resolve hostname in URI %s for remote\n",
|
|
_uri);
|
|
return -EHOSTUNREACH;
|
|
}
|
|
if (uri_comp->port == NULL) {
|
|
if (strncmp(_uri, "coap:", sizeof("coap:") - 1) == 0) {
|
|
remote->port = CONFIG_GCOAP_PORT;
|
|
}
|
|
else {
|
|
remote->port = CONFIG_GCOAPS_PORT;
|
|
}
|
|
}
|
|
else {
|
|
char port_str[uri_comp->port_len + 1];
|
|
|
|
strncpy(port_str, uri_comp->port, uri_comp->port_len);
|
|
port_str[uri_comp->port_len] = '\0';
|
|
remote->port = atoi(port_str);
|
|
if (remote->port == 0U) {
|
|
DEBUG("gcoap_dns: invalid port %s\n", port_str);
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int _gen_uri(uri_parser_result_t *uri_comp)
|
|
{
|
|
const char *uri = (_is_proxied()) ? _proxy : _uri;
|
|
int res = uri_parser_process_string(uri_comp, uri);
|
|
|
|
if (res < 0) {
|
|
return -EINVAL;
|
|
}
|
|
return strlen(uri);
|
|
}
|
|
|
|
static ssize_t _req_init(coap_pkt_t *pdu, uri_parser_result_t *uri_comp, bool con)
|
|
{
|
|
gcoap_req_init_path_buffer(pdu, pdu->payload, pdu->payload_len, COAP_METHOD_FETCH,
|
|
uri_comp->path, uri_comp->path_len);
|
|
if (con) {
|
|
coap_hdr_set_type(pdu->hdr, COAP_TYPE_CON);
|
|
}
|
|
|
|
if (coap_opt_add_format(pdu, COAP_FORMAT_DNS_MESSAGE) < 0) {
|
|
DEBUG("gcoap_dns: unable to add Content-Format option to request\n");
|
|
}
|
|
if (coap_opt_add_accept(pdu, COAP_FORMAT_DNS_MESSAGE) < 0) {
|
|
DEBUG("gcoap_dns: unable to add Accept option to request\n");
|
|
}
|
|
if (IS_USED(MODULE_GCOAP_DTLS) &&
|
|
(uri_comp->scheme_len == (sizeof("coaps") - 1)) &&
|
|
(strncmp(uri_comp->scheme, "coaps", uri_comp->scheme_len) == 0)) {
|
|
return GCOAP_SOCKET_TYPE_DTLS;
|
|
}
|
|
return GCOAP_SOCKET_TYPE_UDP;
|
|
}
|
|
|
|
static int _do_block(coap_pkt_t *pdu, const sock_udp_ep_t *remote,
|
|
_req_ctx_t *context)
|
|
{
|
|
gcoap_socket_type_t tl_type;
|
|
ssize_t len;
|
|
bool more;
|
|
coap_block_slicer_t slicer;
|
|
|
|
coap_block_slicer_init(&slicer, context->cur_blk_num++,
|
|
CONFIG_GCOAP_DNS_BLOCK_SIZE);
|
|
tl_type = _req_init(pdu, &_uri_comp, true);
|
|
if (tl_type == GCOAP_SOCKET_TYPE_UNDEF) {
|
|
return -EINVAL;
|
|
}
|
|
more = (CONFIG_GCOAP_DNS_PDU_BUF_SIZE >= CONFIG_DNS_MSG_LEN) ||
|
|
(slicer.end < context->dns_buf_len);
|
|
if (!more && (_add_init_block2_opt(pdu) < 0)) {
|
|
DEBUG("gcoap_dns: unable to add Block2 option to request\n");
|
|
return -ENOBUFS;
|
|
}
|
|
if (coap_opt_add_block1(pdu, &slicer, context->dns_buf_len > CONFIG_GCOAP_DNS_BLOCK_SIZE) < 0) {
|
|
DEBUG("gcoap_dns: unable to add Block1 option to request\n");
|
|
return -ENOBUFS;
|
|
}
|
|
if (_add_remaining_options(pdu, _uri, context) < 0) {
|
|
return -ENOBUFS;
|
|
}
|
|
len = coap_opt_finish(pdu, COAP_OPT_FINISH_PAYLOAD);
|
|
|
|
len += coap_blockwise_put_bytes(&slicer, pdu->payload,
|
|
context->dns_buf,
|
|
context->dns_buf_len);
|
|
|
|
coap_block1_finish(&slicer);
|
|
|
|
if ((len = _send(pdu->hdr, len, remote, slicer.start == 0, context, tl_type)) <= 0) {
|
|
DEBUG("gcoap_dns: msg send failed: %d\n", (int)len);
|
|
return len;
|
|
}
|
|
return len;
|
|
}
|
|
|
|
static ssize_t _req(_req_ctx_t *context)
|
|
{
|
|
coap_pkt_t *pdu = context->pkt;
|
|
ssize_t len;
|
|
|
|
if ((len = _gen_uri(&_uri_comp)) < 0) {
|
|
DEBUG("gcoap_dns: unable to parse URI\n");
|
|
return len;
|
|
}
|
|
if ((len = _set_remote(&_uri_comp, &_remote)) < 0) {
|
|
return len;
|
|
}
|
|
if (context->dns_buf_len > CONFIG_GCOAP_DNS_BLOCK_SIZE) {
|
|
context->cur_blk_num = 0U;
|
|
return _do_block(pdu, &_remote, context);
|
|
}
|
|
else {
|
|
gcoap_socket_type_t tl_type = _req_init(pdu, &_uri_comp, true);
|
|
if (tl_type == GCOAP_SOCKET_TYPE_UNDEF) {
|
|
return -EINVAL;
|
|
}
|
|
if (_add_init_block2_opt(pdu) < 0) {
|
|
DEBUG("gcoap_dns: unable to add Block2 option to request\n");
|
|
return -ENOBUFS;
|
|
}
|
|
if (_add_remaining_options(pdu, _uri, context) < 0) {
|
|
DEBUG("gcoap_dns: unable to add Proxy-URI option to request\n");
|
|
return -ENOBUFS;
|
|
}
|
|
len = coap_opt_finish(pdu, COAP_OPT_FINISH_PAYLOAD);
|
|
memcpy(pdu->payload, context->dns_buf, context->dns_buf_len);
|
|
return _send(pdu->hdr, len + context->dns_buf_len, &_remote, true, context, tl_type);
|
|
}
|
|
}
|
|
|
|
static int _dns_query(const char *domain_name, _req_ctx_t *req_ctx)
|
|
{
|
|
int res;
|
|
|
|
assert(domain_name != NULL);
|
|
if (!_dns_server_uri_isset()) {
|
|
DEBUG("gcoap_dns: no URI template provided\n");
|
|
return -ECONNREFUSED;
|
|
}
|
|
req_ctx->dns_buf_len = _dns_msg_compose(
|
|
req_ctx->dns_buf,
|
|
domain_name,
|
|
req_ctx->family
|
|
);
|
|
#if IS_USED(MODULE_DNS_CACHE)
|
|
req_ctx->domain_name = domain_name;
|
|
#endif
|
|
#if IS_USED(MODULE_GCOAP_DTLS)
|
|
req_ctx->req_tag = _req_tag++;
|
|
#endif
|
|
res = _req(req_ctx);
|
|
if (res <= 0) {
|
|
DEBUG("gcoap_dns: unable to send request\n");
|
|
if (res == 0) {
|
|
res = -ENOTRECOVERABLE;
|
|
}
|
|
}
|
|
return res;
|
|
}
|
|
|
|
static const char *_domain_name_from_ctx(_req_ctx_t *context)
|
|
{
|
|
#if IS_USED(MODULE_DNS_CACHE)
|
|
return context->domain_name;
|
|
#else
|
|
(void)context;
|
|
return NULL;
|
|
#endif
|
|
}
|
|
|
|
static void _resp_handler(const gcoap_request_memo_t *memo, coap_pkt_t *pdu,
|
|
const sock_udp_ep_t *remote)
|
|
{
|
|
coap_block1_t block;
|
|
_req_ctx_t *context = memo->context;
|
|
void *data;
|
|
size_t data_len;
|
|
int family = context->family;
|
|
|
|
if (memo->state == GCOAP_MEMO_TIMEOUT) {
|
|
DEBUG("gcoap_dns: CoAP request timed out\n");
|
|
context->res = -ETIMEDOUT;
|
|
goto unlock;
|
|
}
|
|
else if (memo->state != GCOAP_MEMO_RESP) {
|
|
DEBUG("gcoap_dns: error in response\n");
|
|
context->res = -EBADMSG;
|
|
goto unlock;
|
|
}
|
|
|
|
if ((remote->port != _remote.port) ||
|
|
(remote->family != _remote.family) ||
|
|
(!ipv6_addr_is_unspecified((ipv6_addr_t *)_remote.addr.ipv6) &&
|
|
(memcmp(remote->addr.ipv6, _remote.addr.ipv6,
|
|
sizeof(_remote.addr.ipv6)) != 0))) {
|
|
DEBUG("gcoap_dns: unexpected remote for reply\n");
|
|
context->res = -EDESTADDRREQ;
|
|
goto unlock;
|
|
}
|
|
if (coap_get_code_class(pdu) != COAP_CLASS_SUCCESS) {
|
|
DEBUG("gcoap_dns: unsuccessful response: %1u.%02u\n",
|
|
coap_get_code_class(pdu), coap_get_code_detail(pdu));
|
|
context->res = -EBADMSG;
|
|
goto unlock;
|
|
}
|
|
if (coap_get_code_raw(pdu) == COAP_CODE_CONTINUE) {
|
|
int res = _do_block(pdu, remote, context);
|
|
|
|
if (res < 0) {
|
|
context->res = res;
|
|
goto unlock;
|
|
}
|
|
return;
|
|
}
|
|
if (coap_get_block2(pdu, &block)) {
|
|
uint8_t *dns_buf = context->dns_buf;
|
|
|
|
if ((block.offset + pdu->payload_len) > CONFIG_DNS_MSG_LEN) {
|
|
DEBUG("gcoap_dns: No buffer space for block-wise transfer "
|
|
"(%lu + %u) > %u\n", (long unsigned)block.offset,
|
|
pdu->payload_len, CONFIG_DNS_MSG_LEN);
|
|
context->res = -ENOBUFS;
|
|
goto unlock;
|
|
}
|
|
memcpy(&dns_buf[block.offset], pdu->payload, pdu->payload_len);
|
|
if (block.blknum == 0) {
|
|
context->dns_buf_len = pdu->payload_len;
|
|
if (block.more && strlen(_uri) == 0) {
|
|
DEBUG("gcoap_dns: Cannot complete block-wise\n");
|
|
context->res = -EINVAL;
|
|
goto unlock;
|
|
}
|
|
}
|
|
else {
|
|
context->dns_buf_len += pdu->payload_len;
|
|
}
|
|
if (block.more) {
|
|
gcoap_socket_type_t tl_type;
|
|
unsigned msg_type = coap_get_type(pdu);
|
|
int len;
|
|
|
|
pdu->payload = (uint8_t *)pdu->hdr;
|
|
pdu->payload_len = CONFIG_GCOAP_DNS_PDU_BUF_SIZE;
|
|
tl_type = _req_init(pdu, &_uri_comp, msg_type == COAP_TYPE_ACK);
|
|
block.blknum++;
|
|
if (coap_opt_add_block2_control(pdu, &block) < 0) {
|
|
DEBUG("gcoap_dns: unable to add Block2 option to request\n");
|
|
context->res = -ENOBUFS;
|
|
goto unlock;
|
|
}
|
|
if (_add_remaining_options(pdu, _uri, context) < 0) {
|
|
context->res = -ENOBUFS;
|
|
goto unlock;
|
|
}
|
|
len = coap_opt_finish(pdu, COAP_OPT_FINISH_NONE);
|
|
if ((len = _send((uint8_t *)pdu->hdr, len, remote, false, context, tl_type)) <= 0) {
|
|
DEBUG("gcoap_dns: Unable to request next block: %d\n", len);
|
|
context->res = len;
|
|
goto unlock;
|
|
}
|
|
return;
|
|
}
|
|
data = context->dns_buf;
|
|
data_len = context->dns_buf_len;
|
|
}
|
|
else {
|
|
data = pdu->payload;
|
|
data_len = pdu->payload_len;
|
|
}
|
|
switch (coap_get_content_type(pdu)) {
|
|
case COAP_FORMAT_DNS_MESSAGE:
|
|
case COAP_FORMAT_NONE: {
|
|
uint32_t ttl;
|
|
|
|
context->res = dns_msg_parse_reply(data, data_len, family,
|
|
context->addr_out, &ttl);
|
|
if (context->res > 0) {
|
|
dns_cache_add(_domain_name_from_ctx(context), context->addr_out, context->res, ttl);
|
|
}
|
|
else if (ENABLE_DEBUG && (context->res < 0)) {
|
|
DEBUG("gcoap_dns: Unable to parse DNS reply: %d\n",
|
|
context->res);
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
context->res = -ENOMSG;
|
|
break;
|
|
}
|
|
unlock:
|
|
mutex_unlock(&context->resp_wait);
|
|
}
|
|
|
|
static ssize_t _send(const void *buf, size_t len, const sock_udp_ep_t *remote,
|
|
bool lock_resp_wait, _req_ctx_t *context, gcoap_socket_type_t tl_type)
|
|
{
|
|
if (lock_resp_wait) {
|
|
mutex_lock(&context->resp_wait);
|
|
}
|
|
return gcoap_req_send_tl(buf, len, remote, _resp_handler, context, tl_type);
|
|
}
|
|
/** @} */
|