blob: 7a5f2574f8dc9b98766940c504dac7b0fd834f49 [file] [log] [blame]
/*
*
* Copyright (c) 2020 Project CHIP Authors
* Copyright (c) 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 AsyncDNSResolverSockets, the object that implements
* Asynchronous Domain Name System (DNS) resolution in InetLayer.
*
*/
#include <inet/InetLayer.h>
#include <support/CodeUtils.h>
#include <support/logging/CHIPLogging.h>
#include <errno.h>
#include <stdlib.h>
#include <string.h>
#if CHIP_SYSTEM_CONFIG_USE_SOCKETS
#if INET_CONFIG_ENABLE_DNS_RESOLVER && INET_CONFIG_ENABLE_ASYNC_DNS_SOCKETS
#include "AsyncDNSResolverSockets.h"
namespace chip {
namespace Inet {
/**
* The explicit initializer for the AsynchronousDNSResolverSockets class.
* This initializes the mutex and semaphore variables and creates the
* threads for handling the asynchronous DNS resolution.
*
* @param[in] aInet A pointer to the InetLayer object.
*
* @retval #CHIP_NO_ERROR if initialization is
* successful.
* @retval other appropriate POSIX network or OS error.
*/
CHIP_ERROR AsyncDNSResolverSockets::Init(InetLayer * aInet)
{
CHIP_ERROR err = CHIP_NO_ERROR;
int pthreadErr;
mInet = aInet;
mAsyncDNSQueueHead = nullptr;
mAsyncDNSQueueTail = nullptr;
pthreadErr = pthread_cond_init(&mAsyncDNSCondVar, nullptr);
VerifyOrDie(pthreadErr == 0);
pthreadErr = pthread_mutex_init(&mAsyncDNSMutex, nullptr);
VerifyOrDie(pthreadErr == 0);
// Create the thread pool for asynchronous DNS resolution.
for (int i = 0; i < INET_CONFIG_DNS_ASYNC_MAX_THREAD_COUNT; i++)
{
pthreadErr = pthread_create(&mAsyncDNSThreadHandle[i], nullptr, &AsyncDNSThreadRun, this);
VerifyOrDie(pthreadErr == 0);
}
return err;
}
/**
* This is the explicit deinitializer of the AsyncDNSResolverSockets class
* and it takes care of shutting the threads down and destroying the mutex
* and semaphore variables.
*
* @retval #CHIP_NO_ERROR if shutdown is successful.
* @retval other appropriate POSIX network or OS error.
*/
CHIP_ERROR AsyncDNSResolverSockets::Shutdown()
{
CHIP_ERROR err = CHIP_NO_ERROR;
int pthreadErr;
AsyncMutexLock();
mInet->State = InetLayer::kState_ShutdownInProgress;
pthreadErr = pthread_cond_broadcast(&mAsyncDNSCondVar);
VerifyOrDie(pthreadErr == 0);
AsyncMutexUnlock();
// Have the CHIP thread join the thread pool for asynchronous DNS resolution.
for (pthread_t thread : mAsyncDNSThreadHandle)
{
pthreadErr = pthread_join(thread, nullptr);
VerifyOrDie(pthreadErr == 0);
}
pthreadErr = pthread_mutex_destroy(&mAsyncDNSMutex);
VerifyOrDie(pthreadErr == 0);
pthreadErr = pthread_cond_destroy(&mAsyncDNSCondVar);
VerifyOrDie(pthreadErr == 0);
return err;
}
/**
* This method prepares a DNSResolver object prior to asynchronous resolution.
*
* @param[in] resolver A reference to an allocated DNSResolver object.
*
* @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] options An integer value controlling how host name address
* resolution is performed. Values are from the #DNSOptions
* enumeration.
* @param[in] maxAddrs The maximum number of addresses to store in the DNS
* table.
* @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.
*
*/
CHIP_ERROR AsyncDNSResolverSockets::PrepareDNSResolver(DNSResolver & resolver, const char * hostName, uint16_t hostNameLen,
uint8_t options, uint8_t maxAddrs, IPAddress * addrArray,
DNSResolver::OnResolveCompleteFunct onComplete, void * appState)
{
CHIP_ERROR err = CHIP_NO_ERROR;
memcpy(resolver.asyncHostNameBuf, hostName, hostNameLen);
resolver.asyncHostNameBuf[hostNameLen] = 0;
resolver.MaxAddrs = maxAddrs;
resolver.NumAddrs = 0;
resolver.DNSOptions = options;
resolver.AddrArray = addrArray;
resolver.AppState = appState;
resolver.OnComplete = onComplete;
resolver.asyncDNSResolveResult = CHIP_NO_ERROR;
resolver.mState = DNSResolver::kState_Active;
resolver.pNextAsyncDNSResolver = nullptr;
return err;
}
/**
* Enqueue a DNSResolver object for asynchronous IP address resolution of a specified hostname.
*
* @param[in] resolver A reference to the DNSResolver object.
*
* @retval #CHIP_NO_ERROR if a DNS request is queued
* successfully.
* @retval #CHIP_ERROR_NO_MEMORY if the Inet layer resolver pool
* is full.
* @retval other appropriate POSIX network or OS error.
*
*/
CHIP_ERROR AsyncDNSResolverSockets::EnqueueRequest(DNSResolver & resolver)
{
CHIP_ERROR err = CHIP_NO_ERROR;
int pthreadErr;
AsyncMutexLock();
// Add the DNSResolver object to the queue.
if (mAsyncDNSQueueHead == nullptr)
{
mAsyncDNSQueueHead = &resolver;
}
if (mAsyncDNSQueueTail != nullptr)
{
mAsyncDNSQueueTail->pNextAsyncDNSResolver = &resolver;
}
mAsyncDNSQueueTail = &resolver;
pthreadErr = pthread_cond_signal(&mAsyncDNSCondVar);
VerifyOrDie(pthreadErr == 0);
AsyncMutexUnlock();
return err;
}
/**
* Dequeue a DNSResolver object from the queue if there is one.
* Make the worker thread block if there is no item in the queue.
*
*/
CHIP_ERROR AsyncDNSResolverSockets::DequeueRequest(DNSResolver ** outResolver)
{
CHIP_ERROR err = CHIP_NO_ERROR;
int pthreadErr;
AsyncMutexLock();
// block until there is work to do or we detect a shutdown
while ((mAsyncDNSQueueHead == nullptr) && (mInet->State == InetLayer::kState_Initialized))
{
pthreadErr = pthread_cond_wait(&mAsyncDNSCondVar, &mAsyncDNSMutex);
VerifyOrDie(pthreadErr == 0);
}
ChipLogDetail(Inet, "Async DNS worker thread woke up.");
// on shutdown, return NULL. Otherwise, pop the head of the DNS request queue
if (mInet->State != InetLayer::kState_Initialized)
{
*outResolver = nullptr;
}
else
{
*outResolver = const_cast<DNSResolver *>(mAsyncDNSQueueHead);
mAsyncDNSQueueHead = mAsyncDNSQueueHead->pNextAsyncDNSResolver;
if (mAsyncDNSQueueHead == nullptr)
{
// Queue is empty
mAsyncDNSQueueTail = nullptr;
}
}
AsyncMutexUnlock();
return err;
}
/**
* Cancel an outstanding DNS query that may still be active.
*
* @param[in] resolver A reference to the DNSResolver object.
*/
CHIP_ERROR AsyncDNSResolverSockets::Cancel(DNSResolver & resolver)
{
CHIP_ERROR err = CHIP_NO_ERROR;
AsyncMutexLock();
resolver.mState = DNSResolver::kState_Canceled;
AsyncMutexUnlock();
return err;
}
void AsyncDNSResolverSockets::UpdateDNSResult(DNSResolver & resolver, struct addrinfo * inLookupRes)
{
resolver.NumAddrs = 0;
for (struct addrinfo * addr = inLookupRes; addr != nullptr && resolver.NumAddrs < resolver.MaxAddrs;
addr = addr->ai_next, resolver.NumAddrs++)
{
resolver.AddrArray[resolver.NumAddrs] = IPAddress::FromSockAddr(*addr->ai_addr);
}
}
void AsyncDNSResolverSockets::Resolve(DNSResolver & resolver)
{
struct addrinfo gaiHints;
struct addrinfo * gaiResults = nullptr;
int gaiReturnCode;
// Configure the hints argument for getaddrinfo()
resolver.InitAddrInfoHints(gaiHints);
// Call getaddrinfo() to perform the name resolution.
gaiReturnCode = getaddrinfo(resolver.asyncHostNameBuf, nullptr, &gaiHints, &gaiResults);
// Mutex protects the read and write operation on resolver->mState
AsyncMutexLock();
// 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.
resolver.asyncDNSResolveResult = resolver.ProcessGetAddrInfoResult(gaiReturnCode, gaiResults);
// Set the DNS resolver state.
resolver.mState = DNSResolver::kState_Complete;
// Release lock.
AsyncMutexUnlock();
}
/* Event handler function for asynchronous DNS notification */
void AsyncDNSResolverSockets::DNSResultEventHandler(chip::System::Layer * aLayer, void * aAppState, CHIP_ERROR aError)
{
DNSResolver * resolver = static_cast<DNSResolver *>(aAppState);
if (resolver)
{
resolver->HandleAsyncResolveComplete();
}
}
void AsyncDNSResolverSockets::NotifyChipThread(DNSResolver * resolver)
{
// Post work item via Timer Event for the CHIP thread
chip::System::Layer & lSystemLayer = resolver->SystemLayer();
ChipLogDetail(Inet, "Posting DNS completion event to CHIP thread.");
lSystemLayer.ScheduleWork(AsyncDNSResolverSockets::DNSResultEventHandler, resolver);
}
void * AsyncDNSResolverSockets::AsyncDNSThreadRun(void * args)
{
CHIP_ERROR err = CHIP_NO_ERROR;
AsyncDNSResolverSockets * asyncResolver = static_cast<AsyncDNSResolverSockets *>(args);
while (true)
{
DNSResolver * request = nullptr;
// Dequeue a DNSResolver for resolution. This function would block until there
// is an item in the queue or shutdown has been called.
err = asyncResolver->DequeueRequest(&request);
// If shutdown has been called, DeQueue would return with an empty request.
// In that case, break out of the loop and exit thread.
VerifyOrExit(err == CHIP_NO_ERROR && request != nullptr, );
if (request->mState != DNSResolver::kState_Canceled)
{
asyncResolver->Resolve(*request);
}
asyncResolver->NotifyChipThread(request);
}
exit:
ChipLogDetail(Inet, "Async DNS worker thread exiting.");
return nullptr;
}
void AsyncDNSResolverSockets::AsyncMutexLock()
{
int pthreadErr;
pthreadErr = pthread_mutex_lock(&mAsyncDNSMutex);
VerifyOrDie(pthreadErr == 0);
}
void AsyncDNSResolverSockets::AsyncMutexUnlock()
{
int pthreadErr;
pthreadErr = pthread_mutex_unlock(&mAsyncDNSMutex);
VerifyOrDie(pthreadErr == 0);
}
} // namespace Inet
} // namespace chip
#endif // INET_CONFIG_ENABLE_DNS_RESOLVER && INET_CONFIG_ENABLE_ASYNC_DNS_SOCKETS
#endif // CHIP_SYSTEM_CONFIG_USE_SOCKETS