From 043ae55029275decd1e72b115d08745bc8891cce Mon Sep 17 00:00:00 2001 From: Wang Qixiang <43193572+wqx6@users.noreply.github.com> Date: Fri, 21 Apr 2023 09:38:05 +0800 Subject: [PATCH] ESP32: Send AAAA query when the SRV response does not include IP addresses records (#26139) ESP32: Improve the performance of platform mDNS resolving --- src/platform/ESP32/DnssdImpl.cpp | 334 ++++++++++++++++++++----------- src/platform/ESP32/DnssdImpl.h | 131 ++++++------ 2 files changed, 276 insertions(+), 189 deletions(-) diff --git a/src/platform/ESP32/DnssdImpl.cpp b/src/platform/ESP32/DnssdImpl.cpp index 1615ea9e1b9119..a16659a642141d 100644 --- a/src/platform/ESP32/DnssdImpl.cpp +++ b/src/platform/ESP32/DnssdImpl.cpp @@ -44,9 +44,9 @@ struct MdnsQuery }; static MdnsQuery * sQueryList = nullptr; -void MdnsQueryNotifier(mdns_search_once_t * searchHandle); +static void MdnsQueryNotifier(mdns_search_once_t * queryHandle); -CHIP_ERROR AddQueryList(GenericContext * ctx) +static CHIP_ERROR AddQueryList(GenericContext * ctx) { MdnsQuery * ret = static_cast(chip::Platform::MemoryAlloc(sizeof(MdnsQuery))); if (ret == nullptr) @@ -60,21 +60,36 @@ CHIP_ERROR AddQueryList(GenericContext * ctx) return CHIP_NO_ERROR; } -GenericContext * FindMdnsQuery(mdns_search_once_t * searchHandle) +static GenericContext * FindMdnsQuery(mdns_search_once_t * queryHandle) { MdnsQuery * current = sQueryList; while (current) { - if (current->ctx && current->ctx->mSearchHandle == searchHandle) + if (current->ctx) { - return current->ctx; + if (current->ctx->mContextType == ContextType::Browse) + { + BrowseContext * browseCtx = reinterpret_cast(current->ctx); + if (browseCtx->mPtrQueryHandle == queryHandle) + { + return current->ctx; + } + } + else if (current->ctx->mContextType == ContextType::Resolve) + { + ResolveContext * resolveCtx = reinterpret_cast(current->ctx); + if (resolveCtx->mSrvQueryHandle == queryHandle || resolveCtx->mTxtQueryHandle == queryHandle) + { + return current->ctx; + } + } } current = current->next; } return nullptr; } -CHIP_ERROR RemoveMdnsQuery(GenericContext * ctx) +static CHIP_ERROR RemoveMdnsQuery(GenericContext * ctx) { MdnsQuery * current = sQueryList; MdnsQuery * front = nullptr; @@ -200,7 +215,7 @@ CHIP_ERROR ChipDnssdFinalizeServiceUpdate() return CHIP_NO_ERROR; } -Inet::IPAddressType MapAddressType(mdns_ip_protocol_t ip_protocol) +static Inet::IPAddressType MapAddressType(mdns_ip_protocol_t ip_protocol) { switch (ip_protocol) { @@ -215,7 +230,7 @@ Inet::IPAddressType MapAddressType(mdns_ip_protocol_t ip_protocol) } } -TextEntry * GetTextEntry(mdns_txt_item_t * txt_array, uint8_t * txt_value_len, size_t txt_count) +static TextEntry * GetTextEntry(mdns_txt_item_t * txt_array, uint8_t * txt_value_len, size_t txt_count) { if (txt_count == 0 || txt_array == NULL) { @@ -234,7 +249,7 @@ TextEntry * GetTextEntry(mdns_txt_item_t * txt_array, uint8_t * txt_value_len, s return ret; } -CHIP_ERROR GetIPAddress(Inet::IPAddress & outIPAddress, mdns_ip_addr_t * mdnsIPAddr) +static CHIP_ERROR GetIPAddress(Inet::IPAddress & outIPAddress, mdns_ip_addr_t * mdnsIPAddr) { if (!mdnsIPAddr) { @@ -269,48 +284,48 @@ size_t GetResultSize(mdns_result_t * result) return ret; } -CHIP_ERROR OnBrowseDone(BrowseContext * ctx) +static CHIP_ERROR OnBrowseDone(BrowseContext * ctx) { CHIP_ERROR error = CHIP_NO_ERROR; mdns_result_t * currentResult = nullptr; size_t servicesIndex = 0; VerifyOrExit(ctx && ctx->mBrowseCb, error = CHIP_ERROR_INVALID_ARGUMENT); - if (ctx->mResult) + if (ctx->mPtrQueryResult) { - ctx->mServiceSize = GetResultSize(ctx->mResult); + ctx->mServiceSize = GetResultSize(ctx->mPtrQueryResult); if (ctx->mServiceSize > 0) { - ctx->mServices = static_cast(chip::Platform::MemoryCalloc(ctx->mServiceSize, sizeof(DnssdService))); - if (!ctx->mServices) + ctx->mService = static_cast(chip::Platform::MemoryCalloc(ctx->mServiceSize, sizeof(DnssdService))); + if (!ctx->mService) { ChipLogError(DeviceLayer, "Failed to alloc memory for Dnssd services"); ctx->mServiceSize = 0; error = CHIP_ERROR_NO_MEMORY; ExitNow(); } - currentResult = ctx->mResult; + currentResult = ctx->mPtrQueryResult; servicesIndex = 0; while (currentResult) { - Platform::CopyString(ctx->mServices[servicesIndex].mName, currentResult->instance_name); - Platform::CopyString(ctx->mServices[servicesIndex].mHostName, currentResult->hostname); - Platform::CopyString(ctx->mServices[servicesIndex].mType, currentResult->service_type); - ctx->mServices[servicesIndex].mProtocol = ctx->mProtocol; - ctx->mServices[servicesIndex].mAddressType = MapAddressType(currentResult->ip_protocol); - ctx->mServices[servicesIndex].mTransportType = ctx->mAddressType; - ctx->mServices[servicesIndex].mPort = currentResult->port; - ctx->mServices[servicesIndex].mInterface = ctx->mInterfaceId; - ctx->mServices[servicesIndex].mTextEntries = + Platform::CopyString(ctx->mService[servicesIndex].mName, currentResult->instance_name); + Platform::CopyString(ctx->mService[servicesIndex].mHostName, currentResult->hostname); + Platform::CopyString(ctx->mService[servicesIndex].mType, currentResult->service_type); + ctx->mService[servicesIndex].mProtocol = ctx->mProtocol; + ctx->mService[servicesIndex].mAddressType = MapAddressType(currentResult->ip_protocol); + ctx->mService[servicesIndex].mTransportType = ctx->mAddressType; + ctx->mService[servicesIndex].mPort = currentResult->port; + ctx->mService[servicesIndex].mInterface = ctx->mInterfaceId; + ctx->mService[servicesIndex].mTextEntries = GetTextEntry(currentResult->txt, currentResult->txt_value_len, currentResult->txt_count); - ctx->mServices[servicesIndex].mTextEntrySize = currentResult->txt_count; - ctx->mServices[servicesIndex].mSubTypes = NULL; - ctx->mServices[servicesIndex].mSubTypeSize = 0; + ctx->mService[servicesIndex].mTextEntrySize = currentResult->txt_count; + ctx->mService[servicesIndex].mSubTypes = NULL; + ctx->mService[servicesIndex].mSubTypeSize = 0; if (currentResult->addr) { Inet::IPAddress IPAddr; error = GetIPAddress(IPAddr, currentResult->addr); SuccessOrExit(error); - ctx->mServices[servicesIndex].mAddress.SetValue(IPAddr); + ctx->mService[servicesIndex].mAddress.SetValue(IPAddr); } currentResult = currentResult->next; servicesIndex++; @@ -318,7 +333,7 @@ CHIP_ERROR OnBrowseDone(BrowseContext * ctx) } } exit: - ctx->mBrowseCb(ctx->mCbContext, ctx->mServices, ctx->mServiceSize, true, error); + ctx->mBrowseCb(ctx->mCbContext, ctx->mService, ctx->mServiceSize, true, error); return RemoveMdnsQuery(reinterpret_cast(ctx)); } @@ -333,91 +348,95 @@ size_t GetAddressCount(mdns_ip_addr_t * addr) return ret; } -CHIP_ERROR OnResolveQuerySrvDone(ResolveContext * ctx) +static CHIP_ERROR ParseIPAddresses(ResolveContext * ctx) { - CHIP_ERROR error = CHIP_NO_ERROR; size_t addressIndex = 0; - - VerifyOrExit(ctx && ctx->mResolveCb, error = CHIP_ERROR_INVALID_ARGUMENT); - VerifyOrExit(ctx->mService == nullptr && ctx->mResolveState == ResolveContext::ResolveState::QuerySrv, - error = CHIP_ERROR_INCORRECT_STATE); - if (ctx->mResult) - { - ctx->mService = static_cast(chip::Platform::MemoryAlloc(sizeof(DnssdService))); - VerifyOrExit(ctx->mService != nullptr, error = CHIP_ERROR_NO_MEMORY); - Platform::CopyString(ctx->mService->mName, ctx->mResult->instance_name); - Platform::CopyString(ctx->mService->mHostName, ctx->mResult->hostname); - Platform::CopyString(ctx->mService->mType, ctx->mResult->service_type); - ctx->mService->mProtocol = ctx->mProtocol; - ctx->mService->mAddressType = MapAddressType(ctx->mResult->ip_protocol); - ctx->mService->mTransportType = ctx->mService->mAddressType; - ctx->mService->mPort = ctx->mResult->port; - ctx->mService->mInterface = ctx->mInterfaceId; - ctx->mService->mSubTypes = nullptr; - ctx->mService->mSubTypeSize = 0; - - if (ctx->mResult->addr) + if (ctx->mAddrQueryResult && ctx->mAddrQueryResult->addr) + { + ctx->mAddressCount = GetAddressCount(ctx->mAddrQueryResult->addr); + if (ctx->mAddressCount > 0) { - ctx->mAddressCount = GetAddressCount(ctx->mResult->addr); - if (ctx->mAddressCount > 0) + ctx->mAddresses = + static_cast(chip::Platform::MemoryCalloc(ctx->mAddressCount, sizeof(Inet::IPAddress))); + if (ctx->mAddresses == nullptr) { - ctx->mAddresses = - static_cast(chip::Platform::MemoryCalloc(ctx->mAddressCount, sizeof(Inet::IPAddress))); - if (ctx->mAddresses == nullptr) - { - ChipLogError(DeviceLayer, "Failed to alloc memory for addresses"); - error = CHIP_ERROR_NO_MEMORY; - ctx->mAddressCount = 0; - ExitNow(); - } - auto * addr = ctx->mResult->addr; - while (addr) - { - GetIPAddress(ctx->mAddresses[addressIndex], addr); - addressIndex++; - addr = addr->next; - } + ChipLogError(DeviceLayer, "Failed to alloc memory for addresses"); + ctx->mAddressCount = 0; + return CHIP_ERROR_NO_MEMORY; } - else + auto * addr = ctx->mAddrQueryResult->addr; + while (addr) { - ctx->mAddresses = nullptr; - ctx->mAddressCount = 0; + GetIPAddress(ctx->mAddresses[addressIndex], addr); + addressIndex++; + addr = addr->next; } + return CHIP_NO_ERROR; } } -exit: - if (error != CHIP_NO_ERROR) - { - ctx->mResolveCb(ctx->mCbContext, nullptr, Span(nullptr, 0), error); - RemoveMdnsQuery(reinterpret_cast(ctx)); - return error; - } - mdns_query_results_free(ctx->mResult); - mdns_query_async_delete(ctx->mSearchHandle); - ctx->mResult = nullptr; - ctx->mResolveState = ResolveContext::ResolveState::QueryTxt; - // then query the text entries - ctx->mSearchHandle = mdns_query_async_new(ctx->mInstanceName, ctx->mType, GetProtocolString(ctx->mProtocol), MDNS_TYPE_TXT, - kTimeoutMilli, kMaxResults, MdnsQueryNotifier); - return CHIP_NO_ERROR; + return CHIP_ERROR_INVALID_ARGUMENT; } -CHIP_ERROR OnResolveQueryTxtDone(ResolveContext * ctx) +static CHIP_ERROR ParseSrvResult(ResolveContext * ctx) { - CHIP_ERROR error = CHIP_NO_ERROR; + if (ctx->mSrvQueryResult) + { + if (!ctx->mService) + { + ctx->mService = static_cast(chip::Platform::MemoryAlloc(sizeof(DnssdService))); + } + VerifyOrReturnError(ctx->mService, CHIP_ERROR_NO_MEMORY); + ctx->mServiceSize = 1; + Platform::CopyString(ctx->mService->mName, ctx->mSrvQueryResult->instance_name); + Platform::CopyString(ctx->mService->mHostName, ctx->mSrvQueryResult->hostname); + Platform::CopyString(ctx->mService->mType, ctx->mSrvQueryResult->service_type); + ctx->mService->mProtocol = ctx->mProtocol; + ctx->mService->mAddressType = MapAddressType(ctx->mSrvQueryResult->ip_protocol); + ctx->mService->mTransportType = ctx->mService->mAddressType; + ctx->mService->mPort = ctx->mSrvQueryResult->port; + ctx->mService->mInterface = ctx->mInterfaceId; + ctx->mService->mSubTypes = nullptr; + ctx->mService->mSubTypeSize = 0; + return CHIP_NO_ERROR; + } + else + { + ctx->mService = nullptr; + ctx->mServiceSize = 0; + } + return CHIP_ERROR_INVALID_ARGUMENT; +} - VerifyOrExit(ctx && ctx->mResolveCb, error = CHIP_ERROR_INVALID_ARGUMENT); - VerifyOrExit(ctx->mService && ctx->mResolveState == ResolveContext::ResolveState::QueryTxt, error = CHIP_ERROR_INCORRECT_STATE); - if (ctx->mResult) +// ParseTxtResult should be called after ParseSrvResult +static CHIP_ERROR ParseTxtResult(ResolveContext * ctx) +{ + VerifyOrReturnError(ctx->mService, CHIP_ERROR_INCORRECT_STATE); + if (ctx->mTxtQueryResult) { - ctx->mService->mTextEntries = GetTextEntry(ctx->mResult->txt, ctx->mResult->txt_value_len, ctx->mResult->txt_count); - ctx->mService->mTextEntrySize = ctx->mResult->txt_count; + ctx->mService->mTextEntries = + GetTextEntry(ctx->mTxtQueryResult->txt, ctx->mTxtQueryResult->txt_value_len, ctx->mTxtQueryResult->txt_count); + ctx->mService->mTextEntrySize = ctx->mTxtQueryResult->txt_count; } else { ctx->mService->mTextEntries = nullptr; ctx->mService->mTextEntrySize = 0; } + return CHIP_NO_ERROR; +} + +static CHIP_ERROR OnResolveDone(ResolveContext * ctx) +{ + CHIP_ERROR error = CHIP_NO_ERROR; + + VerifyOrExit(ctx && ctx->mResolveCb, error = CHIP_ERROR_INVALID_ARGUMENT); + VerifyOrExit(!ctx->mService && ctx->mSrvAddrQueryFinished && ctx->mTxtQueryFinished, error = CHIP_ERROR_INCORRECT_STATE); + error = ParseSrvResult(ctx); + SuccessOrExit(error); + error = ParseIPAddresses(ctx); + SuccessOrExit(error); + error = ParseTxtResult(ctx); + SuccessOrExit(error); exit: if (error != CHIP_NO_ERROR) { @@ -431,40 +450,103 @@ CHIP_ERROR OnResolveQueryTxtDone(ResolveContext * ctx) return error; } -void MdnsQueryDone(intptr_t context) +static mdns_result_t * MdnsQueryGetResults(mdns_search_once_t * queryHandle) +{ + mdns_result_t * ret = nullptr; +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 0, 0) + if (mdns_query_async_get_results(queryHandle, kTimeoutMilli, &ret, NULL)) +#else + if (mdns_query_async_get_results(queryHandle, kTimeoutMilli, &ret)) +#endif + { + return ret; + } + return nullptr; +} + +static void MdnsQueryDone(intptr_t context) { if (!context) { return; } - mdns_search_once_t * searchHandle = reinterpret_cast(context); - GenericContext * ctx = FindMdnsQuery(searchHandle); -#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 0, 0) - if (mdns_query_async_get_results(searchHandle, kTimeoutMilli, &(ctx->mResult), NULL)) -#else - if (mdns_query_async_get_results(searchHandle, kTimeoutMilli, &(ctx->mResult))) -#endif // ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 0, 0) + mdns_search_once_t * queryHandle = reinterpret_cast(context); + mdns_result_t * result = MdnsQueryGetResults(queryHandle); + GenericContext * ctx = FindMdnsQuery(queryHandle); + if (!ctx) { - if (ctx->mContextType == ContextType::Browse) - { - OnBrowseDone(reinterpret_cast(ctx)); - } - else if (ctx->mContextType == ContextType::Resolve) + mdns_query_results_free(result); + mdns_query_async_delete(queryHandle); + return; + } + if (ctx->mContextType == ContextType::Browse) + { + BrowseContext * browseCtx = reinterpret_cast(ctx); + browseCtx->mPtrQueryResult = result; + OnBrowseDone(browseCtx); + } + else if (ctx->mContextType == ContextType::Resolve) + { + + ResolveContext * resolveCtx = reinterpret_cast(ctx); + if (resolveCtx->mSrvQueryHandle == queryHandle) { - ResolveContext * resolveCtx = reinterpret_cast(ctx); - if (resolveCtx->mResolveState == ResolveContext::ResolveState::QuerySrv) + // No result found. + if (!result) + { + resolveCtx->mResolveCb(ctx->mCbContext, nullptr, Span(nullptr, 0), CHIP_ERROR_INVALID_ARGUMENT); + RemoveMdnsQuery(ctx); + return; + } + // If SRV Query Result is empty, the result is for SRV Query. + if (!resolveCtx->mSrvQueryResult) + { + resolveCtx->mSrvQueryResult = result; + if (result->addr) + { + resolveCtx->mAddrQueryResult = result; + resolveCtx->mSrvAddrQueryFinished = true; + } + else + { + // If there is no A/AAAA records in SRV query response, we will send an AAAA query for the IP addresses. + mdns_query_async_delete(resolveCtx->mSrvQueryHandle); + resolveCtx->mAddrQueryResult = nullptr; + resolveCtx->mSrvQueryHandle = mdns_query_async_new(result->hostname, NULL, NULL, MDNS_TYPE_AAAA, kTimeoutMilli, + kMaxResults, MdnsQueryNotifier); + if (!resolveCtx->mSrvQueryHandle) + { + resolveCtx->mResolveCb(ctx->mCbContext, nullptr, Span(nullptr, 0), CHIP_ERROR_NO_MEMORY); + RemoveMdnsQuery(ctx); + return; + } + } + } + else if (!resolveCtx->mAddrQueryResult) { - OnResolveQuerySrvDone(resolveCtx); + resolveCtx->mAddrQueryResult = result; + resolveCtx->mSrvAddrQueryFinished = true; } - else if (resolveCtx->mResolveState == ResolveContext::ResolveState::QueryTxt) + else { - OnResolveQueryTxtDone(resolveCtx); + resolveCtx->mResolveCb(ctx->mCbContext, nullptr, Span(nullptr, 0), CHIP_ERROR_INCORRECT_STATE); + RemoveMdnsQuery(ctx); + return; } } + else if (resolveCtx->mTxtQueryHandle == queryHandle) + { + resolveCtx->mTxtQueryResult = result; + resolveCtx->mTxtQueryFinished = true; + } + if (resolveCtx->mTxtQueryFinished && resolveCtx->mSrvAddrQueryFinished) + { + OnResolveDone(resolveCtx); + } } } -void MdnsQueryNotifier(mdns_search_once_t * searchHandle) +static void MdnsQueryNotifier(mdns_search_once_t * searchHandle) { chip::DeviceLayer::PlatformMgr().ScheduleWork(MdnsQueryDone, reinterpret_cast(searchHandle)); } @@ -474,14 +556,15 @@ CHIP_ERROR ChipDnssdBrowse(const char * type, DnssdServiceProtocol protocol, chi intptr_t * browseIdentifier) { CHIP_ERROR error = CHIP_NO_ERROR; - mdns_search_once_t * searchHandle = + mdns_search_once_t * queryHandle = mdns_query_async_new(NULL, type, GetProtocolString(protocol), MDNS_TYPE_PTR, kTimeoutMilli, kMaxResults, MdnsQueryNotifier); + VerifyOrReturnError(queryHandle, CHIP_ERROR_NO_MEMORY); BrowseContext * ctx = - chip::Platform::New(type, protocol, interface, searchHandle, addressType, callback, context); + chip::Platform::New(type, protocol, interface, queryHandle, addressType, callback, context); if (!ctx) { ChipLogError(DeviceLayer, "Failed to alloc memory for browse context"); - mdns_query_async_delete(searchHandle); + mdns_query_async_delete(queryHandle); return CHIP_ERROR_NO_MEMORY; } error = AddQueryList(reinterpret_cast(ctx)); @@ -504,14 +587,23 @@ CHIP_ERROR ChipDnssdStopBrowse(intptr_t browseIdentifier) CHIP_ERROR ChipDnssdResolve(DnssdService * service, chip::Inet::InterfaceId interface, DnssdResolveCallback callback, void * context) { - CHIP_ERROR error = CHIP_NO_ERROR; - mdns_search_once_t * searchHandle = mdns_query_async_new(service->mName, service->mType, GetProtocolString(service->mProtocol), - MDNS_TYPE_SRV, kTimeoutMilli, kMaxResults, MdnsQueryNotifier); - ResolveContext * ctx = chip::Platform::New(service, interface, searchHandle, callback, context); + CHIP_ERROR error = CHIP_NO_ERROR; + mdns_search_once_t * querySrv = mdns_query_async_new(service->mName, service->mType, GetProtocolString(service->mProtocol), + MDNS_TYPE_SRV, kTimeoutMilli, kMaxResults, MdnsQueryNotifier); + VerifyOrReturnError(querySrv, CHIP_ERROR_NO_MEMORY); + mdns_search_once_t * queryTxt = mdns_query_async_new(service->mName, service->mType, GetProtocolString(service->mProtocol), + MDNS_TYPE_TXT, kTimeoutMilli, kMaxResults, MdnsQueryNotifier); + if (!queryTxt) + { + mdns_query_async_delete(querySrv); + return CHIP_ERROR_NO_MEMORY; + } + ResolveContext * ctx = chip::Platform::New(service, interface, querySrv, queryTxt, callback, context); if (!ctx) { ChipLogError(DeviceLayer, "Failed to alloc memory for resolve context"); - mdns_query_async_delete(searchHandle); + mdns_query_async_delete(querySrv); + mdns_query_async_delete(queryTxt); return CHIP_ERROR_NO_MEMORY; } error = AddQueryList(reinterpret_cast(ctx)); diff --git a/src/platform/ESP32/DnssdImpl.h b/src/platform/ESP32/DnssdImpl.h index 7f84eebb0fcbd9..658ec8fe67c7f6 100644 --- a/src/platform/ESP32/DnssdImpl.h +++ b/src/platform/ESP32/DnssdImpl.h @@ -32,57 +32,56 @@ enum class ContextType struct GenericContext { ContextType mContextType; - void * mCbContext; char mType[kDnssdTypeMaxSize + 1]; DnssdServiceProtocol mProtocol; Inet::InterfaceId mInterfaceId; - mdns_search_once_t * mSearchHandle; - mdns_result_t * mResult; -}; - -struct BrowseContext : public GenericContext -{ - DnssdBrowseCallback mBrowseCb; - Inet::IPAddressType mAddressType; - DnssdService * mServices; - size_t mServiceSize; - BrowseContext(const char * type, DnssdServiceProtocol protocol, Inet::InterfaceId ifId, mdns_search_once_t * searchHandle, - Inet::IPAddressType addrType, DnssdBrowseCallback cb, void * cbCtx) - + void * mCbContext; + DnssdService * mService = nullptr; + size_t mServiceSize = 0; + GenericContext(ContextType ctxType, const char * type, DnssdServiceProtocol protocol, Inet::InterfaceId ifId, void * cbCtx) : + mContextType(ctxType), mProtocol(protocol), mInterfaceId(ifId), mCbContext(cbCtx) { Platform::CopyString(mType, type); - mContextType = ContextType::Browse; - mAddressType = addrType; - mProtocol = protocol; - mBrowseCb = cb; - mCbContext = cbCtx; - mInterfaceId = ifId; - mSearchHandle = searchHandle; - mResult = nullptr; - mServices = nullptr; - mServiceSize = 0; } - - ~BrowseContext() + ~GenericContext() { - if (mServices && mServiceSize > 0) + if (mService && mServiceSize > 0) { for (size_t serviceIndex = 0; serviceIndex < mServiceSize; serviceIndex++) { - if (mServices[serviceIndex].mTextEntries) + if (mService[serviceIndex].mTextEntries) { - chip::Platform::MemoryFree(mServices[serviceIndex].mTextEntries); + chip::Platform::MemoryFree(mService[serviceIndex].mTextEntries); } } - chip::Platform::MemoryFree(mServices); + chip::Platform::MemoryFree(mService); } - if (mResult) + } +}; + +struct BrowseContext : public GenericContext +{ + DnssdBrowseCallback mBrowseCb; + Inet::IPAddressType mAddressType; + mdns_search_once_t * mPtrQueryHandle; + mdns_result_t * mPtrQueryResult = nullptr; + BrowseContext(const char * type, DnssdServiceProtocol protocol, Inet::InterfaceId ifId, mdns_search_once_t * queryHandle, + Inet::IPAddressType addrType, DnssdBrowseCallback cb, void * cbCtx) : + GenericContext(ContextType::Browse, type, protocol, ifId, cbCtx), + mBrowseCb(cb), mAddressType(addrType), mPtrQueryHandle(queryHandle) + + {} + + ~BrowseContext() + { + if (mPtrQueryResult) { - mdns_query_results_free(mResult); + mdns_query_results_free(mPtrQueryResult); } - if (mSearchHandle) + + if (mPtrQueryHandle) { - mdns_query_async_delete(mSearchHandle); + mdns_query_async_delete(mPtrQueryHandle); } } }; @@ -91,55 +90,51 @@ struct ResolveContext : public GenericContext { char mInstanceName[Common::kInstanceNameMaxLength + 1]; DnssdResolveCallback mResolveCb; - DnssdService * mService; - Inet::IPAddress * mAddresses; - size_t mAddressCount; + Inet::IPAddress * mAddresses = nullptr; + size_t mAddressCount = 0; - enum class ResolveState - { - QuerySrv, - QueryTxt, - } mResolveState; + mdns_search_once_t * mSrvQueryHandle; + mdns_result_t * mSrvQueryResult = nullptr; + mdns_result_t * mAddrQueryResult = nullptr; + bool mSrvAddrQueryFinished = false; + + mdns_search_once_t * mTxtQueryHandle; + mdns_result_t * mTxtQueryResult = nullptr; + bool mTxtQueryFinished = false; - ResolveContext(DnssdService * service, Inet::InterfaceId ifId, mdns_search_once_t * searchHandle, DnssdResolveCallback cb, - void * cbCtx) + ResolveContext(DnssdService * service, Inet::InterfaceId ifId, mdns_search_once_t * srvQuery, mdns_search_once_t * txtQuery, + DnssdResolveCallback cb, void * cbCtx) : + GenericContext(ContextType::Resolve, service->mType, service->mProtocol, ifId, cbCtx), + mResolveCb(cb), mSrvQueryHandle(srvQuery), mTxtQueryHandle(txtQuery) { - Platform::CopyString(mType, service->mType); Platform::CopyString(mInstanceName, service->mName); - mContextType = ContextType::Resolve; - mProtocol = service->mProtocol; - mResolveCb = cb; - mCbContext = cbCtx; - mInterfaceId = ifId; - mSearchHandle = searchHandle; - mResolveState = ResolveState::QuerySrv; - mResult = nullptr; - mService = nullptr; - mAddresses = nullptr; - mAddressCount = 0; } ~ResolveContext() { - if (mService) - { - if (mService->mTextEntries) - { - chip::Platform::MemoryFree(mService->mTextEntries); - } - chip::Platform::MemoryFree(mService); - } if (mAddresses) { chip::Platform::MemoryFree(mAddresses); } - if (mResult) + if (mSrvQueryResult) + { + mdns_query_results_free(mSrvQueryResult); + } + if (mAddrQueryResult && mAddrQueryResult != mSrvQueryResult) + { + mdns_query_results_free(mAddrQueryResult); + } + if (mTxtQueryResult) + { + mdns_query_results_free(mTxtQueryResult); + } + if (mSrvQueryHandle) { - mdns_query_results_free(mResult); + mdns_query_async_delete(mSrvQueryHandle); } - if (mSearchHandle) + if (mTxtQueryHandle) { - mdns_query_async_delete(mSearchHandle); + mdns_query_async_delete(mTxtQueryHandle); } } };