📄 pdns.cxx
字号:
/*
* pdns.cxx
*
* Portable Windows Library
*
* Copyright (c) 2003 Equivalence Pty. Ltd.
*
* The contents of this file are subject to the Mozilla Public License
* Version 1.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.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS IS"
* basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See
* the License for the specific language governing rights and limitations
* under the License.
*
* The Original Code is Portable Windows Library.
*
* The Initial Developer of the Original Code is Equivalence Pty. Ltd.
*
* Contributor(s): ______________________________________.
*
* Copyright 2003 Equivalence Pty. Ltd.
*
* $Log: pdns.cxx,v $
* Revision 1.1 2006/06/29 04:18:03 joegenbaclor
* *** empty log message ***
*
* Revision 1.28 2006/04/22 13:35:24 dsandras
* Fixed wrong behavior with different priorities when building the priorities
* list. Fixes Ekiga bug #339314.
*
* Revision 1.27 2006/04/12 10:38:10 csoutheren
* Fixed problem with looping in SRV records thanks to Damien Sandras
*
* Revision 1.26 2006/02/26 11:51:20 csoutheren
* Extended DNS test program to include URL based SRV lookups
* Re-arranged SRV lookup code to allow access to internal routine
* Reformatted code
*
* Revision 1.25 2006/02/26 09:26:17 shorne
* Added DNS SRV record lookups
*
* Revision 1.24 2005/12/04 22:43:30 csoutheren
* Cleanup patches from Kilian Krause
*
* Revision 1.23 2005/11/30 12:47:41 csoutheren
* Removed tabs, reformatted some code, and changed tags for Doxygen
*
* Revision 1.22 2005/04/27 12:08:11 csoutheren
* Added support for res_minit for thread-safe resolver access
* Added mutex when res_minit not available
*
* Revision 1.21 2004/11/15 23:47:18 csoutheren
* Fixed problem with empty SRV names
*
* Revision 1.20 2004/11/15 11:33:52 csoutheren
* Fixed problem in SRV record handling
*
* Revision 1.19 2004/06/24 07:36:24 csoutheren
* Added definitions of T_SRV and T_NAPTR for hosts that do not have these
*
* Revision 1.18 2004/06/01 23:30:38 csoutheren
* Removed warning under Linux
*
* Revision 1.17 2004/05/31 23:14:17 csoutheren
* Fixed warnings under VS.net and fixed problem with SRV records when returning multiple records
*
* Revision 1.16 2004/05/31 12:49:48 csoutheren
* Added handling of unknown DNS types
*
* Revision 1.15 2004/05/31 12:14:13 rjongbloed
* Fixed missing namespace selector on function definition.
* Opyimised some string processing
*
* Revision 1.14 2004/05/28 06:50:45 csoutheren
* Reorganised DNS functions to use templates, and exposed more internals to allow new DNS lookup types to be added
*
* Revision 1.13 2004/04/09 06:52:17 rjongbloed
* Removed #pargma linker command for /delayload of DLL as documentations sais that
* you cannot do this.
*
* Revision 1.12 2004/02/23 23:52:19 csoutheren
* Added pragmas to avoid every Windows application needing to include libs explicitly
*
* Revision 1.11 2004/01/03 03:37:53 csoutheren
* Fixed compile problem on Linux
*
* Revision 1.10 2004/01/03 03:10:42 csoutheren
* Fixed more problems with looking up SRV records, especially on Windows
*
* Revision 1.9 2004/01/02 13:22:04 csoutheren
* Fixed problem with extracting SRV records from DNS
*
* Revision 1.8 2003/11/02 15:52:58 shawn
* added arpa/nameser_compat.h for Mac OS X 10.3 (Panther)
*
* Revision 1.7 2003/04/28 23:57:40 robertj
* Fixed Solaris compatibility
*
* Revision 1.6 2003/04/22 23:21:37 craigs
* Swapped includes at request of Shawn Hsiao for compatiobility with MacOSX
*
* Revision 1.5 2003/04/16 14:21:12 craigs
* Added set of T_SRV for MacOS
*
* Revision 1.4 2003/04/16 08:00:19 robertj
* Windoes psuedo autoconf support
*
* Revision 1.3 2003/04/15 08:14:32 craigs
* Added single string form of GetSRVRecords
*
* Revision 1.2 2003/04/15 08:05:19 craigs
* Added Unix implementation
*
* Revision 1.1 2003/04/15 04:06:35 craigs
* Initial version
*
*/
#ifdef __GNUC__
#pragma implementation "pdns.h"
#endif
#include <ptlib.h>
#include <ptclib/pdns.h>
#include <ptclib/url.h>
#include <ptlib/ipsock.h>
#if P_DNS
/////////////////////////////////////////////////
#ifdef P_HAS_RESOLVER
static BOOL GetDN(const BYTE * reply, const BYTE * replyEnd, BYTE * & cp, char * buff)
{
int len = dn_expand(reply, replyEnd, cp, buff, MAXDNAME);
if (len < 0)
return FALSE;
cp += len;
return TRUE;
}
static BOOL ProcessDNSRecords(
const BYTE * reply,
const BYTE * replyEnd,
BYTE * cp,
PINDEX anCount,
PINDEX nsCount,
PINDEX arCount,
PDNS_RECORD * results)
{
PDNS_RECORD lastRecord = NULL;
PINDEX rrCount = anCount + nsCount + arCount;
nsCount += anCount;
arCount += nsCount;
PINDEX i;
for (i = 0; i < rrCount; i++) {
int section;
if (i < anCount)
section = DnsSectionAnswer;
else if (i < nsCount)
section = DnsSectionAuthority;
else // if (i < arCount)
section = DnsSectionAddtional;
// get the name
char pName[MAXDNAME];
if (!GetDN(reply, replyEnd, cp, pName))
return FALSE;
// get other common parts of the record
WORD type;
WORD dnsClass;
DWORD ttl;
WORD dlen;
GETSHORT(type, cp);
GETSHORT(dnsClass, cp);
GETLONG (ttl, cp);
GETSHORT(dlen, cp);
BYTE * data = cp;
cp += dlen;
PDNS_RECORD newRecord = NULL;
switch (type) {
default:
newRecord = (PDNS_RECORD)malloc(sizeof(DnsRecord) + sizeof(DWORD) + dlen);
newRecord->Data.Null.dwByteCount = dlen;
memcpy(&newRecord->Data, data, dlen);
break;
case T_SRV:
newRecord = (PDNS_RECORD)malloc(sizeof(DnsRecord));
memset(newRecord, 0, sizeof(DnsRecord));
GETSHORT(newRecord->Data.SRV.wPriority, data);
GETSHORT(newRecord->Data.SRV.wWeight, data);
GETSHORT(newRecord->Data.SRV.wPort, data);
if (!GetDN(reply, replyEnd, data, newRecord->Data.SRV.pNameTarget)) {
free(newRecord);
return FALSE;
}
break;
case T_MX:
newRecord = (PDNS_RECORD)malloc(sizeof(DnsRecord));
memset(newRecord, 0, sizeof(DnsRecord));
GETSHORT(newRecord->Data.MX.wPreference, data);
if (!GetDN(reply, replyEnd, data, newRecord->Data.MX.pNameExchange)) {
free(newRecord);
return FALSE;
}
break;
case T_A:
newRecord = (PDNS_RECORD)malloc(sizeof(DnsRecord));
memset(newRecord, 0, sizeof(DnsRecord));
GETLONG(newRecord->Data.A.IpAddress, data);
break;
case T_NS:
newRecord = (PDNS_RECORD)malloc(sizeof(DnsRecord));
memset(newRecord, 0, sizeof(DnsRecord));
if (!GetDN(reply, replyEnd, data, newRecord->Data.NS.pNameHost)) {
delete newRecord;
return FALSE;
}
break;
}
// initialise the new record
if (newRecord != NULL) {
newRecord->wType = type;
newRecord->Flags.S.Section = section;
newRecord->pNext = NULL;
strcpy(newRecord->pName, pName);
if (*results == NULL)
*results = newRecord;
if (lastRecord != NULL)
lastRecord->pNext = newRecord;
lastRecord = newRecord;
newRecord = NULL;
}
}
return TRUE;
}
void DnsRecordListFree(PDNS_RECORD rec, int /* FreeType */)
{
while (rec != NULL) {
PDNS_RECORD next = rec->pNext;
free(rec);
rec = next;
}
}
#if ! P_HAS_RES_NINIT
static PMutex & GetDNSMutex()
{
static PMutex mutex;
return mutex;
}
#endif
DNS_STATUS DnsQuery_A(const char * service,
WORD requestType,
DWORD options,
void *,
PDNS_RECORD * results,
void *)
{
if (results == NULL)
return -1;
*results = NULL;
#if P_HAS_RES_NINIT
res_ninit(&_res);
#else
res_init();
GetDNSMutex().Wait();
#endif
union {
HEADER hdr;
BYTE buf[PACKETSZ];
} reply;
#if P_HAS_RES_NINIT
int replyLen = res_nsearch(&_res, service, C_IN, requestType, (BYTE *)&reply, sizeof(reply));
#else
int replyLen = res_search(service, C_IN, requestType, (BYTE *)&reply, sizeof(reply));
GetDNSMutex().Signal();
#endif
if (replyLen < 1)
return -1;
BYTE * replyStart = reply.buf;
BYTE * replyEnd = reply.buf + replyLen;
BYTE * cp = reply.buf + sizeof(HEADER);
// ignore questions in response
uint16_t i;
for (i = 0; i < ntohs(reply.hdr.qdcount); i++) {
char qName[MAXDNAME];
if (!GetDN(replyStart, replyEnd, cp, qName))
return -1;
cp += QFIXEDSZ;
}
if (!ProcessDNSRecords(
replyStart,
replyEnd,
cp,
ntohs(reply.hdr.ancount),
ntohs(reply.hdr.nscount),
ntohs(reply.hdr.arcount),
results)) {
DnsRecordListFree(*results, 0);
return -1;
}
return 0;
}
#endif // P_HAS_RESOLVER
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -