blob: 76bbdb9a8c3abbbd2b06b02a7082abcab6590e10 [file] [log] [blame]
/*
*
* Copyright (c) 2020 Project CHIP Authors
* Copyright (c) 2013-2017 Nest Labs, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* @file
* This file implements DNSResolver, the object that abstracts
* Domain Name System (DNS) resolution in InetLayer.
*
*/
#include <inet/DNSResolver.h>
#include <inet/InetLayer.h>
#include <inet/InetLayerEvents.h>
#include <support/CodeUtils.h>
#include <string.h>
#if CHIP_SYSTEM_CONFIG_USE_LWIP
#include <lwip/dns.h>
#include <lwip/init.h>
#include <lwip/tcpip.h>
#if LWIP_VERSION_MAJOR < 2
#define LWIP_DNS_FOUND_CALLBACK_TYPE dns_found_callback
#endif // LWIP_VERSION_MAJOR < 2
#endif // CHIP_SYSTEM_CONFIG_USE_LWIP
#if CHIP_SYSTEM_CONFIG_USE_SOCKETS
#include <errno.h>
#include <netdb.h>
#include <stdlib.h>
#endif // CHIP_SYSTEM_CONFIG_USE_SOCKETS
namespace chip {
namespace Inet {
chip::System::ObjectPool<DNSResolver, INET_CONFIG_NUM_DNS_RESOLVERS> DNSResolver::sPool;
/**
* This method revolves a host name into a list of IP addresses.
*
* @note
* Even if the operation completes successfully,
* the result might be a zero-length list of IP addresses.
* Most of the error generated are returned via the
* application callback.
*
* @param[in] hostName A pointer to a C string representing the host name
* to be queried.
* @param[in] hostNameLen The string length of host name.
* @param[in] maxAddrs The maximum number of addresses to store in the DNS
* table.
* @param[in] options An integer value controlling how host name address
* resolution is performed. Values are from the #DNSOptions
* enumeration.
* @param[in] addrArray A pointer to the DNS table.
* @param[in] onComplete A pointer to the callback function when a DNS
* request is complete.
* @param[in] appState A pointer to the application state to be passed to
* onComplete when a DNS request is complete.
*
* @retval CHIP_NO_ERROR if a DNS request is handled
* successfully.
*
* @retval CHIP_ERROR_NOT_IMPLEMENTED if DNS resolution is not enabled on
* the underlying platform.
*
* @retval _other_ if other POSIX network or OS error
* was returned by the underlying DNS
* resolver implementation.
*
*/
CHIP_ERROR DNSResolver::Resolve(const char * hostName, uint16_t hostNameLen, uint8_t options, uint8_t maxAddrs,
IPAddress * addrArray, DNSResolver::OnResolveCompleteFunct onComplete, void * appState)
{
CHIP_ERROR res = CHIP_NO_ERROR;
#if !CHIP_SYSTEM_CONFIG_USE_SOCKETS && !LWIP_DNS
Release();
return CHIP_ERROR_NOT_IMPLEMENTED;
#endif // !CHIP_SYSTEM_CONFIG_USE_SOCKETS && !LWIP_DNS
uint8_t addrFamilyOption = (options & kDNSOption_AddrFamily_Mask);
uint8_t optionFlags = (options & kDNSOption_Flags_Mask);
// Check that the supplied options are valid.
if ((addrFamilyOption != kDNSOption_AddrFamily_Any && addrFamilyOption != kDNSOption_AddrFamily_IPv4Only &&
addrFamilyOption != kDNSOption_AddrFamily_IPv4Preferred && addrFamilyOption != kDNSOption_AddrFamily_IPv6Only &&
addrFamilyOption != kDNSOption_AddrFamily_IPv6Preferred) ||
(optionFlags & ~kDNSOption_ValidFlags) != 0)
{
Release();
return CHIP_ERROR_INVALID_ARGUMENT;
}
#if CHIP_SYSTEM_CONFIG_USE_SOCKETS || (CHIP_SYSTEM_CONFIG_USE_LWIP && LWIP_DNS)
// TODO: Eliminate the need for a local buffer when running on LwIP by changing
// the LwIP DNS interface to support non-nul terminated strings.
char hostNameBuf[NL_DNS_HOSTNAME_MAX_LEN + 1]; // DNS limits hostnames to 253 max characters.
memcpy(hostNameBuf, hostName, hostNameLen);
hostNameBuf[hostNameLen] = 0;
AppState = appState;
AddrArray = addrArray;
MaxAddrs = maxAddrs;
NumAddrs = 0;
DNSOptions = options;
OnComplete = onComplete;
#if CHIP_SYSTEM_CONFIG_USE_LWIP
#if LWIP_VERSION_MAJOR > 1 || LWIP_VERSION_MINOR >= 5
u8_t lwipAddrType;
#if INET_CONFIG_ENABLE_IPV4
switch (addrFamilyOption)
{
case kDNSOption_AddrFamily_IPv4Only:
lwipAddrType = LWIP_DNS_ADDRTYPE_IPV4;
break;
case kDNSOption_AddrFamily_Any:
case kDNSOption_AddrFamily_IPv4Preferred:
lwipAddrType = LWIP_DNS_ADDRTYPE_IPV4_IPV6;
break;
case kDNSOption_AddrFamily_IPv6Only:
lwipAddrType = LWIP_DNS_ADDRTYPE_IPV6;
break;
case kDNSOption_AddrFamily_IPv6Preferred:
lwipAddrType = LWIP_DNS_ADDRTYPE_IPV6_IPV4;
break;
default:
chipDie();
}
#else // INET_CONFIG_ENABLE_IPV4
lwipAddrType = LWIP_DNS_ADDRTYPE_IPV6;
#endif // INET_CONFIG_ENABLE_IPV4
#else // LWIP_VERSION_MAJOR <= 1 || LWIP_VERSION_MINOR < 5
#if INET_CONFIG_ENABLE_IPV4
if (addrFamilyOption == kDNSOption_AddrFamily_IPv6Only)
#endif
{
Release();
return CHIP_ERROR_NOT_IMPLEMENTED;
}
#endif // LWIP_VERSION_MAJOR <= 1 || LWIP_VERSION_MINOR < 5
// Lock LwIP stack
LOCK_TCPIP_CORE();
ip_addr_t lwipAddr;
LWIP_DNS_FOUND_CALLBACK_TYPE lwipCallback = reinterpret_cast<LWIP_DNS_FOUND_CALLBACK_TYPE>(LwIPHandleResolveComplete);
err_t lwipErr =
#if LWIP_VERSION_MAJOR > 1 || LWIP_VERSION_MINOR >= 5
dns_gethostbyname_addrtype(hostNameBuf, &lwipAddr, lwipCallback, this, lwipAddrType);
#else
dns_gethostbyname(hostNameBuf, &lwipAddr, lwipCallback, this);
#endif
// Unlock LwIP stack
UNLOCK_TCPIP_CORE();
if (lwipErr == ERR_OK)
{
chip::System::Layer & lSystemLayer = SystemLayer();
#if LWIP_VERSION_MAJOR > 1 || LWIP_VERSION_MINOR >= 5
AddrArray[0] = IPAddress::FromLwIPAddr(lwipAddr);
#else
AddrArray[0] = IPAddress::FromIPv4(lwipAddr);
#endif
NumAddrs = 1;
lSystemLayer.PostEvent(*this, kInetEvent_DNSResolveComplete, 0);
}
else if (lwipErr != ERR_INPROGRESS)
{
res = chip::System::MapErrorLwIP(lwipErr);
Release();
}
return res;
#endif // CHIP_SYSTEM_CONFIG_USE_LWIP
#if CHIP_SYSTEM_CONFIG_USE_SOCKETS
struct addrinfo gaiHints;
struct addrinfo * gaiResults = nullptr;
int gaiReturnCode;
// Configure the hints argument for getaddrinfo()
InitAddrInfoHints(gaiHints);
// Call getaddrinfo() to perform the name resolution.
gaiReturnCode = getaddrinfo(hostNameBuf, nullptr, &gaiHints, &gaiResults);
// Process the return code and results list returned by getaddrinfo(). If the call
// was successful this will copy the resultant addresses into the caller's array.
res = ProcessGetAddrInfoResult(gaiReturnCode, gaiResults);
// Invoke the caller's completion function.
onComplete(appState, res, NumAddrs, addrArray);
// Release DNSResolver object.
Release();
return CHIP_NO_ERROR;
#endif // CHIP_SYSTEM_CONFIG_USE_SOCKETS
#endif // CHIP_SYSTEM_CONFIG_USE_SOCKETS || (CHIP_SYSTEM_CONFIG_USE_LWIP && LWIP_DNS)
}
/**
* This method cancels DNS requests that are in progress.
*
* @retval CHIP_NO_ERROR.
*
*/
CHIP_ERROR DNSResolver::Cancel()
{
#if CHIP_SYSTEM_CONFIG_USE_LWIP
// NOTE: LwIP does not support canceling DNS requests that are in progress. As a consequence,
// we can't release the DNSResolver object until LwIP calls us back (because LwIP retains a
// pointer to the DNSResolver object while the request is active). However, now that the
// application has called Cancel() we have to make sure to NOT call their OnComplete function
// when the request completes.
//
// To ensure the right thing happens, we NULL the OnComplete pointer here, which signals the
// code in HandleResolveComplete() and LwIPHandleResolveComplete() to not interact with the
// application's state data (AddrArray) and to not call the application's callback. This has
// to happen with the LwIP lock held, since LwIPHandleResolveComplete() runs on LwIP's thread.
// Lock LwIP stack
LOCK_TCPIP_CORE();
// Signal that the request has been canceled by clearing the state of the resolver object.
OnComplete = NULL;
AddrArray = NULL;
MaxAddrs = 0;
NumAddrs = 0;
// Unlock LwIP stack
UNLOCK_TCPIP_CORE();
#endif // CHIP_SYSTEM_CONFIG_USE_LWIP
#if CHIP_SYSTEM_CONFIG_USE_SOCKETS
#if INET_CONFIG_ENABLE_ASYNC_DNS_SOCKETS
// NOTE: DNS lookups can be canceled only when using the asynchronous mode.
InetLayer & inet = Layer();
OnComplete = nullptr;
AppState = nullptr;
inet.mAsyncDNSResolver.Cancel(*this);
#endif // INET_CONFIG_ENABLE_ASYNC_DNS_SOCKETS
#endif // CHIP_SYSTEM_CONFIG_USE_SOCKETS
return CHIP_NO_ERROR;
}
#if CHIP_SYSTEM_CONFIG_USE_LWIP
/**
* This method is called by InetLayer on success, failure, or timeout of a
* DNS request.
*
*/
void DNSResolver::HandleResolveComplete()
{
// Call the application's completion handler if the request hasn't been canceled.
if (OnComplete != NULL)
OnComplete(AppState, (NumAddrs > 0) ? CHIP_NO_ERROR : INET_ERROR_HOST_NOT_FOUND, NumAddrs, AddrArray);
// Release the resolver object.
Release();
}
/**
* This method is called by LwIP network stack on success, failure, or timeout
* of a DNS request.
*
* @param[in] name A pointer to a NULL-terminated C string
* representing the host name that is queried.
* @param[in] ipaddr A pointer to a list of resolved IP addresses.
* @param[in] callback_arg A pointer to the arguments that are passed to
* the callback function.
*
*/
#if LWIP_VERSION_MAJOR > 1
void DNSResolver::LwIPHandleResolveComplete(const char * name, const ip_addr_t * ipaddr, void * callback_arg)
#else // LWIP_VERSION_MAJOR <= 1
void DNSResolver::LwIPHandleResolveComplete(const char * name, ip_addr_t * ipaddr, void * callback_arg)
#endif // LWIP_VERSION_MAJOR <= 1
{
DNSResolver * resolver = (DNSResolver *) callback_arg;
if (resolver != NULL)
{
chip::System::Layer & lSystemLayer = resolver->SystemLayer();
// Copy the resolved address to the application supplied buffer, but only if the request hasn't been canceled.
if (resolver->OnComplete != NULL && ipaddr != NULL)
{
#if LWIP_VERSION_MAJOR > 1 || LWIP_VERSION_MINOR >= 5
resolver->AddrArray[0] = IPAddress::FromLwIPAddr(*ipaddr);
#else
resolver->AddrArray[0] = IPAddress::FromIPv4(*ipaddr);
#endif
resolver->NumAddrs = 1;
}
lSystemLayer.PostEvent(*resolver, kInetEvent_DNSResolveComplete, 0);
}
}
#endif // CHIP_SYSTEM_CONFIG_USE_LWIP
#if CHIP_SYSTEM_CONFIG_USE_SOCKETS
void DNSResolver::InitAddrInfoHints(struct addrinfo & hints)
{
uint8_t addrFamilyOption = (DNSOptions & kDNSOption_AddrFamily_Mask);
memset(&hints, 0, sizeof(hints));
#if INET_CONFIG_ENABLE_IPV4
if (addrFamilyOption == kDNSOption_AddrFamily_IPv4Only)
{
hints.ai_family = AF_INET;
}
else if (addrFamilyOption == kDNSOption_AddrFamily_IPv6Only)
{
hints.ai_family = AF_INET6;
}
else
{
hints.ai_family = AF_UNSPEC;
}
#else // INET_CONFIG_ENABLE_IPV4
hints.ai_family = AF_INET6;
#endif // INET_CONFIG_ENABLE_IPV4
hints.ai_flags = AI_ADDRCONFIG;
}
CHIP_ERROR DNSResolver::ProcessGetAddrInfoResult(int returnCode, struct addrinfo * results)
{
CHIP_ERROR err = CHIP_NO_ERROR;
// If getaddrinfo() succeeded, copy addresses in the returned addrinfo structures into the
// application's output array...
if (returnCode == 0)
{
NumAddrs = 0;
#if INET_CONFIG_ENABLE_IPV4
// Based on the address family option specified by the application, determine which
// types of addresses should be returned and the order in which they should appear.
uint8_t addrFamilyOption = (DNSOptions & kDNSOption_AddrFamily_Mask);
int primaryFamily, secondaryFamily;
switch (addrFamilyOption)
{
case kDNSOption_AddrFamily_Any:
primaryFamily = AF_UNSPEC;
secondaryFamily = AF_UNSPEC;
break;
case kDNSOption_AddrFamily_IPv4Only:
primaryFamily = AF_INET;
secondaryFamily = AF_UNSPEC;
break;
case kDNSOption_AddrFamily_IPv4Preferred:
primaryFamily = AF_INET;
secondaryFamily = AF_INET6;
break;
case kDNSOption_AddrFamily_IPv6Only:
primaryFamily = AF_INET6;
secondaryFamily = AF_UNSPEC;
break;
case kDNSOption_AddrFamily_IPv6Preferred:
primaryFamily = AF_INET6;
secondaryFamily = AF_INET;
break;
default:
chipDie();
}
// Determine the number of addresses of each family present in the results.
// In the case of the secondary address family, only count these if they are
// to be returned in the results.
uint8_t numPrimaryAddrs = CountAddresses(primaryFamily, results);
uint8_t numSecondaryAddrs = (secondaryFamily != AF_UNSPEC) ? CountAddresses(secondaryFamily, results) : 0;
// Make sure numAddrs can actually fit the sum.
uint16_t numAddrs = static_cast<uint16_t>(numPrimaryAddrs + numSecondaryAddrs);
// If the total number of addresses to be returned exceeds the application
// specified max, ensure that at least 1 address from the secondary family
// appears in the result (unless of course there are no such addresses or
// the max is set to 1).
// This ensures the application will try at least one secondary address
// when attempting to communicate with the host.
if (numAddrs > MaxAddrs && MaxAddrs > 1 && numPrimaryAddrs > 0 && numSecondaryAddrs > 0)
{
numPrimaryAddrs = ::chip::min(numPrimaryAddrs, static_cast<uint8_t>(MaxAddrs - 1));
}
// Copy the primary addresses into the beginning of the application's output array,
// up to the limit determined above.
CopyAddresses(primaryFamily, numPrimaryAddrs, results);
// If secondary addresses are being returned, copy them into the output array after
// the primary addresses.
if (numSecondaryAddrs != 0)
{
CopyAddresses(secondaryFamily, numSecondaryAddrs, results);
}
#else // INET_CONFIG_ENABLE_IPV4
// Copy IPv6 addresses into the application's output array.
CopyAddresses(AF_INET6, UINT8_MAX, results);
#endif // INET_CONFIG_ENABLE_IPV4
// If in the end no addresses were returned, treat this as a "host not found" error.
if (NumAddrs == 0)
{
err = INET_ERROR_HOST_NOT_FOUND;
}
}
// Otherwise, getaddrinfo() failed, so translate the return code to an appropriate
// Inet error...
else
{
switch (returnCode)
{
case EAI_NONAME:
case EAI_NODATA:
case EAI_ADDRFAMILY:
// Each of these errors is translated to "host not found" for simplicity at the
// application layer. On most systems, the errors have the following meanings:
// EAI_NONAME is returned when there are no DNS records for the requested host name.
// EAI_NODATA is returned when there are no host records (A or AAAA) for the requested
// name, but other records do exist (e.g. MX or TXT).
// EAI_ADDRFAMILY is returned when a text-form address is given as the name, but its
// address family (IPv4 or IPv6) does not match the value specified in hints.ai_family.
err = INET_ERROR_HOST_NOT_FOUND;
break;
case EAI_AGAIN:
err = INET_ERROR_DNS_TRY_AGAIN;
break;
case EAI_SYSTEM:
err = chip::System::MapErrorPOSIX(errno);
break;
default:
err = INET_ERROR_DNS_NO_RECOVERY;
break;
}
}
// Free the results structure.
if (results != nullptr)
freeaddrinfo(results);
return err;
}
void DNSResolver::CopyAddresses(int family, uint8_t count, const struct addrinfo * addrs)
{
for (const struct addrinfo * addr = addrs; addr != nullptr && NumAddrs < MaxAddrs && count > 0; addr = addr->ai_next)
{
if (family == AF_UNSPEC || addr->ai_addr->sa_family == family)
{
AddrArray[NumAddrs++] = IPAddress::FromSockAddr(*addr->ai_addr);
count--;
}
}
}
uint8_t DNSResolver::CountAddresses(int family, const struct addrinfo * addrs)
{
uint8_t count = 0;
for (const struct addrinfo * addr = addrs; addr != nullptr && count < UINT8_MAX; addr = addr->ai_next)
{
if (family == AF_UNSPEC || addr->ai_addr->sa_family == family)
{
count++;
}
}
return count;
}
#if INET_CONFIG_ENABLE_ASYNC_DNS_SOCKETS
void DNSResolver::HandleAsyncResolveComplete()
{
// Copy the resolved address to the application supplied buffer, but only if the request hasn't been canceled.
if (OnComplete && mState != kState_Canceled)
{
OnComplete(AppState, asyncDNSResolveResult, NumAddrs, AddrArray);
}
Release();
}
#endif // INET_CONFIG_ENABLE_ASYNC_DNS_SOCKETS
#endif // CHIP_SYSTEM_CONFIG_USE_SOCKETS
} // namespace Inet
} // namespace chip