#define mDNS_InstantiateInlines 1
#include "DNSCommon.h"
#if (defined(_MSC_VER))
#pragma warning(disable:4127)
#pragma warning(disable:4295)
#endif
#if COMPILER_LIKES_PRAGMA_MARK
#pragma mark -
#pragma mark - DNameList copy/deallocation routines
#endif
mDNSexport DNameListElem *mDNS_CopyDNameList(const DNameListElem *orig)
{
DNameListElem *copy = mDNSNULL, *newelem;
const DNameListElem *ptr;
for (ptr = orig; ptr; ptr = ptr->next)
{
newelem = (DNameListElem*)mDNSPlatformMemAllocate(sizeof(DNameListElem));
if (!newelem) { LogMsg("ERROR: malloc"); return mDNSNULL; }
AssignDomainName(newelem->name, ptr->name);
newelem->next = copy;
copy = newelem;
}
return copy;
}
mDNSexport void mDNS_FreeDNameList(DNameListElem *list)
{
DNameListElem *fptr;
while (list)
{
fptr = list;
list = list->next;
mDNSPlatformMemFree(fptr);
}
}
#if COMPILER_LIKES_PRAGMA_MARK
#pragma mark -
#pragma mark - General Utility Functions
#endif
mDNSexport mDNSBool IsPrivateV4Addr(mDNSAddr *addr)
{
mDNSu8 *b;
if (addr->type != mDNSAddrType_IPv4) return mDNSfalse;
b = addr->ip.v4.b;
return ((b[0] == 10) || (b[0] == 172 && b[1] > 15 && b[1] < 32) || (b[0] == 192 && b[1] == 168)); }
mDNSexport const NetworkInterfaceInfo *GetFirstActiveInterface(const NetworkInterfaceInfo *intf)
{
while (intf && !intf->InterfaceActive) intf = intf->next;
return(intf);
}
mDNSexport mDNSInterfaceID GetNextActiveInterfaceID(const NetworkInterfaceInfo *intf)
{
const NetworkInterfaceInfo *next = GetFirstActiveInterface(intf->next);
if (next) return(next->InterfaceID); else return(mDNSNULL);
}
mDNSexport mDNSu32 NumCacheRecordsForInterfaceID(const mDNS *const m, mDNSInterfaceID id)
{
mDNSu32 slot, used = 0;
CacheRecord *rr;
for (slot = 0; slot < CACHE_HASH_SLOTS; slot++)
for (rr = m->rrcache_hash[slot]; rr; rr=rr->next)
if (rr->resrec.InterfaceID == id) used++;
return(used);
}
mDNSexport char *DNSTypeName(mDNSu16 rrtype)
{
switch (rrtype)
{
case kDNSType_A: return("Addr");
case kDNSType_CNAME:return("CNAME");
case kDNSType_NULL: return("NULL");
case kDNSType_PTR: return("PTR");
case kDNSType_HINFO:return("HINFO");
case kDNSType_TXT: return("TXT");
case kDNSType_AAAA: return("AAAA");
case kDNSType_SRV: return("SRV");
case kDNSQType_ANY: return("ANY");
default: {
static char buffer[16];
mDNS_snprintf(buffer, sizeof(buffer), "(%d)", rrtype);
return(buffer);
}
}
}
mDNSexport char *GetRRDisplayString_rdb(const ResourceRecord *rr, RDataBody *rd, char *buffer)
{
char *ptr = buffer;
mDNSu32 length = mDNS_snprintf(buffer, 79, "%4d %##s %s ", rr->rdlength, rr->name.c, DNSTypeName(rr->rrtype));
switch (rr->rrtype)
{
case kDNSType_A: mDNS_snprintf(buffer+length, 79-length, "%.4a", &rd->ipv4); break;
case kDNSType_NS: case kDNSType_CNAME: case kDNSType_PTR: mDNS_snprintf(buffer+length, 79-length, "%##s", rd->name.c); break;
case kDNSType_HINFO: case kDNSType_TXT: mDNS_snprintf(buffer+length, 79-length, "%#s", rd->txt.c); break;
case kDNSType_AAAA: mDNS_snprintf(buffer+length, 79-length, "%.16a", &rd->ipv6); break;
case kDNSType_SRV: mDNS_snprintf(buffer+length, 79-length, "%##s", rd->srv.target.c); break;
default: mDNS_snprintf(buffer+length, 79-length, "RDLen %d: %s", rr->rdlength, rd->data); break;
}
for (ptr = buffer; *ptr; ptr++) if (*ptr < ' ') *ptr='.';
return(buffer);
}
mDNSexport mDNSu32 mDNSRandom(mDNSu32 max)
{
static mDNSu32 seed = 0;
mDNSu32 mask = 1;
if (!seed)
{
int i;
seed = mDNSPlatformRandomSeed(); for (i=0; i<100; i++) seed = seed * 21 + 1; }
while (mask < max) mask = (mask << 1) | 1;
do seed = seed * 21 + 1; while ((seed & mask) > max);
return (seed & mask);
}
mDNSexport mDNSBool mDNSSameAddress(const mDNSAddr *ip1, const mDNSAddr *ip2)
{
if (ip1->type == ip2->type)
{
switch (ip1->type)
{
case mDNSAddrType_None : return(mDNStrue); case mDNSAddrType_IPv4 : return(mDNSBool)(mDNSSameIPv4Address(ip1->ip.v4, ip2->ip.v4));
case mDNSAddrType_IPv6 : return(mDNSBool)(mDNSSameIPv6Address(ip1->ip.v6, ip2->ip.v6));
}
}
return(mDNSfalse);
}
mDNSexport mDNSBool mDNSAddrIsDNSMulticast(const mDNSAddr *ip)
{
switch(ip->type)
{
case mDNSAddrType_IPv4: return(mDNSBool)(ip->ip.v4.NotAnInteger == AllDNSLinkGroupv4.NotAnInteger);
case mDNSAddrType_IPv6: return(mDNSBool)(ip->ip.v6.l[0] == AllDNSLinkGroupv6.l[0] &&
ip->ip.v6.l[1] == AllDNSLinkGroupv6.l[1] &&
ip->ip.v6.l[2] == AllDNSLinkGroupv6.l[2] &&
ip->ip.v6.l[3] == AllDNSLinkGroupv6.l[3] );
default: return(mDNSfalse);
}
}
#if COMPILER_LIKES_PRAGMA_MARK
#pragma mark -
#pragma mark - Domain Name Utility Functions
#endif
mDNSexport mDNSBool SameDomainLabel(const mDNSu8 *a, const mDNSu8 *b)
{
int i;
const int len = *a++;
if (len > MAX_DOMAIN_LABEL)
{ debugf("Malformed label (too long)"); return(mDNSfalse); }
if (len != *b++) return(mDNSfalse);
for (i=0; i<len; i++)
{
mDNSu8 ac = *a++;
mDNSu8 bc = *b++;
if (mDNSIsUpperCase(ac)) ac += 'a' - 'A';
if (mDNSIsUpperCase(bc)) bc += 'a' - 'A';
if (ac != bc) return(mDNSfalse);
}
return(mDNStrue);
}
mDNSexport mDNSBool SameDomainName(const domainname *const d1, const domainname *const d2)
{
const mDNSu8 * a = d1->c;
const mDNSu8 * b = d2->c;
const mDNSu8 *const max = d1->c + MAX_DOMAIN_NAME;
while (*a || *b)
{
if (a + 1 + *a >= max)
{ debugf("Malformed domain name (more than 255 characters)"); return(mDNSfalse); }
if (!SameDomainLabel(a, b)) return(mDNSfalse);
a += 1 + *a;
b += 1 + *b;
}
return(mDNStrue);
}
mDNSexport mDNSBool IsLocalDomain(const domainname *d)
{
static const domainname *n0 = (domainname*)"\x5" "local";
static const domainname *n1 = (domainname*)"\x3" "254" "\x3" "169" "\x7" "in-addr" "\x4" "arpa";
static const domainname *n2 = (domainname*)"\x1" "0" "\x1" "8" "\x1" "e" "\x1" "f" "\x3" "ip6" "\x4" "arpa";
const domainname *d1, *d2, *d3, *d4, *d5, *d6; d1 = d2 = d3 = d4 = d5 = d6 = mDNSNULL;
while (d->c[0])
{
d6 = d5; d5 = d4; d4 = d3; d3 = d2; d2 = d1; d1 = d;
d = (domainname*)(d->c + 1 + d->c[0]);
}
if (d1 && SameDomainName(d1, n0)) return(mDNStrue);
if (d4 && SameDomainName(d4, n1)) return(mDNStrue);
if (d6 && SameDomainName(d6, n2)) return(mDNStrue);
return(mDNSfalse);
}
mDNSexport mDNSu16 DomainNameLength(const domainname *const name)
{
const mDNSu8 *src = name->c;
while (*src)
{
if (*src > MAX_DOMAIN_LABEL) return(MAX_DOMAIN_NAME+1);
src += 1 + *src;
if (src - name->c >= MAX_DOMAIN_NAME) return(MAX_DOMAIN_NAME+1);
}
return((mDNSu16)(src - name->c + 1));
}
mDNSexport mDNSu16 CompressedDomainNameLength(const domainname *const name, const domainname *parent)
{
const mDNSu8 *src = name->c;
if (parent && parent->c[0] == 0) parent = mDNSNULL;
while (*src)
{
if (*src > MAX_DOMAIN_LABEL) return(MAX_DOMAIN_NAME+1);
if (parent && SameDomainName((domainname *)src, parent)) return((mDNSu16)(src - name->c + 2));
src += 1 + *src;
if (src - name->c >= MAX_DOMAIN_NAME) return(MAX_DOMAIN_NAME+1);
}
return((mDNSu16)(src - name->c + 1));
}
mDNSexport mDNSu8 *AppendLiteralLabelString(domainname *const name, const char *cstr)
{
mDNSu8 * ptr = name->c + DomainNameLength(name) - 1; const mDNSu8 *const lim1 = name->c + MAX_DOMAIN_NAME - 1; const mDNSu8 *const lim2 = ptr + 1 + MAX_DOMAIN_LABEL;
const mDNSu8 *const lim = (lim1 < lim2) ? lim1 : lim2;
mDNSu8 *lengthbyte = ptr++;
while (*cstr && ptr < lim) *ptr++ = (mDNSu8)*cstr++; *lengthbyte = (mDNSu8)(ptr - lengthbyte - 1); *ptr++ = 0; if (*cstr) return(mDNSNULL); else return(ptr); }
mDNSexport mDNSu8 *AppendDNSNameString(domainname *const name, const char *cstring)
{
const char *cstr = cstring;
mDNSu8 * ptr = name->c + DomainNameLength(name) - 1; const mDNSu8 *const lim = name->c + MAX_DOMAIN_NAME - 1; while (*cstr && ptr < lim) {
mDNSu8 *lengthbyte = ptr++; if (*cstr == '.') { LogMsg("AppendDNSNameString: Illegal empty label in name \"%s\"", cstring); return(mDNSNULL); }
while (*cstr && *cstr != '.' && ptr < lim) {
mDNSu8 c = (mDNSu8)*cstr++; if (c == '\\') {
c = (mDNSu8)*cstr++; if (mdnsIsDigit(cstr[-1]) && mdnsIsDigit(cstr[0]) && mdnsIsDigit(cstr[1]))
{ int v0 = cstr[-1] - '0'; int v1 = cstr[ 0] - '0';
int v2 = cstr[ 1] - '0';
int val = v0 * 100 + v1 * 10 + v2;
if (val <= 255) { c = (mDNSu8)val; cstr += 2; } }
}
*ptr++ = c; }
if (*cstr) cstr++; if (ptr - lengthbyte - 1 > MAX_DOMAIN_LABEL) return(mDNSNULL);
*lengthbyte = (mDNSu8)(ptr - lengthbyte - 1); }
*ptr++ = 0; if (*cstr) return(mDNSNULL); else return(ptr); }
mDNSexport mDNSu8 *AppendDomainLabel(domainname *const name, const domainlabel *const label)
{
int i;
mDNSu8 *ptr = name->c + DomainNameLength(name) - 1;
if (label->c[0] > MAX_DOMAIN_LABEL) return(mDNSNULL);
if (ptr + 1 + label->c[0] + 1 > name->c + MAX_DOMAIN_NAME) return(mDNSNULL);
for (i=0; i<=label->c[0]; i++) *ptr++ = label->c[i]; *ptr++ = 0; return(ptr);
}
mDNSexport mDNSu8 *AppendDomainName(domainname *const name, const domainname *const append)
{
mDNSu8 * ptr = name->c + DomainNameLength(name) - 1; const mDNSu8 *const lim = name->c + MAX_DOMAIN_NAME - 1; const mDNSu8 * src = append->c;
while(src[0])
{
int i;
if (ptr + 1 + src[0] > lim) return(mDNSNULL);
for (i=0; i<=src[0]; i++) *ptr++ = src[i];
*ptr = 0; src += i;
}
return(ptr);
}
mDNSexport mDNSBool MakeDomainLabelFromLiteralString(domainlabel *const label, const char *cstr)
{
mDNSu8 * ptr = label->c + 1; const mDNSu8 *const limit = label->c + 1 + MAX_DOMAIN_LABEL; while (*cstr && ptr < limit) *ptr++ = (mDNSu8)*cstr++; label->c[0] = (mDNSu8)(ptr - label->c - 1); return(*cstr == 0); }
mDNSexport mDNSu8 *MakeDomainNameFromDNSNameString(domainname *const name, const char *cstr)
{
name->c[0] = 0; return(AppendDNSNameString(name, cstr)); }
mDNSexport char *ConvertDomainLabelToCString_withescape(const domainlabel *const label, char *ptr, char esc)
{
const mDNSu8 * src = label->c; const mDNSu8 len = *src++; const mDNSu8 *const end = src + len; if (len > MAX_DOMAIN_LABEL) return(mDNSNULL); while (src < end) {
mDNSu8 c = *src++;
if (esc)
{
if (c == '.' || c == esc) *ptr++ = esc; else if (c <= ' ') { *ptr++ = esc;
*ptr++ = (char) ('0' + (c / 100) );
*ptr++ = (char) ('0' + (c / 10) % 10);
c = (mDNSu8)('0' + (c ) % 10);
}
}
*ptr++ = (char)c; }
*ptr = 0; return(ptr); }
mDNSexport char *ConvertDomainNameToCString_withescape(const domainname *const name, char *ptr, char esc)
{
const mDNSu8 *src = name->c; const mDNSu8 *const max = name->c + MAX_DOMAIN_NAME;
if (*src == 0) *ptr++ = '.';
while (*src) {
if (src + 1 + *src >= max) return(mDNSNULL);
ptr = ConvertDomainLabelToCString_withescape((const domainlabel *)src, ptr, esc);
if (!ptr) return(mDNSNULL);
src += 1 + *src;
*ptr++ = '.'; }
*ptr++ = 0; return(ptr); }
mDNSexport void ConvertUTF8PstringToRFC1034HostLabel(const mDNSu8 UTF8Name[], domainlabel *const hostlabel)
{
const mDNSu8 * src = &UTF8Name[1];
const mDNSu8 *const end = &UTF8Name[1] + UTF8Name[0];
mDNSu8 * ptr = &hostlabel->c[1];
const mDNSu8 *const lim = &hostlabel->c[1] + MAX_DOMAIN_LABEL;
while (src < end)
{
if (src[0] == '\'') { src++; continue; } if (src + 2 < end && src[0] == 0xE2 && src[1] == 0x80 && src[2] == 0x99)
{ src += 3; continue; } if (ptr < lim)
{
if (mdnsValidHostChar(*src, (ptr > &hostlabel->c[1]), (src < end-1))) *ptr++ = *src;
else if (ptr > &hostlabel->c[1] && ptr[-1] != '-') *ptr++ = '-';
}
src++;
}
while (ptr > &hostlabel->c[1] && ptr[-1] == '-') ptr--; hostlabel->c[0] = (mDNSu8)(ptr - &hostlabel->c[1]);
}
mDNSexport mDNSu8 *ConstructServiceName(domainname *const fqdn,
const domainlabel *name, const domainname *type, const domainname *const domain)
{
int i, len;
mDNSu8 *dst = fqdn->c;
const mDNSu8 *src;
const char *errormsg;
if (!name && type)
{
const mDNSu8 *s0 = type->c;
if (s0[0] && s0[0] < 0x40) {
const mDNSu8 * s1 = s0 + 1 + s0[0];
if (s1[0] && s1[0] < 0x40) {
const mDNSu8 *s2 = s1 + 1 + s1[0];
if (s2[0] && s2[0] < 0x40 && s2[1+s2[0]] == 0) {
static const mDNSu8 SubTypeLabel[5] = "\x04_sub";
src = s0; len = *src;
for (i=0; i <= len; i++) *dst++ = *src++;
for (i=0; i < (int)sizeof(SubTypeLabel); i++) *dst++ = SubTypeLabel[i];
type = (domainname *)s1;
if (SameDomainName((domainname*)s0, (domainname*)"\x09_services\x07_dns-sd\x04_udp") ||
SameDomainName((domainname*)s0, (domainname*)"\x09_services\x05_mdns\x04_udp"))
dst -= sizeof(SubTypeLabel);
}
}
}
}
if (name && name->c[0])
{
src = name->c; len = *src;
if (len >= 0x40) { errormsg="Service instance name too long"; goto fail; }
for (i=0; i<=len; i++) *dst++ = *src++;
}
else
name = (domainlabel*)"";
src = type->c; len = *src;
if (len < 2 || len >= 0x40 || (len > 15 && !SameDomainName(domain, (domainname*)"\x05" "local")))
{
errormsg="Application protocol name must be underscore plus 1-14 characters. See <http://www.dns-sd.org/ServiceTypes.html>";
goto fail;
}
if (src[1] != '_') { errormsg="Application protocol name must begin with underscore"; goto fail; }
for (i=2; i<=len; i++)
if (!mdnsIsLetter(src[i]) && !mdnsIsDigit(src[i]) && src[i] != '-' && src[i] != '_')
{ errormsg="Application protocol name must contain only letters, digits, and hyphens"; goto fail; }
for (i=0; i<=len; i++) *dst++ = *src++;
len = *src;
if (!(len == 4 && src[1] == '_' &&
(((src[2] | 0x20) == 'u' && (src[3] | 0x20) == 'd') || ((src[2] | 0x20) == 't' && (src[3] | 0x20) == 'c')) &&
(src[4] | 0x20) == 'p'))
{ errormsg="Transport protocol name must be _udp or _tcp"; goto fail; }
for (i=0; i<=len; i++) *dst++ = *src++;
if (*src) { errormsg="Service type must have only two labels"; goto fail; }
*dst = 0;
if (!domain->c[0]) { errormsg="Service domain must be non-empty"; goto fail; }
if (SameDomainName(domain, (domainname*)"\x05" "local" "\x04" "arpa"))
{ errormsg="Illegal domain \"local.arpa.\" Use \"local.\" (or empty string)"; goto fail; }
dst = AppendDomainName(fqdn, domain);
if (!dst) { errormsg="Service domain too long"; goto fail; }
return(dst);
fail:
LogMsg("ConstructServiceName: %s: %#s.%##s%##s", errormsg, name->c, type->c, domain->c);
return(mDNSNULL);
}
mDNSexport mDNSBool DeconstructServiceName(const domainname *const fqdn,
domainlabel *const name, domainname *const type, domainname *const domain)
{
int i, len;
const mDNSu8 *src = fqdn->c;
const mDNSu8 *max = fqdn->c + MAX_DOMAIN_NAME;
mDNSu8 *dst;
dst = name->c; len = *src;
if (len >= 0x40) { debugf("DeconstructServiceName: service name too long"); return(mDNSfalse); }
for (i=0; i<=len; i++) *dst++ = *src++;
dst = type->c; len = *src;
if (len >= 0x40) { debugf("DeconstructServiceName: service type too long"); return(mDNSfalse); }
for (i=0; i<=len; i++) *dst++ = *src++;
len = *src;
if (len >= 0x40) { debugf("DeconstructServiceName: service type too long"); return(mDNSfalse); }
for (i=0; i<=len; i++) *dst++ = *src++;
*dst++ = 0;
dst = domain->c; while (*src)
{
len = *src;
if (len >= 0x40)
{ debugf("DeconstructServiceName: service domain label too long"); return(mDNSfalse); }
if (src + 1 + len + 1 >= max)
{ debugf("DeconstructServiceName: service domain too long"); return(mDNSfalse); }
for (i=0; i<=len; i++) *dst++ = *src++;
}
*dst++ = 0;
return(mDNStrue);
}
mDNSexport mDNSBool LabelContainsSuffix(const domainlabel *const name, const mDNSBool RichText)
{
mDNSu16 l = name->c[0];
if (RichText)
{
if (l < 4) return mDNSfalse; if (name->c[l--] != ')') return mDNSfalse; if (!mdnsIsDigit(name->c[l])) return mDNSfalse; l--;
while (l > 2 && mdnsIsDigit(name->c[l])) l--; return (name->c[l] == '(' && name->c[l - 1] == ' ');
}
else
{
if (l < 2) return mDNSfalse; if (!mdnsIsDigit(name->c[l])) return mDNSfalse; l--;
while (l > 2 && mdnsIsDigit(name->c[l])) l--; return (name->c[l] == '-');
}
}
mDNSexport mDNSu32 RemoveLabelSuffix(domainlabel *name, mDNSBool RichText)
{
mDNSu32 val = 0, multiplier = 1;
if (RichText && name->c[0] >= 1 && name->c[name->c[0]] == ')') name->c[0]--;
while (mdnsIsDigit(name->c[name->c[0]]))
{ val += (name->c[name->c[0]] - '0') * multiplier; multiplier *= 10; name->c[0]--; }
if (RichText)
{
if (name->c[0] >= 2 && name->c[name->c[0]] == '(' && name->c[name->c[0]-1] == ' ') name->c[0] -= 2;
}
else
{
if (name->c[0] >= 1 && name->c[name->c[0]] == '-') name->c[0] -= 1;
}
return(val);
}
mDNSexport void AppendLabelSuffix(domainlabel *name, mDNSu32 val, mDNSBool RichText)
{
mDNSu32 divisor = 1, chars = 2; if (RichText) chars = 4;
if (RichText) while (name->c[name->c[0]] == ' ') name->c[0]--;
while (val >= divisor * 10) { divisor *= 10; chars++; }
if (name->c[0] > (mDNSu8)(MAX_DOMAIN_LABEL - chars))
{
name->c[0] = (mDNSu8)(MAX_DOMAIN_LABEL - chars);
while (name->c[0] > 0 && (name->c[name->c[0]+1] & 0xC0) == 0x80) name->c[0]--;
}
if (RichText) { name->c[++name->c[0]] = ' '; name->c[++name->c[0]] = '('; }
else { name->c[++name->c[0]] = '-'; }
while (divisor)
{
name->c[++name->c[0]] = (mDNSu8)('0' + val / divisor);
val %= divisor;
divisor /= 10;
}
if (RichText) name->c[++name->c[0]] = ')';
}
mDNSexport void IncrementLabelSuffix(domainlabel *name, mDNSBool RichText)
{
mDNSu32 val = 0;
if (LabelContainsSuffix(name, RichText))
val = RemoveLabelSuffix(name, RichText);
if (val == 0) val = 2;
else if (val < 10) val++;
else val += 1 + mDNSRandom(99);
AppendLabelSuffix(name, val, RichText);
}
#if COMPILER_LIKES_PRAGMA_MARK
#pragma mark -
#pragma mark - Resource Record Utility Functions
#endif
mDNSexport mDNSu32 RDataHashValue(mDNSu16 const rdlength, const RDataBody *const rdb)
{
mDNSu32 sum = 0;
int i;
for (i=0; i+1 < rdlength; i+=2)
{
sum += (((mDNSu32)(rdb->data[i])) << 8) | rdb->data[i+1];
sum = (sum<<3) | (sum>>29);
}
if (i < rdlength)
{
sum += ((mDNSu32)(rdb->data[i])) << 8;
}
return(sum);
}
mDNSexport mDNSBool SameRData(const ResourceRecord *const r1, const ResourceRecord *const r2)
{
if (r1->rrtype != r2->rrtype) return(mDNSfalse);
if (r1->rdlength != r2->rdlength) return(mDNSfalse);
if (r1->rdatahash != r2->rdatahash) return(mDNSfalse);
if (r1->rdnamehash != r2->rdnamehash) return(mDNSfalse);
switch(r1->rrtype)
{
case kDNSType_CNAME: case kDNSType_PTR: return(SameDomainName(&r1->rdata->u.name, &r2->rdata->u.name));
case kDNSType_SRV: return(mDNSBool)( r1->rdata->u.srv.priority == r2->rdata->u.srv.priority &&
r1->rdata->u.srv.weight == r2->rdata->u.srv.weight &&
r1->rdata->u.srv.port.NotAnInteger == r2->rdata->u.srv.port.NotAnInteger &&
SameDomainName(&r1->rdata->u.srv.target, &r2->rdata->u.srv.target) );
default: return(mDNSPlatformMemSame(r1->rdata->u.data, r2->rdata->u.data, r1->rdlength));
}
}
mDNSexport mDNSBool SameResourceRecord(ResourceRecord *r1, ResourceRecord *r2)
{
return (r1->namehash == r2->namehash &&
r1->rrtype == r2->rrtype &&
SameDomainName(&r1->name, &r2->name) &&
SameRData(r1, r2));
}
mDNSexport mDNSBool ResourceRecordAnswersQuestion(const ResourceRecord *const rr, const DNSQuestion *const q)
{
if (rr->InterfaceID &&
q ->InterfaceID &&
rr->InterfaceID != q->InterfaceID) return(mDNSfalse);
if (rr->rrtype != kDNSType_CNAME && rr->rrtype != q->qtype && q->qtype != kDNSQType_ANY ) return(mDNSfalse);
if ( rr->rrclass != q->qclass && q->qclass != kDNSQClass_ANY) return(mDNSfalse);
return(rr->namehash == q->qnamehash && SameDomainName(&rr->name, &q->qname));
}
mDNSexport mDNSu16 GetRDLength(const ResourceRecord *const rr, mDNSBool estimate)
{
const RDataBody *rd = &rr->rdata->u;
const domainname *const name = estimate ? &rr->name : mDNSNULL;
switch (rr->rrtype)
{
case kDNSType_A: return(sizeof(rd->ipv4));
case kDNSType_CNAME: case kDNSType_NS: case kDNSType_PTR: return(CompressedDomainNameLength(&rd->name, name));
case kDNSType_HINFO:return(mDNSu16)(2 + (int)rd->data[0] + (int)rd->data[1 + (int)rd->data[0]]);
case kDNSType_NULL: case kDNSType_TXT: return(rr->rdlength); case kDNSType_AAAA: return(sizeof(rd->ipv6));
case kDNSType_SRV: return(mDNSu16)(6 + CompressedDomainNameLength(&rd->srv.target, name));
case kDNSType_SOA: return (mDNSu16)(CompressedDomainNameLength(&rd->soa.mname, name) +
CompressedDomainNameLength(&rd->soa.rname, name) +
5 * sizeof(mDNSOpaque32));
case kDNSType_OPT: return(rr->rdlength);
default: debugf("Warning! Don't know how to get length of resource type %d", rr->rrtype);
return(rr->rdlength);
}
}
mDNSexport mDNSBool ValidateRData(const mDNSu16 rrtype, const mDNSu16 rdlength, const RData *const rd)
{
mDNSu16 len;
if (!rdlength) return(mDNSfalse);
switch(rrtype)
{
case kDNSType_A: return(rdlength == sizeof(mDNSv4Addr));
case kDNSType_NS: case kDNSType_MD: case kDNSType_MF: case kDNSType_CNAME: case kDNSType_MB: case kDNSType_MG: case kDNSType_MR: case kDNSType_PTR: len = DomainNameLength(&rd->u.name);
return(len <= MAX_DOMAIN_NAME && rdlength == len);
case kDNSType_HINFO: case kDNSType_MINFO: case kDNSType_TXT: {
const mDNSu8 *ptr = rd->u.txt.c;
const mDNSu8 *end = rd->u.txt.c + rdlength;
while (ptr < end) ptr += 1 + ptr[0];
return (ptr == end);
}
case kDNSType_AAAA: return(rdlength == sizeof(mDNSv6Addr));
case kDNSType_MX: len = DomainNameLength(&rd->u.mx.exchange);
return(len <= MAX_DOMAIN_NAME && rdlength == 2+len);
case kDNSType_SRV: len = DomainNameLength(&rd->u.srv.target);
return(len <= MAX_DOMAIN_NAME && rdlength == 6+len);
default: return(mDNStrue); }
}
#if COMPILER_LIKES_PRAGMA_MARK
#pragma mark -
#pragma mark -
#pragma mark - DNS Message Creation Functions
#endif
mDNSexport void InitializeDNSMessage(DNSMessageHeader *h, mDNSOpaque16 id, mDNSOpaque16 flags)
{
h->id = id;
h->flags = flags;
h->numQuestions = 0;
h->numAnswers = 0;
h->numAuthorities = 0;
h->numAdditionals = 0;
}
mDNSexport const mDNSu8 *FindCompressionPointer(const mDNSu8 *const base, const mDNSu8 *const end, const mDNSu8 *const domname)
{
const mDNSu8 *result = end - *domname - 1;
if (*domname == 0) return(mDNSNULL);
while (result >= base)
{
if (result[0] == domname[0] && result[1] == domname[1])
{
const mDNSu8 *name = domname;
const mDNSu8 *targ = result;
while (targ + *name < end)
{
int i;
const mDNSu8 *pointertarget;
for (i=0; i <= *name; i++) if (targ[i] != name[i]) break;
if (i <= *name) break; targ += 1 + *name; name += 1 + *name; if (*name == 0 && *targ == 0) return(result); if (*name == 0) break;
if (targ[0] < 0x40) continue; if (targ[0] < 0xC0) break; if (targ+1 >= end) break; pointertarget = base + (((mDNSu16)(targ[0] & 0x3F)) << 8) + targ[1];
if (targ < pointertarget) break; if (pointertarget[0] >= 0x40) break; targ = pointertarget;
}
}
result--; }
return(mDNSNULL);
}
mDNSexport mDNSu8 *putDomainNameAsLabels(const DNSMessage *const msg,
mDNSu8 *ptr, const mDNSu8 *const limit, const domainname *const name)
{
const mDNSu8 *const base = (const mDNSu8 *)msg;
const mDNSu8 * np = name->c;
const mDNSu8 *const max = name->c + MAX_DOMAIN_NAME; const mDNSu8 * pointer = mDNSNULL;
const mDNSu8 *const searchlimit = ptr;
while (*np && ptr < limit-1) {
if (*np > MAX_DOMAIN_LABEL)
{ LogMsg("Malformed domain name %##s (label more than 63 bytes)", name->c); return(mDNSNULL); }
if (np + 1 + *np >= max)
{ LogMsg("Malformed domain name %##s (more than 255 bytes)", name->c); return(mDNSNULL); }
if (base) pointer = FindCompressionPointer(base, searchlimit, np);
if (pointer) {
mDNSu16 offset = (mDNSu16)(pointer - base);
*ptr++ = (mDNSu8)(0xC0 | (offset >> 8));
*ptr++ = (mDNSu8)( offset & 0xFF);
return(ptr);
}
else {
int i;
mDNSu8 len = *np++;
if (ptr + 1 + len >= limit) return(mDNSNULL);
*ptr++ = len;
for (i=0; i<len; i++) *ptr++ = *np++;
}
}
if (ptr < limit) {
*ptr++ = 0; return(ptr); }
return(mDNSNULL);
}
mDNSlocal mDNSu8 *putVal16(mDNSu8 *ptr, mDNSu16 val)
{
ptr[0] = (mDNSu8)((val >> 8 ) & 0xFF);
ptr[1] = (mDNSu8)((val ) & 0xFF);
return ptr + sizeof(mDNSOpaque16);
}
mDNSlocal mDNSu8 *putVal32(mDNSu8 *ptr, mDNSu32 val)
{
ptr[0] = (mDNSu8)((val >> 24) & 0xFF);
ptr[1] = (mDNSu8)((val >> 16) & 0xFF);
ptr[2] = (mDNSu8)((val >> 8) & 0xFF);
ptr[3] = (mDNSu8)((val ) & 0xFF);
return ptr + sizeof(mDNSu32);
}
mDNSlocal mDNSu8 *putOptRData(mDNSu8 *ptr, const mDNSu8 *limit, ResourceRecord *rr)
{
int nput = 0;
rdataOpt *opt;
while (nput < rr->rdlength)
{
if (ptr + (2 * sizeof(mDNSu16)) > limit) goto space_err;
(mDNSu8 *)opt = rr->rdata->u.data + nput;
ptr = putVal16(ptr, opt->opt);
ptr = putVal16(ptr, opt->optlen);
nput += 2 * sizeof(mDNSu16);
if (opt->opt == kDNSOpt_LLQ)
{
if (ptr + sizeof(LLQOptData) > limit) goto space_err;
ptr = putVal16(ptr, opt->OptData.llq.vers);
ptr = putVal16(ptr, opt->OptData.llq.llqOp);
ptr = putVal16(ptr, opt->OptData.llq.err);
mDNSPlatformMemCopy(opt->OptData.llq.id, ptr, 8); ptr += 8;
ptr = putVal32(ptr, opt->OptData.llq.lease);
nput += sizeof(LLQOptData);
}
else if (opt->opt == kDNSOpt_Lease)
{
if (ptr + sizeof(mDNSs32) > limit) goto space_err;
ptr = putVal32(ptr, opt->OptData.lease);
nput += sizeof(mDNSs32);
}
else { LogMsg("putOptRData - unknown option %d", opt->opt); return mDNSNULL; }
}
return ptr;
space_err:
LogMsg("ERROR: putOptRData - out of space");
return mDNSNULL;
}
mDNSlocal mDNSu16 getVal16(const mDNSu8 **ptr)
{
mDNSu16 val = (mDNSu16)(((mDNSu16)(*ptr)[0]) << 8 | (*ptr)[1]);
*ptr += sizeof(mDNSOpaque16);
return val;
}
mDNSlocal const mDNSu8 *getOptRdata(const mDNSu8 *ptr, const mDNSu8 *limit, ResourceRecord *rr, mDNSu16 pktRDLen)
{
int nread = 0;
rdataOpt *opt;
while (nread < pktRDLen)
{
opt = (rdataOpt *)(rr->rdata->u.data + nread);
if (nread + (2 * sizeof(mDNSu16)) > rr->rdata->MaxRDLength) goto space_err;
opt->opt = getVal16(&ptr);
opt->optlen = getVal16(&ptr);
nread += 2 * sizeof(mDNSu16);
if (opt->opt == kDNSOpt_LLQ)
{
if ((unsigned)(limit - ptr) < sizeof(LLQOptData)) goto space_err;
opt->OptData.llq.vers = getVal16(&ptr);
opt->OptData.llq.llqOp = getVal16(&ptr);
opt->OptData.llq.err = getVal16(&ptr);
mDNSPlatformMemCopy(ptr, opt->OptData.llq.id, 8);
ptr += 8;
opt->OptData.llq.lease = (mDNSu32) ((mDNSu32)ptr[0] << 24 | (mDNSu32)ptr[1] << 16 | (mDNSu32)ptr[2] << 8 | ptr[3]);
if (opt->OptData.llq.lease > 0x70000000UL / mDNSPlatformOneSecond)
opt->OptData.llq.lease = 0x70000000UL / mDNSPlatformOneSecond;
ptr += sizeof(mDNSOpaque32);
nread += sizeof(LLQOptData);
}
else if (opt->opt == kDNSOpt_Lease)
{
if ((unsigned)(limit - ptr) < sizeof(mDNSs32)) goto space_err;
opt->OptData.lease = (mDNSu32) ((mDNSu32)ptr[0] << 24 | (mDNSu32)ptr[1] << 16 | (mDNSu32)ptr[2] << 8 | ptr[3]);
if (opt->OptData.lease > 0x70000000UL / mDNSPlatformOneSecond)
opt->OptData.lease = 0x70000000UL / mDNSPlatformOneSecond;
ptr += sizeof(mDNSs32);
nread += sizeof(mDNSs32);
}
else { LogMsg("ERROR: getOptRdata - unknown opt %d", opt->opt); return mDNSNULL; }
}
rr->rdlength = pktRDLen;
return ptr;
space_err:
LogMsg("ERROR: getLLQRdata - out of space");
return mDNSNULL;
}
mDNSexport mDNSu8 *putRData(const DNSMessage *const msg, mDNSu8 *ptr, const mDNSu8 *const limit, ResourceRecord *rr)
{
switch (rr->rrtype)
{
case kDNSType_A: if (rr->rdlength != 4)
{
debugf("putRData: Illegal length %d for kDNSType_A", rr->rdlength);
return(mDNSNULL);
}
if (ptr + 4 > limit) return(mDNSNULL);
*ptr++ = rr->rdata->u.ipv4.b[0];
*ptr++ = rr->rdata->u.ipv4.b[1];
*ptr++ = rr->rdata->u.ipv4.b[2];
*ptr++ = rr->rdata->u.ipv4.b[3];
return(ptr);
case kDNSType_CNAME: case kDNSType_PTR: return(putDomainNameAsLabels(msg, ptr, limit, &rr->rdata->u.name));
case kDNSType_AAAA: if (rr->rdlength != sizeof(rr->rdata->u.ipv6))
{
debugf("putRData: Illegal length %d for kDNSType_AAAA", rr->rdlength);
return(mDNSNULL);
}
if (ptr + sizeof(rr->rdata->u.ipv6) > limit) return(mDNSNULL);
mDNSPlatformMemCopy(&rr->rdata->u.ipv6, ptr, sizeof(rr->rdata->u.ipv6));
return(ptr + sizeof(rr->rdata->u.ipv6));
case kDNSType_SRV: if (ptr + 6 > limit) return(mDNSNULL);
*ptr++ = (mDNSu8)(rr->rdata->u.srv.priority >> 8);
*ptr++ = (mDNSu8)(rr->rdata->u.srv.priority & 0xFF);
*ptr++ = (mDNSu8)(rr->rdata->u.srv.weight >> 8);
*ptr++ = (mDNSu8)(rr->rdata->u.srv.weight & 0xFF);
*ptr++ = rr->rdata->u.srv.port.b[0];
*ptr++ = rr->rdata->u.srv.port.b[1];
return(putDomainNameAsLabels(msg, ptr, limit, &rr->rdata->u.srv.target));
case kDNSType_OPT: return putOptRData(ptr, limit, rr);
default: debugf("putRData: Warning! Writing unknown resource type %d as raw data", rr->rrtype);
case kDNSType_HINFO:
case kDNSType_TXT:
case kDNSType_TSIG: if (ptr + rr->rdlength > limit) return(mDNSNULL);
mDNSPlatformMemCopy(rr->rdata->u.data, ptr, rr->rdlength);
return(ptr + rr->rdlength);
}
}
mDNSexport mDNSu8 *PutResourceRecordTTLWithLimit(DNSMessage *const msg, mDNSu8 *ptr, mDNSu16 *count, ResourceRecord *rr, mDNSu32 ttl, const mDNSu8 *limit)
{
mDNSu8 *endofrdata;
mDNSu16 actualLength;
if (rr->RecordType == kDNSRecordTypeUnregistered)
{
LogMsg("PutResourceRecord ERROR! Attempt to put kDNSRecordTypeUnregistered %##s (%s)", rr->name.c, DNSTypeName(rr->rrtype));
return(ptr);
}
ptr = putDomainNameAsLabels(msg, ptr, limit, &rr->name);
if (!ptr || ptr + 10 >= limit) return(mDNSNULL); ptr[0] = (mDNSu8)(rr->rrtype >> 8);
ptr[1] = (mDNSu8)(rr->rrtype & 0xFF);
ptr[2] = (mDNSu8)(rr->rrclass >> 8);
ptr[3] = (mDNSu8)(rr->rrclass & 0xFF);
ptr[4] = (mDNSu8)((ttl >> 24) & 0xFF);
ptr[5] = (mDNSu8)((ttl >> 16) & 0xFF);
ptr[6] = (mDNSu8)((ttl >> 8) & 0xFF);
ptr[7] = (mDNSu8)( ttl & 0xFF);
endofrdata = putRData(msg, ptr+10, limit, rr);
if (!endofrdata) { verbosedebugf("Ran out of space in PutResourceRecord for %##s (%s)", rr->name.c, DNSTypeName(rr->rrtype)); return(mDNSNULL); }
actualLength = (mDNSu16)(endofrdata - ptr - 10);
ptr[8] = (mDNSu8)(actualLength >> 8);
ptr[9] = (mDNSu8)(actualLength & 0xFF);
if (count) (*count)++;
else LogMsg("PutResourceRecordTTL: ERROR: No target count to update for %##s (%s)", rr->name.c, DNSTypeName(rr->rrtype));
return(endofrdata);
}
mDNSexport mDNSu8 *PutResourceRecordCappedTTL(DNSMessage *const msg, mDNSu8 *ptr, mDNSu16 *count, ResourceRecord *rr, mDNSu32
maxttl)
{
if (maxttl > rr->rroriginalttl) maxttl = rr->rroriginalttl;
return(PutResourceRecordTTL(msg, ptr, count, rr, maxttl));
}
mDNSexport mDNSu8 *putEmptyResourceRecord(DNSMessage *const msg, mDNSu8 *ptr, const mDNSu8 *const limit,
mDNSu16 *count, const AuthRecord *rr)
{
ptr = putDomainNameAsLabels(msg, ptr, limit, &rr->resrec.name);
if (!ptr || ptr + 10 > limit) return(mDNSNULL); ptr[0] = (mDNSu8)(rr->resrec.rrtype >> 8); ptr[1] = (mDNSu8)(rr->resrec.rrtype & 0xFF);
ptr[2] = (mDNSu8)(rr->resrec.rrclass >> 8); ptr[3] = (mDNSu8)(rr->resrec.rrclass & 0xFF);
ptr[4] = ptr[5] = ptr[6] = ptr[7] = 0; ptr[8] = ptr[9] = 0; (*count)++;
return(ptr + 10);
}
mDNSexport mDNSu8 *putQuestion(DNSMessage *const msg, mDNSu8 *ptr, const mDNSu8 *const limit, const domainname *const name, mDNSu16 rrtype, mDNSu16 rrclass)
{
ptr = putDomainNameAsLabels(msg, ptr, limit, name);
if (!ptr || ptr+4 >= limit) return(mDNSNULL); ptr[0] = (mDNSu8)(rrtype >> 8);
ptr[1] = (mDNSu8)(rrtype & 0xFF);
ptr[2] = (mDNSu8)(rrclass >> 8);
ptr[3] = (mDNSu8)(rrclass & 0xFF);
msg->h.numQuestions++;
return(ptr+4);
}
mDNSexport mDNSu8 *putZone(DNSMessage *const msg, mDNSu8 *ptr, mDNSu8 *limit, const domainname *zone, mDNSOpaque16 zoneClass)
{
ptr = putDomainNameAsLabels(msg, ptr, limit, zone);
if (!ptr || ptr + 4 > limit) return mDNSNULL; *ptr++ = (mDNSu8)(kDNSType_SOA >> 8);
*ptr++ = (mDNSu8)(kDNSType_SOA & 0xFF);
*ptr++ = zoneClass.b[0];
*ptr++ = zoneClass.b[1];
msg->h.mDNS_numZones++;
return ptr;
}
mDNSexport mDNSu8 *putPrereqNameNotInUse(domainname *name, DNSMessage *msg, mDNSu8 *ptr, mDNSu8 *end)
{
AuthRecord prereq;
mDNSPlatformMemZero(&prereq, sizeof(AuthRecord));
AssignDomainName(prereq.resrec.name, *name);
prereq.resrec.rrtype = kDNSQType_ANY;
prereq.resrec.rrclass = kDNSClass_NONE;
ptr = putEmptyResourceRecord(msg, ptr, end, &msg->h.mDNS_numPrereqs, &prereq);
return ptr;
}
mDNSexport mDNSu8 *putDeletionRecord(DNSMessage *msg, mDNSu8 *ptr, ResourceRecord *rr)
{
mDNSu16 origclass;
origclass = rr->rrclass;
rr->rrclass = kDNSClass_NONE;
ptr = PutResourceRecordTTLJumbo(msg, ptr, &msg->h.mDNS_numUpdates, rr, 0);
rr->rrclass = origclass;
return ptr;
}
mDNSexport mDNSu8 *putDeleteAllRRSets(DNSMessage *msg, mDNSu8 *ptr, const domainname *name)
{
const mDNSu8 *limit = msg->data + AbsoluteMaxDNSMessageData;;
mDNSu16 class = kDNSQClass_ANY;
mDNSu16 rrtype = kDNSQType_ANY;
ptr = putDomainNameAsLabels(msg, ptr, limit, name);
if (!ptr || ptr + 10 >= limit) return mDNSNULL; ptr[0] = (mDNSu8)(rrtype >> 8);
ptr[1] = (mDNSu8)(rrtype & 0xFF);
ptr[2] = (mDNSu8)(class >> 8);
ptr[3] = (mDNSu8)(class & 0xFF);
ptr[4] = ptr[5] = ptr[6] = ptr[7] = 0; ptr[8] = ptr[9] = 0;
msg->h.mDNS_numUpdates++;
return ptr + 10;
}
mDNSexport mDNSu8 *putUpdateLease(DNSMessage *msg, mDNSu8 *end, mDNSu32 lease)
{
AuthRecord rr;
ResourceRecord *opt = &rr.resrec;
rdataOpt *optRD;
mDNSPlatformMemZero(&rr, sizeof(AuthRecord));
opt->rdata = &rr.rdatastorage;
opt->RecordType = kDNSRecordTypeKnownUnique; opt->rrtype = kDNSType_OPT;
opt->rdlength = LEASE_OPT_SIZE;
opt->rdestimate = LEASE_OPT_SIZE;
optRD = &rr.resrec.rdata->u.opt;
optRD->opt = kDNSOpt_Lease;
optRD->optlen = sizeof(mDNSs32);
optRD->OptData.lease = lease;
end = PutResourceRecordTTLJumbo(msg, end, &msg->h.numAdditionals, opt, 0);
if (!end) { LogMsg("ERROR: putUpdateLease - PutResourceRecordTTL"); return mDNSNULL; }
return end;
}
#if COMPILER_LIKES_PRAGMA_MARK
#pragma mark -
#pragma mark - DNS Message Parsing Functions
#endif
mDNSexport mDNSu32 DomainNameHashValue(const domainname *const name)
{
mDNSu32 sum = 0;
const mDNSu8 *c;
for (c = name->c; c[0] != 0 && c[1] != 0; c += 2)
{
sum += ((mDNSIsUpperCase(c[0]) ? c[0] + 'a' - 'A' : c[0]) << 8) |
(mDNSIsUpperCase(c[1]) ? c[1] + 'a' - 'A' : c[1]);
sum = (sum<<3) | (sum>>29);
}
if (c[0]) sum += ((mDNSIsUpperCase(c[0]) ? c[0] + 'a' - 'A' : c[0]) << 8);
return(sum);
}
mDNSexport void SetNewRData(ResourceRecord *const rr, RData *NewRData, mDNSu16 rdlength)
{
domainname *target;
if (NewRData)
{
rr->rdata = NewRData;
rr->rdlength = rdlength;
}
target = GetRRDomainNameTarget(rr);
rr->rdlength = GetRDLength(rr, mDNSfalse);
rr->rdestimate = GetRDLength(rr, mDNStrue);
rr->rdatahash = RDataHashValue(rr->rdlength, &rr->rdata->u);
rr->rdnamehash = target ? DomainNameHashValue(target) : 0;
}
mDNSexport const mDNSu8 *skipDomainName(const DNSMessage *const msg, const mDNSu8 *ptr, const mDNSu8 *const end)
{
mDNSu16 total = 0;
if (ptr < (mDNSu8*)msg || ptr >= end)
{ debugf("skipDomainName: Illegal ptr not within packet boundaries"); return(mDNSNULL); }
while (1) {
const mDNSu8 len = *ptr++; if (len == 0) return(ptr); switch (len & 0xC0)
{
case 0x00: if (ptr + len >= end) { debugf("skipDomainName: Malformed domain name (overruns packet end)"); return(mDNSNULL); }
if (total + 1 + len >= MAX_DOMAIN_NAME) { debugf("skipDomainName: Malformed domain name (more than 255 characters)"); return(mDNSNULL); }
ptr += len;
total += 1 + len;
break;
case 0x40: debugf("skipDomainName: Extended EDNS0 label types 0x%X not supported", len); return(mDNSNULL);
case 0x80: debugf("skipDomainName: Illegal label length 0x%X", len); return(mDNSNULL);
case 0xC0: return(ptr+1);
}
}
}
mDNSexport const mDNSu8 *getDomainName(const DNSMessage *const msg, const mDNSu8 *ptr, const mDNSu8 *const end,
domainname *const name)
{
const mDNSu8 *nextbyte = mDNSNULL; mDNSu8 *np = name->c; const mDNSu8 *const limit = np + MAX_DOMAIN_NAME;
if (ptr < (mDNSu8*)msg || ptr >= end)
{ debugf("getDomainName: Illegal ptr not within packet boundaries"); return(mDNSNULL); }
*np = 0;
while (1) {
const mDNSu8 len = *ptr++; if (len == 0) break; switch (len & 0xC0)
{
int i;
mDNSu16 offset;
case 0x00: if (ptr + len >= end) { debugf("getDomainName: Malformed domain name (overruns packet end)"); return(mDNSNULL); }
if (np + 1 + len >= limit) { debugf("getDomainName: Malformed domain name (more than 255 characters)"); return(mDNSNULL); }
*np++ = len;
for (i=0; i<len; i++) *np++ = *ptr++;
*np = 0; break;
case 0x40: debugf("getDomainName: Extended EDNS0 label types 0x%X not supported in name %##s", len, name->c);
return(mDNSNULL);
case 0x80: debugf("getDomainName: Illegal label length 0x%X in domain name %##s", len, name->c); return(mDNSNULL);
case 0xC0: offset = (mDNSu16)((((mDNSu16)(len & 0x3F)) << 8) | *ptr++);
if (!nextbyte) nextbyte = ptr; ptr = (mDNSu8 *)msg + offset;
if (ptr < (mDNSu8*)msg || ptr >= end)
{ debugf("getDomainName: Illegal compression pointer not within packet boundaries"); return(mDNSNULL); }
if (*ptr & 0xC0)
{ debugf("getDomainName: Compression pointer must point to real label"); return(mDNSNULL); }
break;
}
}
if (nextbyte) return(nextbyte);
else return(ptr);
}
mDNSexport const mDNSu8 *skipResourceRecord(const DNSMessage *msg, const mDNSu8 *ptr, const mDNSu8 *end)
{
mDNSu16 pktrdlength;
ptr = skipDomainName(msg, ptr, end);
if (!ptr) { debugf("skipResourceRecord: Malformed RR name"); return(mDNSNULL); }
if (ptr + 10 > end) { debugf("skipResourceRecord: Malformed RR -- no type/class/ttl/len!"); return(mDNSNULL); }
pktrdlength = (mDNSu16)((mDNSu16)ptr[8] << 8 | ptr[9]);
ptr += 10;
if (ptr + pktrdlength > end) { debugf("skipResourceRecord: RDATA exceeds end of packet"); return(mDNSNULL); }
return(ptr + pktrdlength);
}
mDNSexport const mDNSu8 *GetLargeResourceRecord(mDNS *const m, const DNSMessage * const msg, const mDNSu8 *ptr,
const mDNSu8 *end, const mDNSInterfaceID InterfaceID, mDNSu8 RecordType, LargeCacheRecord *largecr)
{
CacheRecord *rr = &largecr->r;
mDNSu16 pktrdlength;
if (largecr == &m->rec && rr->resrec.RecordType)
LogMsg("GetLargeResourceRecord: m->rec appears to be already in use");
rr->next = mDNSNULL;
rr->resrec.RecordType = RecordType;
rr->NextInKAList = mDNSNULL;
rr->TimeRcvd = m ? m->timenow : 0;
rr->DelayDelivery = 0;
rr->NextRequiredQuery = m ? m->timenow : 0; rr->LastUsed = m ? m->timenow : 0;
rr->CRActiveQuestion = mDNSNULL;
rr->UnansweredQueries = 0;
rr->LastUnansweredTime= 0;
rr->MPUnansweredQ = 0;
rr->MPLastUnansweredQT= 0;
rr->MPUnansweredKA = 0;
rr->MPExpectingKA = mDNSfalse;
rr->NextInCFList = mDNSNULL;
rr->resrec.InterfaceID = InterfaceID;
ptr = getDomainName(msg, ptr, end, &rr->resrec.name);
if (!ptr) { debugf("GetResourceRecord: Malformed RR name"); return(mDNSNULL); }
if (ptr + 10 > end) { debugf("GetResourceRecord: Malformed RR -- no type/class/ttl/len!"); return(mDNSNULL); }
rr->resrec.rrtype = (mDNSu16) ((mDNSu16)ptr[0] << 8 | ptr[1]);
rr->resrec.rrclass = (mDNSu16)(((mDNSu16)ptr[2] << 8 | ptr[3]) & kDNSClass_Mask);
rr->resrec.rroriginalttl = (mDNSu32) ((mDNSu32)ptr[4] << 24 | (mDNSu32)ptr[5] << 16 | (mDNSu32)ptr[6] << 8 | ptr[7]);
if (rr->resrec.rroriginalttl > 0x70000000UL / mDNSPlatformOneSecond && (mDNSs32)rr->resrec.rroriginalttl != -1)
rr->resrec.rroriginalttl = 0x70000000UL / mDNSPlatformOneSecond;
pktrdlength = (mDNSu16)((mDNSu16)ptr[8] << 8 | ptr[9]);
if (ptr[2] & (kDNSClass_UniqueRRSet >> 8))
rr->resrec.RecordType |= kDNSRecordTypePacketUniqueMask;
ptr += 10;
if (ptr + pktrdlength > end) { debugf("GetResourceRecord: RDATA exceeds end of packet"); return(mDNSNULL); }
end = ptr + pktrdlength;
rr->resrec.rdata = (RData*)&rr->rdatastorage;
rr->resrec.rdata->MaxRDLength = MaximumRDSize;
if (!RecordType) LogMsg("GetLargeResourceRecord: No RecordType for %##s", rr->resrec.name.c);
switch (rr->resrec.rrtype)
{
case kDNSType_A: rr->resrec.rdata->u.ipv4.b[0] = ptr[0];
rr->resrec.rdata->u.ipv4.b[1] = ptr[1];
rr->resrec.rdata->u.ipv4.b[2] = ptr[2];
rr->resrec.rdata->u.ipv4.b[3] = ptr[3];
break;
case kDNSType_CNAME: case kDNSType_NS:
case kDNSType_PTR: if (!getDomainName(msg, ptr, end, &rr->resrec.rdata->u.name))
{ debugf("GetResourceRecord: Malformed CNAME/PTR RDATA name"); return(mDNSNULL); }
break;
case kDNSType_NULL: case kDNSType_HINFO: case kDNSType_TXT: if (pktrdlength > rr->resrec.rdata->MaxRDLength)
{
debugf("GetResourceRecord: %s rdata size (%d) exceeds storage (%d)",
DNSTypeName(rr->resrec.rrtype), pktrdlength, rr->resrec.rdata->MaxRDLength);
return(mDNSNULL);
}
rr->resrec.rdlength = pktrdlength;
mDNSPlatformMemCopy(ptr, rr->resrec.rdata->u.data, pktrdlength);
break;
case kDNSType_AAAA: mDNSPlatformMemCopy(ptr, &rr->resrec.rdata->u.ipv6, sizeof(rr->resrec.rdata->u.ipv6));
break;
case kDNSType_SRV: rr->resrec.rdata->u.srv.priority = (mDNSu16)((mDNSu16)ptr[0] << 8 | ptr[1]);
rr->resrec.rdata->u.srv.weight = (mDNSu16)((mDNSu16)ptr[2] << 8 | ptr[3]);
rr->resrec.rdata->u.srv.port.b[0] = ptr[4];
rr->resrec.rdata->u.srv.port.b[1] = ptr[5];
if (!getDomainName(msg, ptr+6, end, &rr->resrec.rdata->u.srv.target))
{ debugf("GetResourceRecord: Malformed SRV RDATA name"); return(mDNSNULL); }
break;
case kDNSType_SOA: ptr = getDomainName(msg, ptr, end, &rr->resrec.rdata->u.soa.mname);
if (!ptr) { debugf("GetResourceRecord: Malformed SOA RDATA mname"); return mDNSNULL; }
ptr = getDomainName(msg, ptr, end, &rr->resrec.rdata->u.soa.rname);
if (!ptr) { debugf("GetResourceRecord: Malformed SOA RDATA rname"); return mDNSNULL; }
if (ptr + 0x14 != end) { debugf("GetResourceRecord: Malformed SOA RDATA"); return mDNSNULL; }
rr->resrec.rdata->u.soa.serial = (mDNSs32) ((mDNSs32)ptr[0x00] << 24 | (mDNSs32)ptr[0x01] << 16 | (mDNSs32)ptr[0x02] << 8 | ptr[0x03]);
rr->resrec.rdata->u.soa.refresh = (mDNSu32) ((mDNSu32)ptr[0x04] << 24 | (mDNSu32)ptr[0x05] << 16 | (mDNSu32)ptr[0x06] << 8 | ptr[0x07]);
rr->resrec.rdata->u.soa.retry = (mDNSu32) ((mDNSu32)ptr[0x08] << 24 | (mDNSu32)ptr[0x09] << 16 | (mDNSu32)ptr[0x0A] << 8 | ptr[0x0B]);
rr->resrec.rdata->u.soa.expire = (mDNSu32) ((mDNSu32)ptr[0x0C] << 24 | (mDNSu32)ptr[0x0D] << 16 | (mDNSu32)ptr[0x0E] << 8 | ptr[0x0F]);
rr->resrec.rdata->u.soa.min = (mDNSu32) ((mDNSu32)ptr[0x10] << 24 | (mDNSu32)ptr[0x11] << 16 | (mDNSu32)ptr[0x12] << 8 | ptr[0x13]);
break;
case kDNSType_OPT: getOptRdata(ptr, end, &rr->resrec, pktrdlength); break;
default: if (pktrdlength > rr->resrec.rdata->MaxRDLength)
{
debugf("GetResourceRecord: rdata %d (%s) size (%d) exceeds storage (%d)",
rr->resrec.rrtype, DNSTypeName(rr->resrec.rrtype), pktrdlength, rr->resrec.rdata->MaxRDLength);
return(mDNSNULL);
}
debugf("GetResourceRecord: Warning! Reading resource type %d (%s) as opaque data",
rr->resrec.rrtype, DNSTypeName(rr->resrec.rrtype));
rr->resrec.rdlength = pktrdlength;
mDNSPlatformMemCopy(ptr, rr->resrec.rdata->u.data, pktrdlength);
break;
}
rr->resrec.namehash = DomainNameHashValue(&rr->resrec.name);
SetNewRData(&rr->resrec, mDNSNULL, 0);
return(ptr + pktrdlength);
}
mDNSexport const mDNSu8 *skipQuestion(const DNSMessage *msg, const mDNSu8 *ptr, const mDNSu8 *end)
{
ptr = skipDomainName(msg, ptr, end);
if (!ptr) { debugf("skipQuestion: Malformed domain name in DNS question section"); return(mDNSNULL); }
if (ptr+4 > end) { debugf("skipQuestion: Malformed DNS question section -- no query type and class!"); return(mDNSNULL); }
return(ptr+4);
}
mDNSexport const mDNSu8 *getQuestion(const DNSMessage *msg, const mDNSu8 *ptr, const mDNSu8 *end, const mDNSInterfaceID InterfaceID,
DNSQuestion *question)
{
question->InterfaceID = InterfaceID;
ptr = getDomainName(msg, ptr, end, &question->qname);
if (!ptr) { debugf("Malformed domain name in DNS question section"); return(mDNSNULL); }
if (ptr+4 > end) { debugf("Malformed DNS question section -- no query type and class!"); return(mDNSNULL); }
question->qnamehash = DomainNameHashValue(&question->qname);
question->qtype = (mDNSu16)((mDNSu16)ptr[0] << 8 | ptr[1]); question->qclass = (mDNSu16)((mDNSu16)ptr[2] << 8 | ptr[3]); return(ptr+4);
}
mDNSexport const mDNSu8 *LocateAnswers(const DNSMessage *const msg, const mDNSu8 *const end)
{
int i;
const mDNSu8 *ptr = msg->data;
for (i = 0; i < msg->h.numQuestions && ptr; i++) ptr = skipQuestion(msg, ptr, end);
return(ptr);
}
mDNSexport const mDNSu8 *LocateAuthorities(const DNSMessage *const msg, const mDNSu8 *const end)
{
int i;
const mDNSu8 *ptr = LocateAnswers(msg, end);
for (i = 0; i < msg->h.numAnswers && ptr; i++) ptr = skipResourceRecord(msg, ptr, end);
return(ptr);
}
mDNSexport const mDNSu8 *LocateAdditionals(const DNSMessage *const msg, const mDNSu8 *const end)
{
int i;
const mDNSu8 *ptr = LocateAuthorities(msg, end);
for (i = 0; i < msg->h.numAuthorities; i++) ptr = skipResourceRecord(msg, ptr, end);
return (ptr);
}
#if COMPILER_LIKES_PRAGMA_MARK
#pragma mark -
#pragma mark -
#pragma mark - Packet Sending Functions
#endif
mDNSexport mStatus mDNSSendDNSMessage(const mDNS *const m, DNSMessage *const msg, mDNSu8 *end,
mDNSInterfaceID InterfaceID, const mDNSAddr *dst, mDNSIPPort dstport, int sd, uDNS_AuthInfo *authInfo)
{
mStatus status;
int nsent;
mDNSs32 msglen;
mDNSu8 lenbuf[2];
mDNSu16 numQuestions = msg->h.numQuestions;
mDNSu16 numAnswers = msg->h.numAnswers;
mDNSu16 numAuthorities = msg->h.numAuthorities;
mDNSu16 numAdditionals = msg->h.numAdditionals;
mDNSu8 *ptr = (mDNSu8 *)&msg->h.numQuestions;
*ptr++ = (mDNSu8)(numQuestions >> 8);
*ptr++ = (mDNSu8)(numQuestions & 0xFF);
*ptr++ = (mDNSu8)(numAnswers >> 8);
*ptr++ = (mDNSu8)(numAnswers & 0xFF);
*ptr++ = (mDNSu8)(numAuthorities >> 8);
*ptr++ = (mDNSu8)(numAuthorities & 0xFF);
*ptr++ = (mDNSu8)(numAdditionals >> 8);
*ptr++ = (mDNSu8)(numAdditionals & 0xFF);
if (authInfo)
{
end = DNSDigest_SignMessage(msg, &end, &numAdditionals, authInfo);
if (!end) return mStatus_UnknownErr;
}
if (sd >= 0)
{
msglen = (mDNSu16)(end - (mDNSu8 *)msg);
lenbuf[0] = (mDNSu8)(msglen >> 8); lenbuf[1] = (mDNSu8)(msglen & 0xFF);
nsent = mDNSPlatformWriteTCP(sd, (char*)lenbuf, 2);
if (nsent != 2) goto tcp_error;
nsent = mDNSPlatformWriteTCP(sd, (char *)msg, msglen);
if (nsent != msglen) goto tcp_error;
status = mStatus_NoError;
}
else
{
status = mDNSPlatformSendUDP(m, msg, end, InterfaceID, dst, dstport);
}
msg->h.numQuestions = numQuestions;
msg->h.numAnswers = numAnswers;
msg->h.numAuthorities = numAuthorities;
msg->h.numAdditionals = (mDNSu16)(authInfo ? numAdditionals - 1 : numAdditionals);
return(status);
tcp_error:
LogMsg("mDNSSendDNSMessage: error sending message over tcp");
return mStatus_UnknownErr;
}
#if COMPILER_LIKES_PRAGMA_MARK
#pragma mark -
#pragma mark - RR List Management & Task Management
#endif
mDNSexport void mDNS_Lock(mDNS *const m)
{
mDNSPlatformLock(m);
if (m->mDNS_busy != m->mDNS_reentrancy)
LogMsg("mDNS_Lock: Locking failure! mDNS_busy (%ld) != mDNS_reentrancy (%ld)", m->mDNS_busy, m->mDNS_reentrancy);
if (m->mDNS_busy == 0)
{
if (m->timenow)
LogMsg("mDNS_Lock: m->timenow already set (%ld/%ld)", m->timenow, mDNSPlatformRawTime() + m->timenow_adjust);
m->timenow = mDNSPlatformRawTime() + m->timenow_adjust;
if (m->timenow == 0) m->timenow = 1;
}
else if (m->timenow == 0)
{
LogMsg("mDNS_Lock: m->mDNS_busy is %ld but m->timenow not set", m->mDNS_busy);
m->timenow = mDNSPlatformRawTime() + m->timenow_adjust;
if (m->timenow == 0) m->timenow = 1;
}
if (m->timenow_last - m->timenow > 0)
{
m->timenow_adjust += m->timenow_last - m->timenow;
LogMsg("mDNSPlatformRawTime went backwards by %ld ticks; setting correction factor to %ld", m->timenow_last - m->timenow, m->timenow_adjust);
m->timenow = m->timenow_last;
}
m->timenow_last = m->timenow;
m->mDNS_busy++;
}
mDNSlocal mDNSs32 GetNextScheduledEvent(const mDNS *const m)
{
mDNSs32 e = m->timenow + 0x78000000;
if (m->mDNSPlatformStatus != mStatus_NoError || m->SleepState) return(e);
if (m->NewQuestions)
{
if (m->NewQuestions->DelayAnswering) e = m->NewQuestions->DelayAnswering;
else return(m->timenow);
}
if (m->NewLocalOnlyQuestions) return(m->timenow);
if (m->NewLocalOnlyRecords) return(m->timenow);
if (m->DiscardLocalOnlyRecords) return(m->timenow);
if (m->SuppressSending) return(m->SuppressSending);
#ifndef UNICAST_DISABLED
if (e - m->uDNS_info.nextevent > 0) e = m->uDNS_info.nextevent;
#endif
if (e - m->NextCacheCheck > 0) e = m->NextCacheCheck;
if (e - m->NextScheduledQuery > 0) e = m->NextScheduledQuery;
if (e - m->NextScheduledProbe > 0) e = m->NextScheduledProbe;
if (e - m->NextScheduledResponse > 0) e = m->NextScheduledResponse;
return(e);
}
mDNSexport void mDNS_Unlock(mDNS *const m)
{
m->mDNS_busy--;
if (m->mDNS_busy != m->mDNS_reentrancy)
LogMsg("mDNS_Unlock: Locking failure! mDNS_busy (%ld) != mDNS_reentrancy (%ld)", m->mDNS_busy, m->mDNS_reentrancy);
if (m->mDNS_busy == 0)
{
m->NextScheduledEvent = GetNextScheduledEvent(m);
if (m->timenow == 0) LogMsg("mDNS_Unlock: ERROR! m->timenow aready zero");
m->timenow = 0;
}
mDNSPlatformUnlock(m);
}