1 /* -*- Mode: C; tab-width: 4 -*-
3 * Copyright (c) 2003-2004 Apple Computer, Inc. All rights reserved.
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
9 * http://www.apache.org/licenses/LICENSE-2.0
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
17 Change History (most recent first):
19 $Log: dnssd_NET.cpp,v $
20 Revision 1.11 2009/03/30 20:19:05 herscher
21 <rdar://problem/5925472> Current Bonjour code does not compile on Windows
22 <rdar://problem/5187308> Move build train to Visual Studio 2005
24 Revision 1.10 2006/08/14 23:25:43 cheshire
25 Re-licensed mDNSResponder daemon source code under Apache License, Version 2.0
27 Revision 1.9 2004/09/16 18:17:13 shersche
28 Use background threads, cleanup to parameter names.
29 Submitted by: prepin@gmail.com
31 Revision 1.8 2004/09/13 19:35:58 shersche
32 <rdar://problem/3798941> Add Apple.DNSSD namespace to MC++ wrapper class
33 <rdar://problem/3798950> Change all instances of unsigned short to int
34 Bug #: 3798941, 3798950
36 Revision 1.7 2004/09/11 00:36:40 shersche
37 <rdar://problem/3786226> Modified .NET shim code to use host byte order for ports in APIs and callbacks
40 Revision 1.6 2004/09/02 21:20:56 cheshire
41 <rdar://problem/3774871> DLL.NET crashes on null record
43 Revision 1.5 2004/07/27 07:12:56 shersche
44 make TextRecord an instantiable class object
46 Revision 1.4 2004/07/26 06:19:05 shersche
47 Treat byte arrays of zero-length as null arrays
49 Revision 1.3 2004/07/19 16:08:56 shersche
50 fix problems in UTF8/Unicode string translations
52 Revision 1.2 2004/07/19 07:48:34 shersche
53 fix bug in DNSService.Register when passing in NULL text record, add TextRecord APIs
55 Revision 1.1 2004/06/26 04:01:22 shersche
61 // This is the main DLL file.
65 #include "dnssd_NET.h"
66 #include "DebugServices.h"
70 using namespace System::Net::Sockets
;
71 using namespace System::Diagnostics
;
72 using namespace Apple
;
73 using namespace Apple::DNSSD
;
76 //===========================================================================================================================
78 //===========================================================================================================================
80 #define DEBUG_NAME "[dnssd.NET] "
86 ConvertToString(const char * utf8String
)
88 return __gc
new String(utf8String
, 0, strlen(utf8String
), __gc
new UTF8Encoding(true, true));
95 // ServiceRef serves as the base class for all DNSService operations.
97 // It manages the DNSServiceRef, and implements processing the
100 ServiceRef::ServiceRef(Object
* callback
)
103 m_callback(callback
),
106 m_impl
= new ServiceRefImpl(this);
110 ServiceRef::~ServiceRef()
118 // Starts the main processing thread
121 ServiceRef::StartThread()
123 check( m_impl
!= NULL
);
125 m_impl
->SetupEvents();
127 m_thread
= new Thread(new ThreadStart(this, &Apple::DNSSD::ServiceRef::ProcessingThread
));
128 m_thread
->Name
= S
"DNSService Thread";
129 m_thread
->IsBackground
= true;
138 // The Thread class can only invoke methods in MC++ types. So we
139 // make a ProcessingThread method that forwards to the impl
142 ServiceRef::ProcessingThread()
144 m_impl
->ProcessingThread();
151 // Calls impl-Dispose(). This ultimately will call DNSServiceRefDeallocate()
154 ServiceRef::Dispose()
156 check(m_impl
!= NULL
);
157 check(m_bDisposed
== false);
164 // Call Dispose. This won't call DNSServiceRefDeallocate()
165 // necessarily. It depends on what thread this is being
173 GC::SuppressFinalize(this);
179 // EnumerateDomainsDispatch
181 // Dispatch a reply to the delegate.
184 ServiceRef::EnumerateDomainsDispatch
192 if ((m_callback
!= NULL
) && (m_impl
!= NULL
))
194 DNSService::EnumerateDomainsReply
* OnEnumerateDomainsReply
= static_cast<DNSService::EnumerateDomainsReply
*>(m_callback
);
195 OnEnumerateDomainsReply(this, flags
, interfaceIndex
, errorCode
, replyDomain
);
203 // Dispatch a reply to the delegate.
206 ServiceRef::RegisterDispatch
215 if ((m_callback
!= NULL
) && (m_impl
!= NULL
))
217 DNSService::RegisterReply
* OnRegisterReply
= static_cast<DNSService::RegisterReply
*>(m_callback
);
218 OnRegisterReply(this, flags
, errorCode
, name
, regtype
, domain
);
226 // Dispatch a reply to the delegate.
229 ServiceRef::BrowseDispatch
234 String
* serviceName
,
239 if ((m_callback
!= NULL
) && (m_impl
!= NULL
))
241 DNSService::BrowseReply
* OnBrowseReply
= static_cast<DNSService::BrowseReply
*>(m_callback
);
242 OnBrowseReply(this, flags
, interfaceIndex
, errorCode
, serviceName
, regtype
, replyDomain
);
250 // Dispatch a reply to the delegate.
253 ServiceRef::ResolveDispatch
264 if ((m_callback
!= NULL
) && (m_impl
!= NULL
))
266 DNSService::ResolveReply
* OnResolveReply
= static_cast<DNSService::ResolveReply
*>(m_callback
);
267 OnResolveReply(this, flags
, interfaceIndex
, errorCode
, fullname
, hosttarget
, port
, txtRecord
);
273 // RegisterRecordDispatch
275 // Dispatch a reply to the delegate.
278 ServiceRef::RegisterRecordDispatch
285 if ((m_callback
!= NULL
) && (m_impl
!= NULL
))
287 DNSService::RegisterRecordReply
* OnRegisterRecordReply
= static_cast<DNSService::RegisterRecordReply
*>(m_callback
);
288 OnRegisterRecordReply(this, flags
, errorCode
, record
);
294 // QueryRecordDispatch
296 // Dispatch a reply to the delegate.
299 ServiceRef::QueryRecordDispatch
311 if ((m_callback
!= NULL
) && (m_impl
!= NULL
))
313 DNSService::QueryRecordReply
* OnQueryRecordReply
= static_cast<DNSService::QueryRecordReply
*>(m_callback
);
314 OnQueryRecordReply(this, flags
, interfaceIndex
, errorCode
, fullname
, rrtype
, rrclass
, rdata
, ttl
);
320 // ServiceRefImpl::ServiceRefImpl()
322 // Constructs a new ServiceRefImpl. We save the pointer to our enclosing
323 // class in a gcroot handle. This satisfies the garbage collector as
324 // the outer class is a managed type
326 ServiceRef::ServiceRefImpl::ServiceRefImpl(ServiceRef
* outer
)
334 m_threadId
= GetCurrentThreadId();
339 // ServiceRefImpl::~ServiceRefImpl()
341 // Deallocate all resources associated with the ServiceRefImpl
343 ServiceRef::ServiceRefImpl::~ServiceRefImpl()
345 if (m_socketEvent
!= NULL
)
347 CloseHandle(m_socketEvent
);
348 m_socketEvent
= NULL
;
351 if (m_stopEvent
!= NULL
)
353 CloseHandle(m_stopEvent
);
359 DNSServiceRefDeallocate(m_ref
);
366 // ServiceRefImpl::SetupEvents()
368 // Setup the events necessary to manage multi-threaded dispatch
369 // of DNSService Events
372 ServiceRef::ServiceRefImpl::SetupEvents()
374 check(m_ref
!= NULL
);
376 m_socket
= (SOCKET
) DNSServiceRefSockFD(m_ref
);
377 check(m_socket
!= INVALID_SOCKET
);
379 m_socketEvent
= CreateEvent(NULL
, 0, 0, NULL
);
381 if (m_socketEvent
== NULL
)
383 throw new DNSServiceException(Unknown
);
386 int err
= WSAEventSelect(m_socket
, m_socketEvent
, FD_READ
|FD_CLOSE
);
390 throw new DNSServiceException(Unknown
);
393 m_stopEvent
= CreateEvent(NULL
, 0, 0, NULL
);
395 if (m_stopEvent
== NULL
)
397 throw new DNSServiceException(Unknown
);
403 // ServiceRefImpl::ProcessingThread()
405 // Wait for socket events on the DNSServiceRefSockFD(). Also wait
409 ServiceRef::ServiceRefImpl::ProcessingThread()
411 check( m_socketEvent
!= NULL
);
412 check( m_stopEvent
!= NULL
);
413 check( m_ref
!= NULL
);
417 handles
[0] = m_socketEvent
;
418 handles
[1] = m_stopEvent
;
420 while (m_disposed
== false)
422 int ret
= WaitForMultipleObjects(2, handles
, FALSE
, INFINITE
);
425 // it's a socket event
427 if (ret
== WAIT_OBJECT_0
)
429 DNSServiceProcessResult(m_ref
);
432 // else it's a stop event
434 else if (ret
== WAIT_OBJECT_0
+ 1)
441 // unexpected wait result
443 dlog( kDebugLevelWarning
, DEBUG_NAME
"%s: unexpected wait result (result=0x%08X)\n", __ROUTINE__
, ret
);
452 // ServiceRefImpl::Dispose()
454 // Calls DNSServiceRefDeallocate()
457 ServiceRef::ServiceRefImpl::Dispose()
462 check(m_disposed
== false);
466 ok
= SetEvent(m_stopEvent
);
467 err
= translate_errno( ok
, (OSStatus
) GetLastError(), kUnknownErr
);
468 require_noerr( err
, exit
);
477 // ServiceRefImpl::EnumerateDomainsCallback()
479 // This is the callback from dnssd.dll. We pass this up to our outer, managed type
482 ServiceRef::ServiceRefImpl::EnumerateDomainsCallback
485 DNSServiceFlags flags
,
486 uint32_t interfaceIndex
,
487 DNSServiceErrorType errorCode
,
488 const char * replyDomain
,
492 ServiceRef::ServiceRefImpl
* self
= static_cast<ServiceRef::ServiceRefImpl
*>(context
);
494 check( self
!= NULL
);
495 check( self
->m_outer
!= NULL
);
497 if (self
->m_disposed
== false)
499 self
->m_outer
->EnumerateDomainsDispatch((ServiceFlags
) flags
, interfaceIndex
, (ErrorCode
) errorCode
, ConvertToString(replyDomain
));
505 // ServiceRefImpl::RegisterCallback()
507 // This is the callback from dnssd.dll. We pass this up to our outer, managed type
510 ServiceRef::ServiceRefImpl::RegisterCallback
513 DNSServiceFlags flags
,
514 DNSServiceErrorType errorCode
,
516 const char * regtype
,
521 ServiceRef::ServiceRefImpl
* self
= static_cast<ServiceRef::ServiceRefImpl
*>(context
);
523 check( self
!= NULL
);
524 check( self
->m_outer
!= NULL
);
526 if (self
->m_disposed
== false)
528 self
->m_outer
->RegisterDispatch((ServiceFlags
) flags
, (ErrorCode
) errorCode
, ConvertToString(name
), ConvertToString(regtype
), ConvertToString(domain
));
534 // ServiceRefImpl::BrowseCallback()
536 // This is the callback from dnssd.dll. We pass this up to our outer, managed type
539 ServiceRef::ServiceRefImpl::BrowseCallback
542 DNSServiceFlags flags
,
543 uint32_t interfaceIndex
,
544 DNSServiceErrorType errorCode
,
545 const char * serviceName
,
546 const char * regtype
,
547 const char * replyDomain
,
551 ServiceRef::ServiceRefImpl
* self
= static_cast<ServiceRef::ServiceRefImpl
*>(context
);
553 check( self
!= NULL
);
554 check( self
->m_outer
!= NULL
);
556 if (self
->m_disposed
== false)
558 self
->m_outer
->BrowseDispatch((ServiceFlags
) flags
, interfaceIndex
, (ErrorCode
) errorCode
, ConvertToString(serviceName
), ConvertToString(regtype
), ConvertToString(replyDomain
));
564 // ServiceRefImpl::ResolveCallback()
566 // This is the callback from dnssd.dll. We pass this up to our outer, managed type
569 ServiceRef::ServiceRefImpl::ResolveCallback
572 DNSServiceFlags flags
,
573 uint32_t interfaceIndex
,
574 DNSServiceErrorType errorCode
,
575 const char * fullname
,
576 const char * hosttarget
,
577 uint16_t notAnIntPort
,
579 const char * txtRecord
,
583 ServiceRef::ServiceRefImpl
* self
= static_cast<ServiceRef::ServiceRefImpl
*>(context
);
585 check( self
!= NULL
);
586 check( self
->m_outer
!= NULL
);
588 if (self
->m_disposed
== false)
590 Byte txtRecordBytes
[];
592 txtRecordBytes
= NULL
;
597 // copy raw memory into managed byte array
599 txtRecordBytes
= new Byte
[txtLen
];
600 Byte __pin
* p
= &txtRecordBytes
[0];
601 memcpy(p
, txtRecord
, txtLen
);
604 self
->m_outer
->ResolveDispatch((ServiceFlags
) flags
, interfaceIndex
, (ErrorCode
) errorCode
, ConvertToString(fullname
), ConvertToString(hosttarget
), ntohs(notAnIntPort
), txtRecordBytes
);
610 // ServiceRefImpl::RegisterRecordCallback()
612 // This is the callback from dnssd.dll. We pass this up to our outer, managed type
615 ServiceRef::ServiceRefImpl::RegisterRecordCallback
619 DNSServiceFlags flags
,
620 DNSServiceErrorType errorCode
,
624 ServiceRef::ServiceRefImpl
* self
= static_cast<ServiceRef::ServiceRefImpl
*>(context
);
626 check( self
!= NULL
);
627 check( self
->m_outer
!= NULL
);
629 if (self
->m_disposed
== false)
631 RecordRef
* record
= NULL
;
635 record
= new RecordRef
;
637 record
->m_impl
->m_ref
= rrRef
;
640 self
->m_outer
->RegisterRecordDispatch((ServiceFlags
) flags
, (ErrorCode
) errorCode
, record
);
646 // ServiceRefImpl::QueryRecordCallback()
648 // This is the callback from dnssd.dll. We pass this up to our outer, managed type
651 ServiceRef::ServiceRefImpl::QueryRecordCallback
653 DNSServiceRef DNSServiceRef
,
654 DNSServiceFlags flags
,
655 uint32_t interfaceIndex
,
656 DNSServiceErrorType errorCode
,
657 const char * fullname
,
666 ServiceRef::ServiceRefImpl
* self
= static_cast<ServiceRef::ServiceRefImpl
*>(context
);
668 check( self
!= NULL
);
669 check( self
->m_outer
!= NULL
);
671 if (self
->m_disposed
== false)
677 rdataBytes
= new Byte
[rdlen
];
678 Byte __pin
* p
= &rdataBytes
[0];
679 memcpy(p
, rdata
, rdlen
);
682 self
->m_outer
->QueryRecordDispatch((ServiceFlags
) flags
, (int) interfaceIndex
, (ErrorCode
) errorCode
, ConvertToString(fullname
), rrtype
, rrclass
, rdataBytes
, ttl
);
690 * This maps to DNSServiceEnumerateDomains(). Returns an
691 * initialized ServiceRef on success, throws an exception
695 DNSService::EnumerateDomains
699 EnumerateDomainsReply
* callback
702 ServiceRef
* sdRef
= new ServiceRef(callback
);
705 err
= DNSServiceEnumerateDomains(&sdRef
->m_impl
->m_ref
, flags
, interfaceIndex
, ServiceRef::ServiceRefImpl::EnumerateDomainsCallback
, sdRef
->m_impl
);
709 throw new DNSServiceException(err
);
712 sdRef
->StartThread();
721 * This maps to DNSServiceRegister(). Returns an
722 * initialized ServiceRef on success, throws an exception
736 RegisterReply
* callback
739 ServiceRef
* sdRef
= new ServiceRef(callback
);
740 PString
* pName
= new PString(name
);
741 PString
* pType
= new PString(regtype
);
742 PString
* pDomain
= new PString(domain
);
743 PString
* pHost
= new PString(host
);
745 Byte __pin
* p
= NULL
;
748 if ((txtRecord
!= NULL
) && (txtRecord
->Length
> 0))
750 len
= txtRecord
->Length
;
755 int err
= DNSServiceRegister(&sdRef
->m_impl
->m_ref
, flags
, interfaceIndex
, pName
->c_str(), pType
->c_str(), pDomain
->c_str(), pHost
->c_str(), htons(port
), len
, v
, ServiceRef::ServiceRefImpl::RegisterCallback
, sdRef
->m_impl
);
759 throw new DNSServiceException(err
);
762 sdRef
->StartThread();
771 * This maps to DNSServiceAddRecord(). Returns an
772 * initialized ServiceRef on success, throws an exception
776 DNSService::AddRecord
786 Byte __pin
* p
= NULL
;
789 if ((rdata
!= NULL
) && (rdata
->Length
> 0))
796 RecordRef
* record
= new RecordRef
;
798 int err
= DNSServiceAddRecord(sdRef
->m_impl
->m_ref
, &record
->m_impl
->m_ref
, flags
, rrtype
, len
, v
, ttl
);
802 throw new DNSServiceException(err
);
812 * This maps to DNSServiceUpdateRecord(). Returns an
813 * initialized ServiceRef on success, throws an exception
817 DNSService::UpdateRecord
827 Byte __pin
* p
= NULL
;
830 if ((rdata
!= NULL
) && (rdata
->Length
> 0))
837 int err
= DNSServiceUpdateRecord(sdRef
->m_impl
->m_ref
, record
? record
->m_impl
->m_ref
: NULL
, flags
, len
, v
, ttl
);
841 throw new DNSServiceException(err
);
849 * This maps to DNSServiceRemoveRecord(). Returns an
850 * initialized ServiceRef on success, throws an exception
854 DNSService::RemoveRecord
861 int err
= DNSServiceRemoveRecord(sdRef
->m_impl
->m_ref
, record
->m_impl
->m_ref
, flags
);
865 throw new DNSServiceException(err
);
873 * This maps to DNSServiceBrowse(). Returns an
874 * initialized ServiceRef on success, throws an exception
884 BrowseReply
* callback
887 ServiceRef
* sdRef
= new ServiceRef(callback
);
888 PString
* pType
= new PString(regtype
);
889 PString
* pDomain
= new PString(domain
);
891 int err
= DNSServiceBrowse(&sdRef
->m_impl
->m_ref
, flags
, interfaceIndex
, pType
->c_str(), pDomain
->c_str(),(DNSServiceBrowseReply
) ServiceRef::ServiceRefImpl::BrowseCallback
, sdRef
->m_impl
);
895 throw new DNSServiceException(err
);
898 sdRef
->StartThread();
907 * This maps to DNSServiceResolve(). Returns an
908 * initialized ServiceRef on success, throws an exception
919 ResolveReply
* callback
922 ServiceRef
* sdRef
= new ServiceRef(callback
);
923 PString
* pName
= new PString(name
);
924 PString
* pType
= new PString(regtype
);
925 PString
* pDomain
= new PString(domain
);
927 int err
= DNSServiceResolve(&sdRef
->m_impl
->m_ref
, flags
, interfaceIndex
, pName
->c_str(), pType
->c_str(), pDomain
->c_str(),(DNSServiceResolveReply
) ServiceRef::ServiceRefImpl::ResolveCallback
, sdRef
->m_impl
);
931 throw new DNSServiceException(err
);
934 sdRef
->StartThread();
943 * This maps to DNSServiceCreateConnection(). Returns an
944 * initialized ServiceRef on success, throws an exception
948 DNSService::CreateConnection
950 RegisterRecordReply
* callback
953 ServiceRef
* sdRef
= new ServiceRef(callback
);
955 int err
= DNSServiceCreateConnection(&sdRef
->m_impl
->m_ref
);
959 throw new DNSServiceException(err
);
962 sdRef
->StartThread();
971 * This maps to DNSServiceRegisterRecord(). Returns an
972 * initialized ServiceRef on success, throws an exception
977 DNSService::RegisterRecord
989 RecordRef
* record
= new RecordRef
;
991 Byte __pin
* p
= NULL
;
994 PString
* pFullname
= new PString(fullname
);
996 if ((rdata
!= NULL
) && (rdata
->Length
> 0))
1003 int err
= DNSServiceRegisterRecord(sdRef
->m_impl
->m_ref
, &record
->m_impl
->m_ref
, flags
, interfaceIndex
, pFullname
->c_str(), rrtype
, rrclass
, len
, v
, ttl
, (DNSServiceRegisterRecordReply
) ServiceRef::ServiceRefImpl::RegisterRecordCallback
, sdRef
->m_impl
);
1007 throw new DNSServiceException(err
);
1016 * This maps to DNSServiceQueryRecord(). Returns an
1017 * initialized ServiceRef on success, throws an exception
1021 DNSService::QueryRecord
1028 QueryRecordReply
* callback
1031 ServiceRef
* sdRef
= new ServiceRef(callback
);
1032 PString
* pFullname
= new PString(fullname
);
1034 int err
= DNSServiceQueryRecord(&sdRef
->m_impl
->m_ref
, flags
, interfaceIndex
, pFullname
->c_str(), rrtype
, rrclass
, (DNSServiceQueryRecordReply
) ServiceRef::ServiceRefImpl::QueryRecordCallback
, sdRef
->m_impl
);
1038 throw new DNSServiceException(err
);
1041 sdRef
->StartThread();
1050 * This maps to DNSServiceReconfirmRecord(). Returns an
1051 * initialized ServiceRef on success, throws an exception
1055 DNSService::ReconfirmRecord
1066 Byte __pin
* p
= NULL
;
1069 PString
* pFullname
= new PString(fullname
);
1071 if ((rdata
!= NULL
) && (rdata
->Length
> 0))
1073 len
= rdata
->Length
;
1078 DNSServiceReconfirmRecord(flags
, interfaceIndex
, pFullname
->c_str(), rrtype
, rrclass
, len
, v
);
1083 TextRecord::SetValue
1086 Byte value
[] /* may be NULL */
1089 PString
* pKey
= new PString(key
);
1091 Byte __pin
* p
= NULL
;
1093 DNSServiceErrorType err
;
1095 if (value
&& (value
->Length
> 0))
1097 len
= value
->Length
;
1102 err
= TXTRecordSetValue(&m_impl
->m_ref
, pKey
->c_str(), len
, v
);
1106 throw new DNSServiceException(err
);
1112 TextRecord::RemoveValue
1117 PString
* pKey
= new PString(key
);
1118 DNSServiceErrorType err
;
1120 err
= TXTRecordRemoveValue(&m_impl
->m_ref
, pKey
->c_str());
1124 throw new DNSServiceException(err
);
1130 TextRecord::GetLength
1134 return TXTRecordGetLength(&m_impl
->m_ref
);
1139 TextRecord::GetBytes
1143 const void * noGCBytes
= NULL
;
1144 Byte gcBytes
[] = NULL
;
1146 noGCBytes
= TXTRecordGetBytesPtr(&m_impl
->m_ref
);
1147 int len
= GetLength();
1149 if (noGCBytes
&& len
)
1151 gcBytes
= new Byte
[len
];
1152 Byte __pin
* p
= &gcBytes
[0];
1153 memcpy(p
, noGCBytes
, len
);
1161 TextRecord::ContainsKey
1167 PString
* pKey
= new PString(key
);
1168 Byte __pin
* p
= &txtRecord
[0];
1170 return (TXTRecordContainsKey(txtRecord
->Length
, p
, pKey
->c_str()) > 0) ? true : false;
1175 TextRecord::GetValueBytes
1183 PString
* pKey
= new PString(key
);
1184 Byte __pin
* p1
= &txtRecord
[0];
1187 v
= TXTRecordGetValuePtr(txtRecord
->Length
, p1
, pKey
->c_str(), &valueLen
);
1191 ret
= new Byte
[valueLen
];
1192 Byte __pin
* p2
= &ret
[0];
1194 memcpy(p2
, v
, valueLen
);
1202 TextRecord::GetCount
1207 Byte __pin
* p
= &txtRecord
[0];
1209 return TXTRecordGetCount(txtRecord
->Length
, p
);
1214 TextRecord::GetItemAtIndex
1222 uint8_t keyBufLen
= 255;
1226 DNSServiceErrorType err
;
1227 Byte __pin
* p1
= &txtRecord
[0];
1230 err
= TXTRecordGetItemAtIndex(txtRecord
->Length
, p1
, index
, keyBufLen
, keyBuf
, &valueLen
, (const void**) &value
);
1234 throw new DNSServiceException(err
);
1237 *key
= ConvertToString(keyBuf
);
1241 ret
= new Byte
[valueLen
];
1242 Byte __pin
* p2
= &ret
[0];
1244 memcpy(p2
, value
, valueLen
);
1252 // DNSServiceException::DNSServiceException()
1254 // Constructs an exception with an error code
1256 DNSServiceException::DNSServiceException
1267 // This version of the constructor is useful for instances in which
1268 // an inner exception is thrown, caught, and then a new exception
1269 // is thrown in it's place
1271 DNSServiceException::DNSServiceException
1274 System::Exception
* innerException