#include "ares_private.h"
typedef
struct
{
char
*name;
size_t
name_len;
size_t
idx;
} ares_nameoffset_t;
static
void
ares_nameoffset_free(
void
*arg)
{
ares_nameoffset_t *off = arg;
if
(off == NULL) {
return
;
}
ares_free(off->name);
ares_free(off);
}
static
ares_status_t ares_nameoffset_create(ares_llist_t **list,
const
char
*name,
size_t
idx)
{
ares_status_t status;
ares_nameoffset_t *off = NULL;
if
(list == NULL || name == NULL || ares_strlen(name) == 0 ||
ares_strlen(name) > 255) {
return
ARES_EFORMERR;
}
if
(*list == NULL) {
*list = ares_llist_create(ares_nameoffset_free);
}
if
(*list == NULL) {
status = ARES_ENOMEM;
goto
fail;
}
off = ares_malloc_zero(
sizeof
(*off));
if
(off == NULL) {
return
ARES_ENOMEM;
}
off->name = ares_strdup(name);
off->name_len = ares_strlen(off->name);
off->idx = idx;
if
(ares_llist_insert_last(*list, off) == NULL) {
status = ARES_ENOMEM;
goto
fail;
}
return
ARES_SUCCESS;
fail:
ares_nameoffset_free(off);
return
status;
}
static
const
ares_nameoffset_t *ares_nameoffset_find(ares_llist_t *list,
const
char
*name)
{
size_t
name_len = ares_strlen(name);
ares_llist_node_t *node;
const
ares_nameoffset_t *longest_match = NULL;
if
(list == NULL || name == NULL || name_len == 0) {
return
NULL;
}
for
(node = ares_llist_node_first(list); node != NULL;
node = ares_llist_node_next(node)) {
const
ares_nameoffset_t *val = ares_llist_node_val(node);
size_t
prefix_len;
if
(val->name_len > name_len) {
continue
;
}
if
(longest_match != NULL && longest_match->name_len > val->name_len) {
continue
;
}
prefix_len = name_len - val->name_len;
if
(!ares_streq(val->name, name + prefix_len)) {
continue
;
}
if
(prefix_len != 0 && name[prefix_len - 1] !=
'.'
) {
continue
;
}
longest_match = val;
}
return
longest_match;
}
static
void
ares_dns_labels_free_cb(
void
*arg)
{
ares_buf_t **buf = arg;
if
(buf == NULL) {
return
;
}
ares_buf_destroy(*buf);
}
static
ares_buf_t *ares_dns_labels_add(ares_array_t *labels)
{
ares_buf_t **buf;
if
(labels == NULL) {
return
NULL;
}
if
(ares_array_insert_last((
void
**)&buf, labels) != ARES_SUCCESS) {
return
NULL;
}
*buf = ares_buf_create();
if
(*buf == NULL) {
ares_array_remove_last(labels);
return
NULL;
}
return
*buf;
}
static
ares_buf_t *ares_dns_labels_get_last(ares_array_t *labels)
{
ares_buf_t **buf = ares_array_last(labels);
if
(buf == NULL) {
return
NULL;
}
return
*buf;
}
static
ares_buf_t *ares_dns_labels_get_at(ares_array_t *labels,
size_t
idx)
{
ares_buf_t **buf = ares_array_at(labels, idx);
if
(buf == NULL) {
return
NULL;
}
return
*buf;
}
static
void
ares_dns_name_labels_del_last(ares_array_t *labels)
{
ares_array_remove_last(labels);
}
static
ares_status_t ares_parse_dns_name_escape(ares_buf_t *namebuf,
ares_buf_t *label,
ares_bool_t validate_hostname)
{
ares_status_t status;
unsigned
char
c;
status = ares_buf_fetch_bytes(namebuf, &c, 1);
if
(status != ARES_SUCCESS) {
return
ARES_EBADNAME;
}
if
(ares_isdigit(c)) {
size_t
i;
unsigned
int
val = 0;
val = c -
'0'
;
for
(i = 0; i < 2; i++) {
status = ares_buf_fetch_bytes(namebuf, &c, 1);
if
(status != ARES_SUCCESS) {
return
ARES_EBADNAME;
}
if
(!ares_isdigit(c)) {
return
ARES_EBADNAME;
}
val *= 10;
val += c -
'0'
;
}
if
(val > 255) {
return
ARES_EBADNAME;
}
if
(validate_hostname && !ares_is_hostnamech((unsigned
char
)val)) {
return
ARES_EBADNAME;
}
return
ares_buf_append_byte(label, (unsigned
char
)val);
}
if
(validate_hostname && !ares_is_hostnamech(c)) {
return
ARES_EBADNAME;
}
return
ares_buf_append_byte(label, c);
}
static
ares_status_t ares_split_dns_name(ares_array_t *labels,
ares_bool_t validate_hostname,
const
char
*name)
{
ares_status_t status;
ares_buf_t *label = NULL;
ares_buf_t *namebuf = NULL;
size_t
i;
size_t
total_len = 0;
unsigned
char
c;
if
(name == NULL || labels == NULL) {
return
ARES_EFORMERR;
}
namebuf = ares_buf_create();
if
(namebuf == NULL) {
status = ARES_ENOMEM;
goto
done;
}
if
(*name !=
'\0'
) {
status =
ares_buf_append(namebuf, (
const
unsigned
char
*)name, ares_strlen(name));
if
(status != ARES_SUCCESS) {
goto
done;
}
}
label = ares_dns_labels_add(labels);
if
(label == NULL) {
status = ARES_ENOMEM;
goto
done;
}
while
(ares_buf_fetch_bytes(namebuf, &c, 1) == ARES_SUCCESS) {
if
(c ==
'.'
) {
label = ares_dns_labels_add(labels);
if
(label == NULL) {
status = ARES_ENOMEM;
goto
done;
}
continue
;
}
if
(c ==
'\\'
) {
status = ares_parse_dns_name_escape(namebuf, label, validate_hostname);
if
(status != ARES_SUCCESS) {
goto
done;
}
continue
;
}
if
(validate_hostname && !ares_is_hostnamech(c)) {
status = ARES_EBADNAME;
goto
done;
}
status = ares_buf_append_byte(label, c);
if
(status != ARES_SUCCESS) {
goto
done;
}
}
if
(ares_buf_len(ares_dns_labels_get_last(labels)) == 0) {
ares_dns_name_labels_del_last(labels);
}
if
(ares_array_len(labels) == 1 &&
ares_buf_len(ares_dns_labels_get_last(labels)) == 0) {
ares_dns_name_labels_del_last(labels);
}
for
(i = 0; i < ares_array_len(labels); i++) {
const
ares_buf_t *buf = ares_dns_labels_get_at(labels, i);
size_t
len = ares_buf_len(buf);
if
(len == 0 || len > 63) {
status = ARES_EBADNAME;
goto
done;
}
total_len += len;
}
if
(ares_array_len(labels) && total_len + ares_array_len(labels) - 1 > 255) {
status = ARES_EBADNAME;
goto
done;
}
status = ARES_SUCCESS;
done:
ares_buf_destroy(namebuf);
return
status;
}
ares_status_t ares_dns_name_write(ares_buf_t *buf, ares_llist_t **list,
ares_bool_t validate_hostname,
const
char
*name)
{
const
ares_nameoffset_t *off = NULL;
size_t
name_len;
size_t
orig_name_len;
size_t
pos = ares_buf_len(buf);
ares_array_t *labels = NULL;
char
name_copy[512];
ares_status_t status;
if
(buf == NULL || name == NULL) {
return
ARES_EFORMERR;
}
labels = ares_array_create(
sizeof
(ares_buf_t *), ares_dns_labels_free_cb);
if
(labels == NULL) {
return
ARES_ENOMEM;
}
name_len = ares_strcpy(name_copy, name,
sizeof
(name_copy));
orig_name_len = name_len;
if
(list != NULL) {
off = ares_nameoffset_find(*list, name_copy);
if
(off != NULL && off->name_len != name_len) {
name_len -= (off->name_len + 1);
name_copy[name_len] = 0;
}
}
if
(off == NULL || off->name_len != orig_name_len) {
size_t
i;
status = ares_split_dns_name(labels, validate_hostname, name_copy);
if
(status != ARES_SUCCESS) {
goto
done;
}
for
(i = 0; i < ares_array_len(labels); i++) {
size_t
len = 0;
const
ares_buf_t *lbuf = ares_dns_labels_get_at(labels, i);
const
unsigned
char
*ptr = ares_buf_peek(lbuf, &len);
status = ares_buf_append_byte(buf, (unsigned
char
)(len & 0xFF));
if
(status != ARES_SUCCESS) {
goto
done;
}
status = ares_buf_append(buf, ptr, len);
if
(status != ARES_SUCCESS) {
goto
done;
}
}
if
(off == NULL) {
status = ares_buf_append_byte(buf, 0);
if
(status != ARES_SUCCESS) {
goto
done;
}
}
}
if
(off != NULL) {
unsigned
short
u16 =
(unsigned
short
)0xC000 | (unsigned
short
)(off->idx & 0x3FFF);
status = ares_buf_append_be16(buf, u16);
if
(status != ARES_SUCCESS) {
goto
done;
}
}
if
(list != NULL && (off == NULL || off->name_len != orig_name_len) &&
name_len > 0) {
status = ares_nameoffset_create(list, name
, pos);
if
(status != ARES_SUCCESS) {
goto
done;
}
}
status = ARES_SUCCESS;
done:
ares_array_destroy(labels);
return
status;
}
static
ares_bool_t is_reservedch(
int
ch)
{
switch
(ch) {
case
'"'
:
case
'.'
:
case
';'
:
case
'\\'
:
case
'('
:
case
')'
:
case
'@'
:
case
'$'
:
return
ARES_TRUE;
default
:
break
;
}
return
ARES_FALSE;
}
static
ares_status_t ares_fetch_dnsname_into_buf(ares_buf_t *buf,
ares_buf_t *dest,
size_t
len,
ares_bool_t is_hostname)
{
size_t
remaining_len;
const
unsigned
char
*ptr = ares_buf_peek(buf, &remaining_len);
ares_status_t status;
size_t
i;
if
(buf == NULL || len == 0 || remaining_len < len) {
return
ARES_EBADRESP;
}
for
(i = 0; i < len; i++) {
unsigned
char
c = ptr[i];
if
(is_hostname && !ares_is_hostnamech(c)) {
status = ARES_EBADRESP;
goto
fail;
}
if
(dest == NULL) {
continue
;
}
if
(!ares_isprint(c)) {
unsigned
char
escape[4];
escape[0] =
'\\'
;
escape[1] =
'0'
+ (c / 100);
escape[2] =
'0'
+ ((c % 100) / 10);
escape[3] =
'0'
+ (c % 10);
status = ares_buf_append(dest, escape,
sizeof
(escape));
if
(status != ARES_SUCCESS) {
goto
fail;
}
continue
;
}
if
(is_reservedch(c)) {
status = ares_buf_append_byte(dest,
'\\'
);
if
(status != ARES_SUCCESS) {
goto
fail;
}
}
status = ares_buf_append_byte(dest, c);
if
(status != ARES_SUCCESS) {
return
status;
}
}
return
ares_buf_consume(buf, len);
fail:
return
status;
}
ares_status_t ares_dns_name_parse(ares_buf_t *buf,
char
**name,
ares_bool_t is_hostname)
{
size_t
save_offset = 0;
unsigned
char
c;
ares_status_t status;
ares_buf_t *namebuf = NULL;
size_t
label_start = ares_buf_get_position(buf);
if
(buf == NULL) {
return
ARES_EFORMERR;
}
if
(name != NULL) {
namebuf = ares_buf_create();
if
(namebuf == NULL) {
status = ARES_ENOMEM;
goto
fail;
}
}
while
(1) {
if
(label_start > ares_buf_get_position(buf)) {
label_start = ares_buf_get_position(buf);
}
status = ares_buf_fetch_bytes(buf, &c, 1);
if
(status != ARES_SUCCESS) {
goto
fail;
}
if
((c & 0xc0) == 0xc0) {
size_t
offset = (
size_t
)((c & 0x3F) << 8);
status = ares_buf_fetch_bytes(buf, &c, 1);
if
(status != ARES_SUCCESS) {
goto
fail;
}
offset |= ((
size_t
)c);
if
(offset >= label_start) {
status = ARES_EBADNAME;
goto
fail;
}
if
(save_offset == 0) {
save_offset = ares_buf_get_position(buf);
}
status = ares_buf_set_position(buf, offset);
if
(status != ARES_SUCCESS) {
status = ARES_EBADNAME;
goto
fail;
}
continue
;
}
else
if
((c & 0xc0) != 0) {
status = ARES_EBADNAME;
goto
fail;
}
else
if
(c == 0) {
break
;
}
if
(ares_buf_len(namebuf) != 0 && name != NULL) {
status = ares_buf_append_byte(namebuf,
'.'
);
if
(status != ARES_SUCCESS) {
goto
fail;
}
}
status = ares_fetch_dnsname_into_buf(buf, namebuf, c, is_hostname);
if
(status != ARES_SUCCESS) {
goto
fail;
}
}
if
(save_offset) {
ares_buf_set_position(buf, save_offset);
}
if
(name != NULL) {
*name = ares_buf_finish_str(namebuf, NULL);
if
(*name == NULL) {
status = ARES_ENOMEM;
goto
fail;
}
}
return
ARES_SUCCESS;
fail:
if
(status == ARES_EBADRESP) {
status = ARES_EBADNAME;
}
ares_buf_destroy(namebuf);
return
status;
}