mirror of
https://github.com/monero-project/monero.git
synced 2024-11-06 00:17:37 +00:00
a85b5759f3
These files were pulled from the 1.6.3 release tarball. This new version builds against OpenSSL version 1.1 which will be the default in the new Debian Stable which is due to be released RealSoonNow (tm).
788 lines
17 KiB
C
788 lines
17 KiB
C
/*
|
|
* util/data/dname.h - domain name handling
|
|
*
|
|
* Copyright (c) 2007, NLnet Labs. All rights reserved.
|
|
*
|
|
* This software is open source.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* Redistributions of source code must retain the above copyright notice,
|
|
* this list of conditions and the following disclaimer.
|
|
*
|
|
* Redistributions in binary form must reproduce the above copyright notice,
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
* and/or other materials provided with the distribution.
|
|
*
|
|
* Neither the name of the NLNET LABS nor the names of its contributors may
|
|
* be used to endorse or promote products derived from this software without
|
|
* specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
* HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
|
|
* TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
|
|
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
/**
|
|
* \file
|
|
*
|
|
* This file contains domain name handling functions.
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include <ctype.h>
|
|
#include "util/data/dname.h"
|
|
#include "util/data/msgparse.h"
|
|
#include "util/log.h"
|
|
#include "util/storage/lookup3.h"
|
|
#include "sldns/sbuffer.h"
|
|
|
|
/* determine length of a dname in buffer, no compression pointers allowed */
|
|
size_t
|
|
query_dname_len(sldns_buffer* query)
|
|
{
|
|
size_t len = 0;
|
|
size_t labellen;
|
|
while(1) {
|
|
if(sldns_buffer_remaining(query) < 1)
|
|
return 0; /* parse error, need label len */
|
|
labellen = sldns_buffer_read_u8(query);
|
|
if(labellen&0xc0)
|
|
return 0; /* no compression allowed in queries */
|
|
len += labellen + 1;
|
|
if(len > LDNS_MAX_DOMAINLEN)
|
|
return 0; /* too long */
|
|
if(labellen == 0)
|
|
return len;
|
|
if(sldns_buffer_remaining(query) < labellen)
|
|
return 0; /* parse error, need content */
|
|
sldns_buffer_skip(query, (ssize_t)labellen);
|
|
}
|
|
}
|
|
|
|
size_t
|
|
dname_valid(uint8_t* dname, size_t maxlen)
|
|
{
|
|
size_t len = 0;
|
|
size_t labellen;
|
|
labellen = *dname++;
|
|
while(labellen) {
|
|
if(labellen&0xc0)
|
|
return 0; /* no compression ptrs allowed */
|
|
len += labellen + 1;
|
|
if(len >= LDNS_MAX_DOMAINLEN)
|
|
return 0; /* too long */
|
|
if(len > maxlen)
|
|
return 0; /* does not fit in memory allocation */
|
|
dname += labellen;
|
|
labellen = *dname++;
|
|
}
|
|
len += 1;
|
|
if(len > maxlen)
|
|
return 0; /* does not fit in memory allocation */
|
|
return len;
|
|
}
|
|
|
|
/** compare uncompressed, noncanonical, registers are hints for speed */
|
|
int
|
|
query_dname_compare(register uint8_t* d1, register uint8_t* d2)
|
|
{
|
|
register uint8_t lab1, lab2;
|
|
log_assert(d1 && d2);
|
|
lab1 = *d1++;
|
|
lab2 = *d2++;
|
|
while( lab1 != 0 || lab2 != 0 ) {
|
|
/* compare label length */
|
|
/* if one dname ends, it has labellength 0 */
|
|
if(lab1 != lab2) {
|
|
if(lab1 < lab2)
|
|
return -1;
|
|
return 1;
|
|
}
|
|
log_assert(lab1 == lab2 && lab1 != 0);
|
|
/* compare lowercased labels. */
|
|
while(lab1--) {
|
|
/* compare bytes first for speed */
|
|
if(*d1 != *d2 &&
|
|
tolower((unsigned char)*d1) != tolower((unsigned char)*d2)) {
|
|
if(tolower((unsigned char)*d1) < tolower((unsigned char)*d2))
|
|
return -1;
|
|
return 1;
|
|
}
|
|
d1++;
|
|
d2++;
|
|
}
|
|
/* next pair of labels. */
|
|
lab1 = *d1++;
|
|
lab2 = *d2++;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
query_dname_tolower(uint8_t* dname)
|
|
{
|
|
/* the dname is stored uncompressed */
|
|
uint8_t labellen;
|
|
labellen = *dname;
|
|
while(labellen) {
|
|
dname++;
|
|
while(labellen--) {
|
|
*dname = (uint8_t)tolower((unsigned char)*dname);
|
|
dname++;
|
|
}
|
|
labellen = *dname;
|
|
}
|
|
}
|
|
|
|
void
|
|
pkt_dname_tolower(sldns_buffer* pkt, uint8_t* dname)
|
|
{
|
|
uint8_t lablen;
|
|
int count = 0;
|
|
if(dname >= sldns_buffer_end(pkt))
|
|
return;
|
|
lablen = *dname++;
|
|
while(lablen) {
|
|
if(LABEL_IS_PTR(lablen)) {
|
|
if((size_t)PTR_OFFSET(lablen, *dname)
|
|
>= sldns_buffer_limit(pkt))
|
|
return;
|
|
dname = sldns_buffer_at(pkt, PTR_OFFSET(lablen, *dname));
|
|
lablen = *dname++;
|
|
if(count++ > MAX_COMPRESS_PTRS)
|
|
return;
|
|
continue;
|
|
}
|
|
if(dname+lablen >= sldns_buffer_end(pkt))
|
|
return;
|
|
while(lablen--) {
|
|
*dname = (uint8_t)tolower((unsigned char)*dname);
|
|
dname++;
|
|
}
|
|
if(dname >= sldns_buffer_end(pkt))
|
|
return;
|
|
lablen = *dname++;
|
|
}
|
|
}
|
|
|
|
|
|
size_t
|
|
pkt_dname_len(sldns_buffer* pkt)
|
|
{
|
|
size_t len = 0;
|
|
int ptrcount = 0;
|
|
uint8_t labellen;
|
|
size_t endpos = 0;
|
|
|
|
/* read dname and determine length */
|
|
/* check compression pointers, loops, out of bounds */
|
|
while(1) {
|
|
/* read next label */
|
|
if(sldns_buffer_remaining(pkt) < 1)
|
|
return 0;
|
|
labellen = sldns_buffer_read_u8(pkt);
|
|
if(LABEL_IS_PTR(labellen)) {
|
|
/* compression ptr */
|
|
uint16_t ptr;
|
|
if(sldns_buffer_remaining(pkt) < 1)
|
|
return 0;
|
|
ptr = PTR_OFFSET(labellen, sldns_buffer_read_u8(pkt));
|
|
if(ptrcount++ > MAX_COMPRESS_PTRS)
|
|
return 0; /* loop! */
|
|
if(sldns_buffer_limit(pkt) <= ptr)
|
|
return 0; /* out of bounds! */
|
|
if(!endpos)
|
|
endpos = sldns_buffer_position(pkt);
|
|
sldns_buffer_set_position(pkt, ptr);
|
|
} else {
|
|
/* label contents */
|
|
if(labellen > 0x3f)
|
|
return 0; /* label too long */
|
|
len += 1 + labellen;
|
|
if(len > LDNS_MAX_DOMAINLEN)
|
|
return 0;
|
|
if(labellen == 0) {
|
|
/* end of dname */
|
|
break;
|
|
}
|
|
if(sldns_buffer_remaining(pkt) < labellen)
|
|
return 0;
|
|
sldns_buffer_skip(pkt, (ssize_t)labellen);
|
|
}
|
|
}
|
|
if(endpos)
|
|
sldns_buffer_set_position(pkt, endpos);
|
|
|
|
return len;
|
|
}
|
|
|
|
int
|
|
dname_pkt_compare(sldns_buffer* pkt, uint8_t* d1, uint8_t* d2)
|
|
{
|
|
uint8_t len1, len2;
|
|
log_assert(pkt && d1 && d2);
|
|
len1 = *d1++;
|
|
len2 = *d2++;
|
|
while( len1 != 0 || len2 != 0 ) {
|
|
/* resolve ptrs */
|
|
if(LABEL_IS_PTR(len1)) {
|
|
d1 = sldns_buffer_at(pkt, PTR_OFFSET(len1, *d1));
|
|
len1 = *d1++;
|
|
continue;
|
|
}
|
|
if(LABEL_IS_PTR(len2)) {
|
|
d2 = sldns_buffer_at(pkt, PTR_OFFSET(len2, *d2));
|
|
len2 = *d2++;
|
|
continue;
|
|
}
|
|
/* check label length */
|
|
log_assert(len1 <= LDNS_MAX_LABELLEN);
|
|
log_assert(len2 <= LDNS_MAX_LABELLEN);
|
|
if(len1 != len2) {
|
|
if(len1 < len2) return -1;
|
|
return 1;
|
|
}
|
|
log_assert(len1 == len2 && len1 != 0);
|
|
/* compare labels */
|
|
while(len1--) {
|
|
if(tolower((unsigned char)*d1) != tolower((unsigned char)*d2)) {
|
|
if(tolower((unsigned char)*d1) < tolower((unsigned char)*d2))
|
|
return -1;
|
|
return 1;
|
|
}
|
|
d1++;
|
|
d2++;
|
|
}
|
|
len1 = *d1++;
|
|
len2 = *d2++;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
hashvalue_type
|
|
dname_query_hash(uint8_t* dname, hashvalue_type h)
|
|
{
|
|
uint8_t labuf[LDNS_MAX_LABELLEN+1];
|
|
uint8_t lablen;
|
|
int i;
|
|
|
|
/* preserve case of query, make hash label by label */
|
|
lablen = *dname++;
|
|
while(lablen) {
|
|
log_assert(lablen <= LDNS_MAX_LABELLEN);
|
|
labuf[0] = lablen;
|
|
i=0;
|
|
while(lablen--) {
|
|
labuf[++i] = (uint8_t)tolower((unsigned char)*dname);
|
|
dname++;
|
|
}
|
|
h = hashlittle(labuf, labuf[0] + 1, h);
|
|
lablen = *dname++;
|
|
}
|
|
|
|
return h;
|
|
}
|
|
|
|
hashvalue_type
|
|
dname_pkt_hash(sldns_buffer* pkt, uint8_t* dname, hashvalue_type h)
|
|
{
|
|
uint8_t labuf[LDNS_MAX_LABELLEN+1];
|
|
uint8_t lablen;
|
|
int i;
|
|
|
|
/* preserve case of query, make hash label by label */
|
|
lablen = *dname++;
|
|
while(lablen) {
|
|
if(LABEL_IS_PTR(lablen)) {
|
|
/* follow pointer */
|
|
dname = sldns_buffer_at(pkt, PTR_OFFSET(lablen, *dname));
|
|
lablen = *dname++;
|
|
continue;
|
|
}
|
|
log_assert(lablen <= LDNS_MAX_LABELLEN);
|
|
labuf[0] = lablen;
|
|
i=0;
|
|
while(lablen--) {
|
|
labuf[++i] = (uint8_t)tolower((unsigned char)*dname);
|
|
dname++;
|
|
}
|
|
h = hashlittle(labuf, labuf[0] + 1, h);
|
|
lablen = *dname++;
|
|
}
|
|
|
|
return h;
|
|
}
|
|
|
|
void dname_pkt_copy(sldns_buffer* pkt, uint8_t* to, uint8_t* dname)
|
|
{
|
|
/* copy over the dname and decompress it at the same time */
|
|
size_t len = 0;
|
|
uint8_t lablen;
|
|
lablen = *dname++;
|
|
while(lablen) {
|
|
if(LABEL_IS_PTR(lablen)) {
|
|
/* follow pointer */
|
|
dname = sldns_buffer_at(pkt, PTR_OFFSET(lablen, *dname));
|
|
lablen = *dname++;
|
|
continue;
|
|
}
|
|
log_assert(lablen <= LDNS_MAX_LABELLEN);
|
|
len += (size_t)lablen+1;
|
|
if(len >= LDNS_MAX_DOMAINLEN) {
|
|
*to = 0; /* end the result prematurely */
|
|
log_err("bad dname in dname_pkt_copy");
|
|
return;
|
|
}
|
|
*to++ = lablen;
|
|
memmove(to, dname, lablen);
|
|
dname += lablen;
|
|
to += lablen;
|
|
lablen = *dname++;
|
|
}
|
|
/* copy last \0 */
|
|
*to = 0;
|
|
}
|
|
|
|
void dname_print(FILE* out, struct sldns_buffer* pkt, uint8_t* dname)
|
|
{
|
|
uint8_t lablen;
|
|
if(!out) out = stdout;
|
|
if(!dname) return;
|
|
|
|
lablen = *dname++;
|
|
if(!lablen)
|
|
fputc('.', out);
|
|
while(lablen) {
|
|
if(LABEL_IS_PTR(lablen)) {
|
|
/* follow pointer */
|
|
if(!pkt) {
|
|
fputs("??compressionptr??", out);
|
|
return;
|
|
}
|
|
dname = sldns_buffer_at(pkt, PTR_OFFSET(lablen, *dname));
|
|
lablen = *dname++;
|
|
continue;
|
|
}
|
|
if(lablen > LDNS_MAX_LABELLEN) {
|
|
fputs("??extendedlabel??", out);
|
|
return;
|
|
}
|
|
while(lablen--)
|
|
fputc((int)*dname++, out);
|
|
fputc('.', out);
|
|
lablen = *dname++;
|
|
}
|
|
}
|
|
|
|
int
|
|
dname_count_labels(uint8_t* dname)
|
|
{
|
|
uint8_t lablen;
|
|
int labs = 1;
|
|
|
|
lablen = *dname++;
|
|
while(lablen) {
|
|
labs++;
|
|
dname += lablen;
|
|
lablen = *dname++;
|
|
}
|
|
return labs;
|
|
}
|
|
|
|
int
|
|
dname_count_size_labels(uint8_t* dname, size_t* size)
|
|
{
|
|
uint8_t lablen;
|
|
int labs = 1;
|
|
size_t sz = 1;
|
|
|
|
lablen = *dname++;
|
|
while(lablen) {
|
|
labs++;
|
|
sz += lablen+1;
|
|
dname += lablen;
|
|
lablen = *dname++;
|
|
}
|
|
*size = sz;
|
|
return labs;
|
|
}
|
|
|
|
/**
|
|
* Compare labels in memory, lowercase while comparing.
|
|
* @param p1: label 1
|
|
* @param p2: label 2
|
|
* @param len: number of bytes to compare.
|
|
* @return: 0, -1, +1 comparison result.
|
|
*/
|
|
static int
|
|
memlowercmp(uint8_t* p1, uint8_t* p2, uint8_t len)
|
|
{
|
|
while(len--) {
|
|
if(*p1 != *p2 && tolower((unsigned char)*p1) != tolower((unsigned char)*p2)) {
|
|
if(tolower((unsigned char)*p1) < tolower((unsigned char)*p2))
|
|
return -1;
|
|
return 1;
|
|
}
|
|
p1++;
|
|
p2++;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
dname_lab_cmp(uint8_t* d1, int labs1, uint8_t* d2, int labs2, int* mlabs)
|
|
{
|
|
uint8_t len1, len2;
|
|
int atlabel = labs1;
|
|
int lastmlabs;
|
|
int lastdiff = 0;
|
|
/* first skip so that we compare same label. */
|
|
if(labs1 > labs2) {
|
|
while(atlabel > labs2) {
|
|
len1 = *d1++;
|
|
d1 += len1;
|
|
atlabel--;
|
|
}
|
|
log_assert(atlabel == labs2);
|
|
} else if(labs1 < labs2) {
|
|
atlabel = labs2;
|
|
while(atlabel > labs1) {
|
|
len2 = *d2++;
|
|
d2 += len2;
|
|
atlabel--;
|
|
}
|
|
log_assert(atlabel == labs1);
|
|
}
|
|
lastmlabs = atlabel+1;
|
|
/* now at same label in d1 and d2, atlabel */
|
|
/* www.example.com. */
|
|
/* 4 3 2 1 atlabel number */
|
|
/* repeat until at root label (which is always the same) */
|
|
while(atlabel > 1) {
|
|
len1 = *d1++;
|
|
len2 = *d2++;
|
|
if(len1 != len2) {
|
|
log_assert(len1 != 0 && len2 != 0);
|
|
if(len1<len2)
|
|
lastdiff = -1;
|
|
else lastdiff = 1;
|
|
lastmlabs = atlabel;
|
|
d1 += len1;
|
|
d2 += len2;
|
|
} else {
|
|
/* memlowercmp is inlined here; or just like
|
|
* if((c=memlowercmp(d1, d2, len1)) != 0) {
|
|
* lastdiff = c;
|
|
* lastmlabs = atlabel; } apart from d1++,d2++ */
|
|
while(len1) {
|
|
if(*d1 != *d2 && tolower((unsigned char)*d1)
|
|
!= tolower((unsigned char)*d2)) {
|
|
if(tolower((unsigned char)*d1) <
|
|
tolower((unsigned char)*d2)) {
|
|
lastdiff = -1;
|
|
lastmlabs = atlabel;
|
|
d1 += len1;
|
|
d2 += len1;
|
|
break;
|
|
}
|
|
lastdiff = 1;
|
|
lastmlabs = atlabel;
|
|
d1 += len1;
|
|
d2 += len1;
|
|
break; /* out of memlowercmp */
|
|
}
|
|
d1++;
|
|
d2++;
|
|
len1--;
|
|
}
|
|
}
|
|
atlabel--;
|
|
}
|
|
/* last difference atlabel number, so number of labels matching,
|
|
* at the right side, is one less. */
|
|
*mlabs = lastmlabs-1;
|
|
if(lastdiff == 0) {
|
|
/* all labels compared were equal, check if one has more
|
|
* labels, so that example.com. > com. */
|
|
if(labs1 > labs2)
|
|
return 1;
|
|
else if(labs1 < labs2)
|
|
return -1;
|
|
}
|
|
return lastdiff;
|
|
}
|
|
|
|
int
|
|
dname_buffer_write(sldns_buffer* pkt, uint8_t* dname)
|
|
{
|
|
uint8_t lablen;
|
|
|
|
if(sldns_buffer_remaining(pkt) < 1)
|
|
return 0;
|
|
lablen = *dname++;
|
|
sldns_buffer_write_u8(pkt, lablen);
|
|
while(lablen) {
|
|
if(sldns_buffer_remaining(pkt) < (size_t)lablen+1)
|
|
return 0;
|
|
sldns_buffer_write(pkt, dname, lablen);
|
|
dname += lablen;
|
|
lablen = *dname++;
|
|
sldns_buffer_write_u8(pkt, lablen);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
void dname_str(uint8_t* dname, char* str)
|
|
{
|
|
size_t len = 0;
|
|
uint8_t lablen = 0;
|
|
char* s = str;
|
|
if(!dname || !*dname) {
|
|
*s++ = '.';
|
|
*s = 0;
|
|
return;
|
|
}
|
|
lablen = *dname++;
|
|
while(lablen) {
|
|
if(lablen > LDNS_MAX_LABELLEN) {
|
|
*s++ = '#';
|
|
*s = 0;
|
|
return;
|
|
}
|
|
len += lablen+1;
|
|
if(len >= LDNS_MAX_DOMAINLEN-1) {
|
|
*s++ = '&';
|
|
*s = 0;
|
|
return;
|
|
}
|
|
while(lablen--) {
|
|
if(isalnum((unsigned char)*dname)
|
|
|| *dname == '-' || *dname == '_'
|
|
|| *dname == '*')
|
|
*s++ = *(char*)dname++;
|
|
else {
|
|
*s++ = '?';
|
|
dname++;
|
|
}
|
|
}
|
|
*s++ = '.';
|
|
lablen = *dname++;
|
|
}
|
|
*s = 0;
|
|
}
|
|
|
|
int
|
|
dname_strict_subdomain(uint8_t* d1, int labs1, uint8_t* d2, int labs2)
|
|
{
|
|
int m;
|
|
/* check subdomain: d1: www.example.com. and d2: example.com. */
|
|
if(labs2 >= labs1)
|
|
return 0;
|
|
if(dname_lab_cmp(d1, labs1, d2, labs2, &m) > 0) {
|
|
/* subdomain if all labels match */
|
|
return (m == labs2);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
dname_strict_subdomain_c(uint8_t* d1, uint8_t* d2)
|
|
{
|
|
return dname_strict_subdomain(d1, dname_count_labels(d1), d2,
|
|
dname_count_labels(d2));
|
|
}
|
|
|
|
int
|
|
dname_subdomain_c(uint8_t* d1, uint8_t* d2)
|
|
{
|
|
int m;
|
|
/* check subdomain: d1: www.example.com. and d2: example.com. */
|
|
/* or d1: example.com. and d2: example.com. */
|
|
int labs1 = dname_count_labels(d1);
|
|
int labs2 = dname_count_labels(d2);
|
|
if(labs2 > labs1)
|
|
return 0;
|
|
if(dname_lab_cmp(d1, labs1, d2, labs2, &m) < 0) {
|
|
/* must have been example.com , www.example.com - wrong */
|
|
/* or otherwise different dnames */
|
|
return 0;
|
|
}
|
|
return (m == labs2);
|
|
}
|
|
|
|
int
|
|
dname_is_root(uint8_t* dname)
|
|
{
|
|
uint8_t len;
|
|
log_assert(dname);
|
|
len = dname[0];
|
|
log_assert(!LABEL_IS_PTR(len));
|
|
return (len == 0);
|
|
}
|
|
|
|
void
|
|
dname_remove_label(uint8_t** dname, size_t* len)
|
|
{
|
|
size_t lablen;
|
|
log_assert(dname && *dname && len);
|
|
lablen = (*dname)[0];
|
|
log_assert(!LABEL_IS_PTR(lablen));
|
|
log_assert(*len > lablen);
|
|
if(lablen == 0)
|
|
return; /* do not modify root label */
|
|
*len -= lablen+1;
|
|
*dname += lablen+1;
|
|
}
|
|
|
|
void
|
|
dname_remove_labels(uint8_t** dname, size_t* len, int n)
|
|
{
|
|
int i;
|
|
for(i=0; i<n; i++)
|
|
dname_remove_label(dname, len);
|
|
}
|
|
|
|
int
|
|
dname_signame_label_count(uint8_t* dname)
|
|
{
|
|
uint8_t lablen;
|
|
int count = 0;
|
|
if(!*dname)
|
|
return 0;
|
|
if(dname[0] == 1 && dname[1] == '*')
|
|
dname += 2;
|
|
lablen = dname[0];
|
|
while(lablen) {
|
|
count++;
|
|
dname += lablen;
|
|
dname += 1;
|
|
lablen = dname[0];
|
|
}
|
|
return count;
|
|
}
|
|
|
|
int
|
|
dname_is_wild(uint8_t* dname)
|
|
{
|
|
return (dname[0] == 1 && dname[1] == '*');
|
|
}
|
|
|
|
/**
|
|
* Compare labels in memory, lowercase while comparing.
|
|
* Returns canonical order for labels. If all is equal, the
|
|
* shortest is first.
|
|
*
|
|
* @param p1: label 1
|
|
* @param len1: length of label 1.
|
|
* @param p2: label 2
|
|
* @param len2: length of label 2.
|
|
* @return: 0, -1, +1 comparison result.
|
|
*/
|
|
static int
|
|
memcanoncmp(uint8_t* p1, uint8_t len1, uint8_t* p2, uint8_t len2)
|
|
{
|
|
uint8_t min = (len1<len2)?len1:len2;
|
|
int c = memlowercmp(p1, p2, min);
|
|
if(c != 0)
|
|
return c;
|
|
/* equal, see who is shortest */
|
|
if(len1 < len2)
|
|
return -1;
|
|
if(len1 > len2)
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
|
|
int
|
|
dname_canon_lab_cmp(uint8_t* d1, int labs1, uint8_t* d2, int labs2, int* mlabs)
|
|
{
|
|
/* like dname_lab_cmp, but with different label comparison,
|
|
* empty character sorts before \000.
|
|
* So ylyly is before z. */
|
|
uint8_t len1, len2;
|
|
int atlabel = labs1;
|
|
int lastmlabs;
|
|
int lastdiff = 0;
|
|
int c;
|
|
/* first skip so that we compare same label. */
|
|
if(labs1 > labs2) {
|
|
while(atlabel > labs2) {
|
|
len1 = *d1++;
|
|
d1 += len1;
|
|
atlabel--;
|
|
}
|
|
log_assert(atlabel == labs2);
|
|
} else if(labs1 < labs2) {
|
|
atlabel = labs2;
|
|
while(atlabel > labs1) {
|
|
len2 = *d2++;
|
|
d2 += len2;
|
|
atlabel--;
|
|
}
|
|
log_assert(atlabel == labs1);
|
|
}
|
|
lastmlabs = atlabel+1;
|
|
/* now at same label in d1 and d2, atlabel */
|
|
/* www.example.com. */
|
|
/* 4 3 2 1 atlabel number */
|
|
/* repeat until at root label (which is always the same) */
|
|
while(atlabel > 1) {
|
|
len1 = *d1++;
|
|
len2 = *d2++;
|
|
|
|
if((c=memcanoncmp(d1, len1, d2, len2)) != 0) {
|
|
if(c<0)
|
|
lastdiff = -1;
|
|
else lastdiff = 1;
|
|
lastmlabs = atlabel;
|
|
}
|
|
|
|
d1 += len1;
|
|
d2 += len2;
|
|
atlabel--;
|
|
}
|
|
/* last difference atlabel number, so number of labels matching,
|
|
* at the right side, is one less. */
|
|
*mlabs = lastmlabs-1;
|
|
if(lastdiff == 0) {
|
|
/* all labels compared were equal, check if one has more
|
|
* labels, so that example.com. > com. */
|
|
if(labs1 > labs2)
|
|
return 1;
|
|
else if(labs1 < labs2)
|
|
return -1;
|
|
}
|
|
return lastdiff;
|
|
}
|
|
|
|
int
|
|
dname_canonical_compare(uint8_t* d1, uint8_t* d2)
|
|
{
|
|
int labs1, labs2, m;
|
|
labs1 = dname_count_labels(d1);
|
|
labs2 = dname_count_labels(d2);
|
|
return dname_canon_lab_cmp(d1, labs1, d2, labs2, &m);
|
|
}
|
|
|
|
uint8_t* dname_get_shared_topdomain(uint8_t* d1, uint8_t* d2)
|
|
{
|
|
int labs1, labs2, m;
|
|
size_t len = LDNS_MAX_DOMAINLEN;
|
|
labs1 = dname_count_labels(d1);
|
|
labs2 = dname_count_labels(d2);
|
|
(void)dname_lab_cmp(d1, labs1, d2, labs2, &m);
|
|
dname_remove_labels(&d1, &len, labs1-m);
|
|
return d1;
|
|
}
|