/* -*- Mode: C; tab-width: 4 -*- * * Copyright (c) 2009 Apple Computer, Inc. All rights reserved. * * 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. */ #pragma warning(disable:4995) #include "stdafx.h" #include <strsafe.h> #include "DNSSDService.h" #include "DNSSDEventManager.h" #include "DNSSDRecord.h" #include "TXTRecord.h" #include "StringServices.h" #include <DebugServices.h> #define WM_SOCKET (WM_APP + 100) // CDNSSDService BOOL CDNSSDService::m_registeredWindowClass = FALSE; HWND CDNSSDService::m_hiddenWindow = NULL; CDNSSDService::SocketMap CDNSSDService::m_socketMap; HRESULT CDNSSDService::FinalConstruct() { DNSServiceErrorType err = 0; HRESULT hr = S_OK; m_isPrimary = TRUE; err = DNSServiceCreateConnection( &m_primary ); require_action( !err, exit, hr = E_FAIL ); if ( !m_hiddenWindow ) { TCHAR windowClassName[ 256 ]; StringCchPrintf( windowClassName, sizeof( windowClassName ) / sizeof( TCHAR ), TEXT( "Bonjour Hidden Window %d" ), GetProcessId( NULL ) ); if ( !m_registeredWindowClass ) { WNDCLASS wc; ATOM atom; wc.style = 0; wc.lpfnWndProc = WndProc; wc.cbClsExtra = 0; wc.cbWndExtra = 0; wc.hInstance = NULL; wc.hIcon = NULL; wc.hCursor = NULL; wc.hbrBackground = NULL; wc.lpszMenuName = NULL; wc.lpszClassName = windowClassName; atom = RegisterClass(&wc); require_action( atom != NULL, exit, hr = E_FAIL ); m_registeredWindowClass = TRUE; } m_hiddenWindow = CreateWindow( windowClassName, windowClassName, WS_OVERLAPPED, 0, 0, 0, 0, NULL, NULL, GetModuleHandle( NULL ), NULL ); require_action( m_hiddenWindow != NULL, exit, hr = E_FAIL ); } err = WSAAsyncSelect( DNSServiceRefSockFD( m_primary ), m_hiddenWindow, WM_SOCKET, FD_READ ); require_action( !err, exit, hr = E_FAIL ); m_socketMap[ DNSServiceRefSockFD( m_primary ) ] = this; exit: return hr; } void CDNSSDService::FinalRelease() { dlog( kDebugLevelTrace, "FinalRelease()\n" ); Stop(); } STDMETHODIMP CDNSSDService::EnumerateDomains(DNSSDFlags flags, ULONG ifIndex, IDNSSDEventManager *eventManager, IDNSSDService **service) { CComObject<CDNSSDService> * object = NULL; DNSServiceRef subord = NULL; DNSServiceErrorType err = 0; HRESULT hr = 0; check( m_primary ); // Initialize *service = NULL; try { object = new CComObject<CDNSSDService>(); } catch ( ... ) { object = NULL; } require_action( object != NULL, exit, err = kDNSServiceErr_NoMemory ); object->AddRef(); subord = m_primary; err = DNSServiceEnumerateDomains( &subord, flags | kDNSServiceFlagsShareConnection, ifIndex, ( DNSServiceDomainEnumReply ) &DomainEnumReply, object ); require_noerr( err, exit ); object->SetPrimaryRef( m_primary ); object->SetSubordRef( subord ); object->SetEventManager( eventManager ); *service = object; exit: if ( err && object ) { object->Release(); } return err; } STDMETHODIMP CDNSSDService::Browse(DNSSDFlags flags, ULONG ifIndex, BSTR regtype, BSTR domain, IDNSSDEventManager* eventManager, IDNSSDService** service ) { CComObject<CDNSSDService> * object = NULL; std::string regtypeUTF8; std::string domainUTF8; DNSServiceRef subord = NULL; DNSServiceErrorType err = 0; HRESULT hr = 0; BOOL ok; check( m_primary ); // Initialize *service = NULL; // Convert BSTR params to utf8 ok = BSTRToUTF8( regtype, regtypeUTF8 ); require_action( ok, exit, err = kDNSServiceErr_BadParam ); ok = BSTRToUTF8( domain, domainUTF8 ); require_action( ok, exit, err = kDNSServiceErr_BadParam ); try { object = new CComObject<CDNSSDService>(); } catch ( ... ) { object = NULL; } require_action( object != NULL, exit, err = kDNSServiceErr_NoMemory ); object->AddRef(); subord = m_primary; err = DNSServiceBrowse( &subord, flags | kDNSServiceFlagsShareConnection, ifIndex, regtypeUTF8.c_str(), ( domainUTF8.size() > 0 ) ? domainUTF8.c_str() : NULL, ( DNSServiceBrowseReply ) &BrowseReply, object ); require_noerr( err, exit ); object->SetPrimaryRef( m_primary ); object->SetSubordRef( subord ); object->SetEventManager( eventManager ); *service = object; exit: if ( err && object ) { object->Release(); } return err; } STDMETHODIMP CDNSSDService::Resolve(DNSSDFlags flags, ULONG ifIndex, BSTR serviceName, BSTR regType, BSTR domain, IDNSSDEventManager* eventManager, IDNSSDService** service) { CComObject<CDNSSDService> * object = NULL; std::string serviceNameUTF8; std::string regTypeUTF8; std::string domainUTF8; DNSServiceRef subord = NULL; DNSServiceErrorType err = 0; HRESULT hr = 0; BOOL ok; check( m_primary ); // Initialize *service = NULL; // Convert BSTR params to utf8 ok = BSTRToUTF8( serviceName, serviceNameUTF8 ); require_action( ok, exit, err = kDNSServiceErr_BadParam ); ok = BSTRToUTF8( regType, regTypeUTF8 ); require_action( ok, exit, err = kDNSServiceErr_BadParam ); ok = BSTRToUTF8( domain, domainUTF8 ); require_action( ok, exit, err = kDNSServiceErr_BadParam ); try { object = new CComObject<CDNSSDService>(); } catch ( ... ) { object = NULL; } require_action( object != NULL, exit, err = kDNSServiceErr_NoMemory ); object->AddRef(); subord = m_primary; err = DNSServiceResolve( &subord, flags | kDNSServiceFlagsShareConnection, ifIndex, serviceNameUTF8.c_str(), regTypeUTF8.c_str(), domainUTF8.c_str(), ( DNSServiceResolveReply ) &ResolveReply, object ); require_noerr( err, exit ); object->SetPrimaryRef( m_primary ); object->SetSubordRef( subord ); object->SetEventManager( eventManager ); *service = object; exit: if ( err && object ) { object->Release(); } return err; } STDMETHODIMP CDNSSDService::Register(DNSSDFlags flags, ULONG ifIndex, BSTR serviceName, BSTR regType, BSTR domain, BSTR host, USHORT port, ITXTRecord *record, IDNSSDEventManager *eventManager, IDNSSDService **service) { CComObject<CDNSSDService> * object = NULL; std::string serviceNameUTF8; std::string regTypeUTF8; std::string domainUTF8; std::string hostUTF8; const void * txtRecord = NULL; uint16_t txtLen = 0; DNSServiceRef subord = NULL; DNSServiceErrorType err = 0; HRESULT hr = 0; BOOL ok; check( m_primary ); // Initialize *service = NULL; // Convert BSTR params to utf8 ok = BSTRToUTF8( serviceName, serviceNameUTF8 ); require_action( ok, exit, err = kDNSServiceErr_BadParam ); ok = BSTRToUTF8( regType, regTypeUTF8 ); require_action( ok, exit, err = kDNSServiceErr_BadParam ); ok = BSTRToUTF8( domain, domainUTF8 ); require_action( ok, exit, err = kDNSServiceErr_BadParam ); ok = BSTRToUTF8( host, hostUTF8 ); require_action( ok, exit, err = kDNSServiceErr_BadParam ); try { object = new CComObject<CDNSSDService>(); } catch ( ... ) { object = NULL; } require_action( object != NULL, exit, err = kDNSServiceErr_NoMemory ); object->AddRef(); if ( record ) { CComObject< CTXTRecord > * realTXTRecord; realTXTRecord = ( CComObject< CTXTRecord >* ) record; txtRecord = realTXTRecord->GetBytes(); txtLen = realTXTRecord->GetLen(); } subord = m_primary; err = DNSServiceRegister( &subord, flags | kDNSServiceFlagsShareConnection, ifIndex, serviceNameUTF8.c_str(), regTypeUTF8.c_str(), ( domainUTF8.size() > 0 ) ? domainUTF8.c_str() : NULL, hostUTF8.c_str(), htons( port ), txtLen, txtRecord, ( DNSServiceRegisterReply ) &RegisterReply, object ); require_noerr( err, exit ); object->SetPrimaryRef( m_primary ); object->SetSubordRef( subord ); object->SetEventManager( eventManager ); *service = object; exit: if ( err && object ) { object->Release(); } return err; } STDMETHODIMP CDNSSDService::QueryRecord(DNSSDFlags flags, ULONG ifIndex, BSTR fullname, DNSSDRRType rrtype, DNSSDRRClass rrclass, IDNSSDEventManager *eventManager, IDNSSDService **service) { CComObject<CDNSSDService> * object = NULL; DNSServiceRef subord = NULL; std::string fullNameUTF8; DNSServiceErrorType err = 0; HRESULT hr = 0; BOOL ok; check( m_primary ); // Initialize *service = NULL; // Convert BSTR params to utf8 ok = BSTRToUTF8( fullname, fullNameUTF8 ); require_action( ok, exit, err = kDNSServiceErr_BadParam ); try { object = new CComObject<CDNSSDService>(); } catch ( ... ) { object = NULL; } require_action( object != NULL, exit, err = kDNSServiceErr_NoMemory ); object->AddRef(); subord = m_primary; err = DNSServiceQueryRecord( &subord, flags | kDNSServiceFlagsShareConnection, ifIndex, fullNameUTF8.c_str(), ( uint16_t ) rrtype, ( uint16_t ) rrclass, ( DNSServiceQueryRecordReply ) &QueryRecordReply, object ); require_noerr( err, exit ); object->SetPrimaryRef( m_primary ); object->SetSubordRef( subord ); object->SetEventManager( eventManager ); *service = object; exit: if ( err && object ) { object->Release(); } return err; } STDMETHODIMP CDNSSDService::RegisterRecord(DNSSDFlags flags, ULONG ifIndex, BSTR fullName, DNSSDRRType rrtype, DNSSDRRClass rrclass, VARIANT rdata, ULONG ttl, IDNSSDEventManager* eventManager, IDNSSDRecord** record) { CComObject<CDNSSDRecord> * object = NULL; DNSRecordRef rref = NULL; std::string fullNameUTF8; std::vector< BYTE > byteArray; const void * byteArrayPtr = NULL; DNSServiceErrorType err = 0; HRESULT hr = 0; BOOL ok; check( m_primary ); // Initialize *object = NULL; // Convert BSTR params to utf8 ok = BSTRToUTF8( fullName, fullNameUTF8 ); require_action( ok, exit, err = kDNSServiceErr_BadParam ); // Convert the VARIANT ok = VariantToByteArray( &rdata, byteArray ); require_action( ok, exit, err = kDNSServiceErr_Unknown ); try { object = new CComObject<CDNSSDRecord>(); } catch ( ... ) { object = NULL; } require_action( object != NULL, exit, err = kDNSServiceErr_NoMemory ); object->AddRef(); err = DNSServiceRegisterRecord( m_primary, &rref, flags, ifIndex, fullNameUTF8.c_str(), rrtype, rrclass, ( uint16_t ) byteArray.size(), byteArray.size() > 0 ? &byteArray[ 0 ] : NULL, ttl, &RegisterRecordReply, object ); require_noerr( err, exit ); object->SetServiceObject( this ); object->SetRecordRef( rref ); this->SetEventManager( eventManager ); *record = object; exit: if ( err && object ) { object->Release(); } return err; } STDMETHODIMP CDNSSDService::AddRecord(DNSSDFlags flags, DNSSDRRType rrtype, VARIANT rdata, ULONG ttl, IDNSSDRecord ** record) { CComObject<CDNSSDRecord> * object = NULL; DNSRecordRef rref = NULL; std::vector< BYTE > byteArray; const void * byteArrayPtr = NULL; DNSServiceErrorType err = 0; HRESULT hr = 0; BOOL ok; check( m_primary ); // Initialize *object = NULL; // Convert the VARIANT ok = VariantToByteArray( &rdata, byteArray ); require_action( ok, exit, err = kDNSServiceErr_Unknown ); try { object = new CComObject<CDNSSDRecord>(); } catch ( ... ) { object = NULL; } require_action( object != NULL, exit, err = kDNSServiceErr_NoMemory ); object->AddRef(); err = DNSServiceAddRecord( m_primary, &rref, flags, rrtype, ( uint16_t ) byteArray.size(), byteArray.size() > 0 ? &byteArray[ 0 ] : NULL, ttl ); require_noerr( err, exit ); object->SetServiceObject( this ); object->SetRecordRef( rref ); *record = object; exit: if ( err && object ) { object->Release(); } return err; } STDMETHODIMP CDNSSDService::ReconfirmRecord(DNSSDFlags flags, ULONG ifIndex, BSTR fullName, DNSSDRRType rrtype, DNSSDRRClass rrclass, VARIANT rdata) { std::string fullNameUTF8; std::vector< BYTE > byteArray; const void * byteArrayPtr = NULL; DNSServiceErrorType err = 0; HRESULT hr = 0; BOOL ok; // Convert BSTR params to utf8 ok = BSTRToUTF8( fullName, fullNameUTF8 ); require_action( ok, exit, err = kDNSServiceErr_BadParam ); // Convert the VARIANT ok = VariantToByteArray( &rdata, byteArray ); require_action( ok, exit, err = kDNSServiceErr_Unknown ); err = DNSServiceReconfirmRecord( flags, ifIndex, fullNameUTF8.c_str(), rrtype, rrclass, ( uint16_t ) byteArray.size(), byteArray.size() > 0 ? &byteArray[ 0 ] : NULL ); require_noerr( err, exit ); exit: return err; } STDMETHODIMP CDNSSDService::GetProperty(BSTR prop, VARIANT * value ) { std::string propUTF8; std::vector< BYTE > byteArray; SAFEARRAY * psa = NULL; BYTE * pData = NULL; uint32_t elems = 0; DNSServiceErrorType err = 0; BOOL ok = TRUE; // Convert BSTR params to utf8 ok = BSTRToUTF8( prop, propUTF8 ); require_action( ok, exit, err = kDNSServiceErr_BadParam ); // Setup the byte array require_action( V_VT( value ) == ( VT_ARRAY|VT_UI1 ), exit, err = kDNSServiceErr_Unknown ); psa = V_ARRAY( value ); require_action( psa, exit, err = kDNSServiceErr_Unknown ); require_action( SafeArrayGetDim( psa ) == 1, exit, err = kDNSServiceErr_Unknown ); byteArray.reserve( psa->rgsabound[0].cElements ); byteArray.assign( byteArray.capacity(), 0 ); elems = ( uint32_t ) byteArray.capacity(); // Call the function and package the return value in the Variant err = DNSServiceGetProperty( propUTF8.c_str(), &byteArray[ 0 ], &elems ); require_noerr( err, exit ); ok = ByteArrayToVariant( &byteArray[ 0 ], elems, value ); require_action( ok, exit, err = kDNSSDError_Unknown ); exit: if ( psa ) { SafeArrayUnaccessData( psa ); psa = NULL; } return err; } STDMETHODIMP CDNSSDService::GetAddrInfo(DNSSDFlags flags, ULONG ifIndex, DNSSDAddressFamily addressFamily, BSTR hostName, IDNSSDEventManager *eventManager, IDNSSDService **service) { CComObject<CDNSSDService> * object = NULL; DNSServiceRef subord = NULL; std::string hostNameUTF8; DNSServiceErrorType err = 0; HRESULT hr = 0; BOOL ok; check( m_primary ); // Initialize *service = NULL; // Convert BSTR params to utf8 ok = BSTRToUTF8( hostName, hostNameUTF8 ); require_action( ok, exit, err = kDNSServiceErr_BadParam ); try { object = new CComObject<CDNSSDService>(); } catch ( ... ) { object = NULL; } require_action( object != NULL, exit, err = kDNSServiceErr_NoMemory ); object->AddRef(); subord = m_primary; err = DNSServiceGetAddrInfo( &subord, flags | kDNSServiceFlagsShareConnection, ifIndex, addressFamily, hostNameUTF8.c_str(), ( DNSServiceGetAddrInfoReply ) &GetAddrInfoReply, object ); require_noerr( err, exit ); object->SetPrimaryRef( m_primary ); object->SetSubordRef( subord ); object->SetEventManager( eventManager ); *service = object; exit: if ( err && object ) { object->Release(); } return err; } STDMETHODIMP CDNSSDService::NATPortMappingCreate(DNSSDFlags flags, ULONG ifIndex, DNSSDAddressFamily addressFamily, DNSSDProtocol protocol, USHORT internalPort, USHORT externalPort, ULONG ttl, IDNSSDEventManager *eventManager, IDNSSDService **service) { CComObject<CDNSSDService> * object = NULL; DNSServiceRef subord = NULL; DNSServiceProtocol prot = 0; DNSServiceErrorType err = 0; HRESULT hr = 0; check( m_primary ); // Initialize *service = NULL; try { object = new CComObject<CDNSSDService>(); } catch ( ... ) { object = NULL; } require_action( object != NULL, exit, err = kDNSServiceErr_NoMemory ); object->AddRef(); prot = ( addressFamily | protocol ); subord = m_primary; err = DNSServiceNATPortMappingCreate( &subord, flags | kDNSServiceFlagsShareConnection, ifIndex, prot, htons( internalPort ), htons( externalPort ), ttl, ( DNSServiceNATPortMappingReply ) &NATPortMappingReply, object ); require_noerr( err, exit ); object->SetPrimaryRef( m_primary ); object->SetSubordRef( subord ); object->SetEventManager( eventManager ); *service = object; exit: if ( err && object ) { object->Release(); } return err; } STDMETHODIMP CDNSSDService::Stop(void) { if ( !m_stopped ) { m_stopped = TRUE; dlog( kDebugLevelTrace, "Stop()\n" ); if ( m_isPrimary && m_primary ) { SocketMap::iterator it; if ( m_hiddenWindow ) { WSAAsyncSelect( DNSServiceRefSockFD( m_primary ), m_hiddenWindow, 0, 0 ); } it = m_socketMap.find( DNSServiceRefSockFD( m_primary ) ); if ( it != m_socketMap.end() ) { m_socketMap.erase( it ); } DNSServiceRefDeallocate( m_primary ); m_primary = NULL; } else if ( m_subord ) { DNSServiceRefDeallocate( m_subord ); m_subord = NULL; } if ( m_eventManager != NULL ) { m_eventManager->Release(); m_eventManager = NULL; } } return S_OK; } void DNSSD_API CDNSSDService::DomainEnumReply ( DNSServiceRef sdRef, DNSServiceFlags flags, uint32_t ifIndex, DNSServiceErrorType errorCode, const char *replyDomainUTF8, void *context ) { CComObject<CDNSSDService> * service = NULL; CDNSSDEventManager * eventManager = NULL; int err = 0; service = ( CComObject< CDNSSDService>* ) context; require_action( service, exit, err = kDNSServiceErr_Unknown ); if ( service->ShouldHandleReply( errorCode, eventManager ) ) { CComBSTR replyDomain; BOOL ok; ok = UTF8ToBSTR( replyDomainUTF8, replyDomain ); require_action( ok, exit, err = kDNSServiceErr_Unknown ); if ( flags & kDNSServiceFlagsAdd ) { eventManager->Fire_DomainFound( service, ( DNSSDFlags ) flags, ifIndex, replyDomain ); } else { eventManager->Fire_DomainLost( service, ( DNSSDFlags ) flags, ifIndex, replyDomain ); } } exit: return; } void DNSSD_API CDNSSDService::BrowseReply ( DNSServiceRef sdRef, DNSServiceFlags flags, uint32_t ifIndex, DNSServiceErrorType errorCode, const char *serviceNameUTF8, const char *regTypeUTF8, const char *replyDomainUTF8, void *context ) { CComObject<CDNSSDService> * service = NULL; CDNSSDEventManager * eventManager = NULL; int err = 0; service = ( CComObject< CDNSSDService>* ) context; require_action( service, exit, err = kDNSServiceErr_Unknown ); if ( service->ShouldHandleReply( errorCode, eventManager ) ) { CComBSTR serviceName; CComBSTR regType; CComBSTR replyDomain; UTF8ToBSTR( serviceNameUTF8, serviceName ); UTF8ToBSTR( regTypeUTF8, regType ); UTF8ToBSTR( replyDomainUTF8, replyDomain ); if ( flags & kDNSServiceFlagsAdd ) { eventManager->Fire_ServiceFound( service, ( DNSSDFlags ) flags, ifIndex, serviceName, regType, replyDomain ); } else { eventManager->Fire_ServiceLost( service, ( DNSSDFlags ) flags, ifIndex, serviceName, regType, replyDomain ); } } exit: return; } void DNSSD_API CDNSSDService::ResolveReply ( DNSServiceRef sdRef, DNSServiceFlags flags, uint32_t ifIndex, DNSServiceErrorType errorCode, const char *fullNameUTF8, const char *hostNameUTF8, uint16_t port, uint16_t txtLen, const unsigned char *txtRecord, void *context ) { CComObject<CDNSSDService> * service = NULL; CDNSSDEventManager * eventManager = NULL; int err = 0; service = ( CComObject< CDNSSDService>* ) context; require_action( service, exit, err = kDNSServiceErr_Unknown ); if ( service->ShouldHandleReply( errorCode, eventManager ) ) { CComBSTR fullName; CComBSTR hostName; CComBSTR regType; CComBSTR replyDomain; CComObject< CTXTRecord >* record; BOOL ok; ok = UTF8ToBSTR( fullNameUTF8, fullName ); require_action( ok, exit, err = kDNSServiceErr_Unknown ); ok = UTF8ToBSTR( hostNameUTF8, hostName ); require_action( ok, exit, err = kDNSServiceErr_Unknown ); try { record = new CComObject<CTXTRecord>(); } catch ( ... ) { record = NULL; } require_action( record, exit, err = kDNSServiceErr_NoMemory ); record->AddRef(); if ( txtLen > 0 ) { record->SetBytes( txtRecord, txtLen ); } eventManager->Fire_ServiceResolved( service, ( DNSSDFlags ) flags, ifIndex, fullName, hostName, ntohs( port ), record ); } exit: return; } void DNSSD_API CDNSSDService::RegisterReply ( DNSServiceRef sdRef, DNSServiceFlags flags, DNSServiceErrorType errorCode, const char *serviceNameUTF8, const char *regTypeUTF8, const char *domainUTF8, void *context ) { CComObject<CDNSSDService> * service = NULL; CDNSSDEventManager * eventManager = NULL; int err = 0; service = ( CComObject< CDNSSDService>* ) context; require_action( service, exit, err = kDNSServiceErr_Unknown ); if ( service->ShouldHandleReply( errorCode, eventManager ) ) { CComBSTR serviceName; CComBSTR regType; CComBSTR domain; BOOL ok; ok = UTF8ToBSTR( serviceNameUTF8, serviceName ); require_action( ok, exit, err = kDNSServiceErr_Unknown ); ok = UTF8ToBSTR( regTypeUTF8, regType ); require_action( ok, exit, err = kDNSServiceErr_Unknown ); ok = UTF8ToBSTR( domainUTF8, domain ); require_action( ok, exit, err = kDNSServiceErr_Unknown ); eventManager->Fire_ServiceRegistered( service, ( DNSSDFlags ) flags, serviceName, regType, domain ); } exit: return; } void DNSSD_API CDNSSDService::QueryRecordReply ( DNSServiceRef sdRef, DNSServiceFlags flags, uint32_t ifIndex, DNSServiceErrorType errorCode, const char *fullNameUTF8, uint16_t rrtype, uint16_t rrclass, uint16_t rdlen, const void *rdata, uint32_t ttl, void *context ) { CComObject<CDNSSDService> * service = NULL; CDNSSDEventManager * eventManager = NULL; int err = 0; service = ( CComObject< CDNSSDService>* ) context; require_action( service, exit, err = kDNSServiceErr_Unknown ); if ( service->ShouldHandleReply( errorCode, eventManager ) ) { CComBSTR fullName; VARIANT var; BOOL ok; ok = UTF8ToBSTR( fullNameUTF8, fullName ); require_action( ok, exit, err = kDNSServiceErr_Unknown ); ok = ByteArrayToVariant( rdata, rdlen, &var ); require_action( ok, exit, err = kDNSServiceErr_Unknown ); eventManager->Fire_QueryRecordAnswered( service, ( DNSSDFlags ) flags, ifIndex, fullName, ( DNSSDRRType ) rrtype, ( DNSSDRRClass ) rrclass, var, ttl ); } exit: return; } void DNSSD_API CDNSSDService::GetAddrInfoReply ( DNSServiceRef sdRef, DNSServiceFlags flags, uint32_t ifIndex, DNSServiceErrorType errorCode, const char *hostNameUTF8, const struct sockaddr *rawAddress, uint32_t ttl, void *context ) { CComObject<CDNSSDService> * service = NULL; CDNSSDEventManager * eventManager = NULL; int err = 0; service = ( CComObject< CDNSSDService>* ) context; require_action( service, exit, err = kDNSServiceErr_Unknown ); if ( service->ShouldHandleReply( errorCode, eventManager ) ) { CComBSTR hostName; DWORD sockaddrLen; DNSSDAddressFamily addressFamily; char addressUTF8[INET6_ADDRSTRLEN]; DWORD addressLen = sizeof( addressUTF8 ); CComBSTR address; BOOL ok; ok = UTF8ToBSTR( hostNameUTF8, hostName ); require_action( ok, exit, err = kDNSServiceErr_Unknown ); switch ( rawAddress->sa_family ) { case AF_INET: { addressFamily = kDNSSDAddressFamily_IPv4; sockaddrLen = sizeof( sockaddr_in ); } break; case AF_INET6: { addressFamily = kDNSSDAddressFamily_IPv6; sockaddrLen = sizeof( sockaddr_in6 ); } break; } err = WSAAddressToStringA( ( LPSOCKADDR ) rawAddress, sockaddrLen, NULL, addressUTF8, &addressLen ); require_noerr( err, exit ); ok = UTF8ToBSTR( addressUTF8, address ); require_action( ok, exit, err = kDNSServiceErr_Unknown ); eventManager->Fire_AddressFound( service, ( DNSSDFlags ) flags, ifIndex, hostName, addressFamily, address, ttl ); } exit: return; } void DNSSD_API CDNSSDService::NATPortMappingReply ( DNSServiceRef sdRef, DNSServiceFlags flags, uint32_t ifIndex, DNSServiceErrorType errorCode, uint32_t externalAddress, /* four byte IPv4 address in network byte order */ DNSServiceProtocol protocol, uint16_t internalPort, uint16_t externalPort, /* may be different than the requested port */ uint32_t ttl, /* may be different than the requested ttl */ void *context ) { CComObject<CDNSSDService> * service = NULL; CDNSSDEventManager * eventManager = NULL; int err = 0; service = ( CComObject< CDNSSDService>* ) context; require_action( service, exit, err = kDNSServiceErr_Unknown ); if ( service->ShouldHandleReply( errorCode, eventManager ) ) { eventManager->Fire_MappingCreated( service, ( DNSSDFlags ) flags, ifIndex, externalAddress, ( DNSSDAddressFamily ) ( protocol & 0x8 ), ( DNSSDProtocol ) ( protocol & 0x80 ), ntohs( internalPort ), ntohs( externalPort ), ttl ); } exit: return; } void DNSSD_API CDNSSDService::RegisterRecordReply ( DNSServiceRef sdRef, DNSRecordRef RecordRef, DNSServiceFlags flags, DNSServiceErrorType errorCode, void *context ) { CComObject<CDNSSDRecord> * record = NULL; CDNSSDService * service = NULL; CDNSSDEventManager * eventManager = NULL; int err = 0; record = ( CComObject< CDNSSDRecord >* ) context; require_action( record, exit, err = kDNSServiceErr_Unknown ); service = record->GetServiceObject(); require_action( service, exit, err = kDNSServiceErr_Unknown ); if ( service->ShouldHandleReply( errorCode, eventManager ) ) { eventManager->Fire_RecordRegistered( record, ( DNSSDFlags ) flags ); } exit: return; } BOOL CDNSSDService::ShouldHandleReply( DNSServiceErrorType errorCode, CDNSSDEventManager *& eventManager ) { BOOL ok = FALSE; if ( !this->Stopped() ) { eventManager = this->GetEventManager(); require_action( eventManager, exit, ok = FALSE ); if ( !errorCode ) { ok = TRUE; } else { eventManager->Fire_OperationFailed( this, ( DNSSDError ) errorCode ); } } exit: return ok; } LRESULT CALLBACK CDNSSDService::WndProc( HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam ) { if ( msg == WM_SOCKET ) { SocketMap::iterator it; it = m_socketMap.find( ( SOCKET ) wParam ); check( it != m_socketMap.end() ); if ( it != m_socketMap.end() ) { DNSServiceProcessResult( it->second->m_primary ); } } return DefWindowProc(hWnd, msg, wParam, lParam);; }