2019-02-14 10:09:11 +00:00
/*
* $ Id $
*
2020-02-20 15:35:16 +00:00
Copyright ( c ) 2016 - 2020 Chung , Hyung - Hwan . All rights reserved .
2019-02-14 10:09:11 +00:00
Redistribution and use in source and binary forms , with or without
modification , are permitted provided that the following conditions
are met :
1. Redistributions of source code must retain the above copyright
notice , this list of conditions and the following disclaimer .
2. 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 .
THIS SOFTWARE IS PROVIDED BY THE AUTHOR " AS IS " AND ANY EXPRESS OR
IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE IMPLIED WAfRRANTIES
OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED .
IN NO EVENT SHALL THE AUTHOR 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 .
*/
2020-05-25 08:04:30 +00:00
# include <mio-dns.h>
2019-02-14 10:09:11 +00:00
# include "mio-prv.h"
/* ----------------------------------------------------------------------- */
# define DN_AT_END(ptr) (ptr[0] == '\0' || (ptr[0] == '.' && ptr[1] == '\0'))
2020-02-24 16:39:29 +00:00
static mio_oow_t to_dn ( const mio_bch_t * str , mio_uint8_t * buf )
2019-02-14 10:09:11 +00:00
{
2020-02-24 16:39:29 +00:00
mio_uint8_t * bp = buf ;
2019-02-14 10:09:11 +00:00
/*MIO_ASSERT (MIO_SIZEOF(mio_uint8_t) == MIO_SIZEOF(mio_bch_t));*/
2019-02-16 18:22:32 +00:00
if ( str & & ! DN_AT_END ( str ) )
2019-02-14 10:09:11 +00:00
{
mio_uint8_t * lp ;
mio_oow_t len ;
const mio_bch_t * seg ;
const mio_bch_t * cur = str - 1 ;
do
{
2020-02-24 16:39:29 +00:00
lp = bp + + ;
2019-02-14 10:09:11 +00:00
seg = + + cur ;
while ( * cur ! = ' \0 ' & & * cur ! = ' . ' )
{
2020-02-24 16:39:29 +00:00
* bp + + = * cur ;
2019-02-14 10:09:11 +00:00
cur + + ;
}
len = cur - seg ;
if ( len < = 0 | | len > 63 ) return 0 ;
2020-02-24 16:39:29 +00:00
* lp = ( mio_uint8_t ) len ;
2019-02-14 10:09:11 +00:00
}
while ( ! DN_AT_END ( cur ) ) ;
}
2020-02-24 16:39:29 +00:00
* bp + + = 0 ;
2019-02-14 10:09:11 +00:00
/* the length includes the terminating zero. */
return bp - buf ;
}
2020-02-24 16:39:29 +00:00
static mio_oow_t to_dn_capa ( const mio_bch_t * str )
{
2020-02-24 17:14:21 +00:00
mio_oow_t capa = 0 ;
2020-02-24 16:39:29 +00:00
/*MIO_ASSERT (MIO_SIZEOF(mio_uint8_t) == MIO_SIZEOF(mio_bch_t));*/
if ( str & & ! DN_AT_END ( str ) )
{
mio_oow_t len ;
const mio_bch_t * seg ;
const mio_bch_t * cur = str - 1 ;
do
{
capa + + ;
seg = + + cur ;
while ( * cur ! = ' \0 ' & & * cur ! = ' . ' )
{
capa + + ;
cur + + ;
}
len = cur - seg ;
if ( len < = 0 | | len > 63 ) return 0 ;
}
while ( ! DN_AT_END ( cur ) ) ;
}
capa + + ;
/* the length includes the terminating zero. */
return capa ;
}
2019-02-14 10:09:11 +00:00
static mio_oow_t dn_length ( mio_uint8_t * ptr , mio_oow_t len )
{
mio_uint8_t * curptr ;
mio_oow_t curlen , seglen ;
curptr = ptr ;
curlen = len ;
do
{
if ( curlen < = 0 ) return 0 ;
seglen = * curptr + + ;
curlen = curlen - 1 ;
if ( seglen = = 0 ) break ;
else if ( seglen > curlen | | seglen > 63 ) return 0 ;
curptr + = seglen ;
curlen - = seglen ;
}
while ( 1 ) ;
return curptr - ptr ;
}
2020-02-11 15:51:41 +00:00
/* ----------------------------------------------------------------------- */
2020-02-13 09:31:02 +00:00
static int parse_domain_name ( mio_t * mio , mio_dns_pkt_info_t * pi )
2020-02-11 15:51:41 +00:00
{
2020-02-13 09:31:02 +00:00
mio_oow_t seglen ;
2020-02-12 10:06:05 +00:00
mio_uint8_t * xptr ;
2020-02-11 15:51:41 +00:00
2020-02-13 16:27:29 +00:00
if ( MIO_UNLIKELY ( pi - > _ptr > = pi - > _end ) ) goto oops ;
2020-02-12 10:06:05 +00:00
xptr = MIO_NULL ;
2020-02-13 16:27:29 +00:00
if ( ( seglen = * pi - > _ptr + + ) = = 0 )
2020-02-13 09:31:02 +00:00
{
2020-02-13 16:27:29 +00:00
if ( pi - > _rrdptr ) pi - > _rrdptr [ 0 ] = ' \0 ' ;
pi - > _rrdlen + + ; /* for a terminating null */
2020-02-13 09:31:02 +00:00
return 0 ;
}
do
2020-02-11 15:51:41 +00:00
{
2020-02-12 10:06:05 +00:00
if ( MIO_LIKELY ( seglen < 64 ) )
2020-02-11 15:51:41 +00:00
{
2020-02-12 10:06:05 +00:00
/* normal. 00XXXXXXXX */
normal :
2020-02-13 16:27:29 +00:00
if ( pi - > _rrdptr )
2020-02-13 09:31:02 +00:00
{
2020-02-13 16:27:29 +00:00
MIO_MEMCPY ( pi - > _rrdptr , pi - > _ptr , seglen ) ;
pi - > _rrdptr + = seglen + 1 ; /* +1 for '.' */
pi - > _rrdptr [ - 1 ] = ' . ' ;
2020-02-13 09:31:02 +00:00
}
2020-02-13 16:27:29 +00:00
pi - > _rrdlen + = seglen + 1 ; /* +1 for '.' */
pi - > _ptr + = seglen ;
if ( MIO_UNLIKELY ( pi - > _ptr > = pi - > _end ) ) goto oops ;
2020-02-11 15:51:41 +00:00
}
2020-02-12 10:06:05 +00:00
else if ( seglen > = 192 )
{
/* compressed. 11XXXXXXXX XXXXXXXX */
mio_oow_t offset ;
2020-02-11 15:51:41 +00:00
2020-02-13 16:27:29 +00:00
if ( MIO_UNLIKELY ( pi - > _ptr > = pi - > _end ) ) goto oops ;
offset = ( ( seglen & 0x3F ) < < 8 ) | * pi - > _ptr + + ;
2020-02-12 10:06:05 +00:00
2020-04-28 11:27:24 +00:00
/*if (MIO_UNLIKELY(pi->_ptr >= pi->_end)) goto oops; <- this condition can be true if the function is called for the domain name at the back of the last RR */
2020-02-13 16:27:29 +00:00
seglen = pi - > _start [ offset ] ;
2020-02-12 10:06:05 +00:00
if ( seglen > = 64 ) goto oops ; /* the pointed position must not contain another pointer */
2020-04-28 11:27:24 +00:00
if ( ! xptr ) xptr = pi - > _ptr ; /* some later parts can also be a pointer again. so xptr, once set, must not be set again */
2020-02-13 16:27:29 +00:00
pi - > _ptr = & pi - > _start [ offset + 1 ] ;
if ( MIO_UNLIKELY ( pi - > _ptr > = pi - > _end ) ) goto oops ;
2020-02-12 10:06:05 +00:00
goto normal ;
}
else if ( seglen > = 128 )
{
/* 128 to 191. 10XXXXXXXX */
goto oops ;
}
else
{
/* 64 to 127. 01XXXXXXXX */
goto oops ;
}
2020-02-11 15:51:41 +00:00
}
2020-02-13 16:27:29 +00:00
while ( ( seglen = * pi - > _ptr + + ) > 0 ) ;
2020-02-11 15:51:41 +00:00
2020-02-13 16:27:29 +00:00
if ( pi - > _rrdptr ) pi - > _rrdptr [ - 1 ] = ' \0 ' ;
2020-02-13 09:31:02 +00:00
2020-02-13 16:27:29 +00:00
if ( xptr ) pi - > _ptr = xptr ;
2020-02-13 09:31:02 +00:00
return 0 ;
2020-02-11 15:51:41 +00:00
2020-02-12 10:06:05 +00:00
oops :
2020-02-13 09:31:02 +00:00
mio_seterrnum ( mio , MIO_EINVAL ) ;
return - 1 ;
2020-02-11 15:51:41 +00:00
}
2020-02-14 10:06:29 +00:00
static int parse_question_rr ( mio_t * mio , mio_oow_t pos , mio_dns_pkt_info_t * pi )
2020-02-12 10:06:05 +00:00
{
mio_dns_qrtr_t * qrtr ;
2020-02-13 09:31:02 +00:00
mio_uint8_t * xrrdptr ;
2020-02-12 10:06:05 +00:00
2020-02-13 16:27:29 +00:00
xrrdptr = pi - > _rrdptr ;
2020-02-13 09:31:02 +00:00
if ( parse_domain_name ( mio , pi ) < = - 1 ) return - 1 ;
2020-02-12 10:06:05 +00:00
2020-02-13 16:27:29 +00:00
qrtr = ( mio_dns_qrtr_t * ) pi - > _ptr ;
2020-06-25 03:08:22 +00:00
if ( MIO_UNLIKELY ( pi - > _ptr > pi - > _end | | pi - > _end - pi - > _ptr < MIO_SIZEOF ( * qrtr ) ) ) goto oops ;
2020-02-13 16:27:29 +00:00
pi - > _ptr + = MIO_SIZEOF ( * qrtr ) ;
2020-06-25 03:08:22 +00:00
/*pi->_rrdlen += MIO_SIZEOF(*qrtr);*/
2020-02-12 10:06:05 +00:00
2020-02-13 16:27:29 +00:00
if ( pi - > _rrdptr )
2020-02-13 09:31:02 +00:00
{
mio_dns_bqr_t * bqr ;
bqr = pi - > rr . qd ;
bqr [ pos ] . qname = ( mio_bch_t * ) xrrdptr ;
bqr [ pos ] . qtype = mio_ntoh16 ( qrtr - > qtype ) ;
bqr [ pos ] . qclass = mio_ntoh16 ( qrtr - > qclass ) ;
}
return 0 ;
2020-02-12 10:06:05 +00:00
oops :
2020-02-13 09:31:02 +00:00
mio_seterrnum ( mio , MIO_EINVAL ) ;
return - 1 ;
2020-02-12 10:06:05 +00:00
}
2020-02-14 10:06:29 +00:00
static int parse_answer_rr ( mio_t * mio , mio_dns_rr_part_t rr_part , mio_oow_t pos , mio_dns_pkt_info_t * pi )
2020-02-11 15:51:41 +00:00
{
mio_dns_rrtr_t * rrtr ;
2020-02-12 15:23:36 +00:00
mio_uint16_t qtype , dlen ;
2020-02-17 14:54:21 +00:00
mio_uint8_t * xrrdptr , * xrrdptr2 ;
2020-02-11 15:51:41 +00:00
2020-02-13 16:27:29 +00:00
xrrdptr = pi - > _rrdptr ;
2020-02-13 09:31:02 +00:00
if ( parse_domain_name ( mio , pi ) < = - 1 ) return - 1 ;
2020-02-11 15:51:41 +00:00
2020-02-13 16:27:29 +00:00
rrtr = ( mio_dns_rrtr_t * ) pi - > _ptr ;
2020-06-25 03:08:22 +00:00
if ( MIO_UNLIKELY ( pi - > _ptr > pi - > _end | | pi - > _end - pi - > _ptr < MIO_SIZEOF ( * rrtr ) ) ) goto oops ;
2020-02-13 16:27:29 +00:00
pi - > _ptr + = MIO_SIZEOF ( * rrtr ) ;
2020-02-12 15:23:36 +00:00
dlen = mio_ntoh16 ( rrtr - > dlen ) ;
2020-06-25 03:08:22 +00:00
if ( MIO_UNLIKELY ( pi - > _ptr > pi - > _end | | pi - > _end - pi - > _ptr < dlen ) ) goto oops ;
2020-02-12 15:23:36 +00:00
2020-02-13 09:31:02 +00:00
qtype = mio_ntoh16 ( rrtr - > rrtype ) ;
2020-02-17 14:54:21 +00:00
xrrdptr2 = pi - > _rrdptr ;
2020-02-12 15:23:36 +00:00
switch ( qtype )
{
2020-02-13 09:31:02 +00:00
case MIO_DNS_RRT_OPT :
2020-02-12 15:23:36 +00:00
{
2020-06-30 16:11:39 +00:00
mio_uint16_t eopt_tot_len , eopt_len ;
mio_dns_eopt_t * eopt ;
2020-02-12 15:23:36 +00:00
/* RFC 6891
The extended RCODE and flags , which OPT stores in the RR Time to Live
( TTL ) field , are structured as follows :
+ - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - +
0 : | EXTENDED - RCODE | VERSION |
+ - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - +
2 : | DO | Z |
+ - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - + - - - +
2020-02-13 09:31:02 +00:00
2020-02-12 15:23:36 +00:00
EXTENDED - RCODE
Forms the upper 8 bits of extended 12 - bit RCODE ( together with the
4 bits defined in [ RFC1035 ] . Note that EXTENDED - RCODE value 0
indicates that an unextended RCODE is in use ( values 0 through
15 ) .
*/
2020-02-13 09:31:02 +00:00
/* TODO: do i need to check if rrname is <ROOT>? */
pi - > edns . exist = 1 ;
pi - > edns . uplen = mio_ntoh16 ( rrtr - > rrclass ) ;
pi - > hdr . rcode | = ( rrtr - > ttl > > 24 ) ;
pi - > edns . version = ( rrtr - > ttl > > 16 ) & 0xFF ;
pi - > edns . dnssecok = ( ( rrtr - > ttl & 0x8000 ) > > 15 ) ;
2020-02-12 15:23:36 +00:00
/*if ((rrtr->ttl & 0x7FFF) != 0) goto oops;*/ /* Z not 0 - ignore this for now */
2020-06-30 16:11:39 +00:00
eopt = ( mio_dns_eopt_t * ) ( rrtr + 1 ) ;
eopt_tot_len = dlen ;
while ( eopt_tot_len > 0 )
{
if ( eopt_tot_len < MIO_SIZEOF ( mio_dns_eopt_t ) ) goto oops ;
eopt_len = mio_ntoh16 ( eopt - > dlen ) ;
if ( eopt_tot_len - MIO_SIZEOF ( mio_dns_eopt_t ) < eopt_len ) goto oops ; /* wrong eopt length */
if ( eopt - > code = = MIO_CONST_HTON16 ( MIO_DNS_EOPT_COOKIE ) )
{
if ( eopt_len < 8 ) goto oops ; /* the client cookie must be 8 bytes */
/* TODO: dns cookies */
}
eopt_tot_len - = MIO_SIZEOF ( mio_dns_eopt_t ) + eopt_len ;
eopt = ( mio_dns_eopt_t * ) ( ( mio_uint8_t * ) eopt + MIO_SIZEOF ( mio_dns_eopt_t ) + eopt_len ) ;
}
goto verbatim ; /* keep the entire option data including cookies */
2020-02-12 15:23:36 +00:00
}
2020-02-13 09:31:02 +00:00
case MIO_DNS_RRT_A :
2020-02-25 07:09:20 +00:00
if ( MIO_UNLIKELY ( dlen ! = MIO_SIZEOF ( mio_ip4ad_t ) ) ) goto oops ;
2020-02-17 14:54:21 +00:00
goto verbatim ;
2020-02-12 15:23:36 +00:00
2020-02-13 09:31:02 +00:00
case MIO_DNS_RRT_AAAA :
2020-02-25 07:09:20 +00:00
if ( MIO_UNLIKELY ( dlen ! = MIO_SIZEOF ( mio_ip6ad_t ) ) ) goto oops ;
2020-02-17 14:54:21 +00:00
goto verbatim ;
2020-02-13 09:31:02 +00:00
2020-04-28 11:33:54 +00:00
/*case MIO_DNS_RRT_MB:
case MIO_DNS_RRT_MD :
case MIO_DNS_RRT_MF :
case MIO_DNS_RRT_MG :
case MIO_DNS_RRT_MR : */
2020-02-13 09:31:02 +00:00
case MIO_DNS_RRT_CNAME :
2020-02-17 14:54:21 +00:00
case MIO_DNS_RRT_NS :
2020-02-19 14:59:06 +00:00
case MIO_DNS_RRT_PTR :
2020-02-12 15:23:36 +00:00
{
2020-02-13 09:31:02 +00:00
# if !defined(MIO_BUILD_RELEASE)
2020-02-13 16:27:29 +00:00
mio_uint8_t * xptr = pi - > _ptr ;
2020-02-13 09:31:02 +00:00
# endif
2020-02-19 14:59:06 +00:00
if ( parse_domain_name ( mio , pi ) < = - 1 ) goto oops ;
2020-02-13 16:27:29 +00:00
MIO_ASSERT ( mio , pi - > _ptr = = xptr + dlen ) ;
2020-02-12 15:23:36 +00:00
break ;
}
2020-02-17 14:54:21 +00:00
2020-04-28 11:27:24 +00:00
case MIO_DNS_RRT_MX :
{
# if !defined(MIO_BUILD_RELEASE)
mio_uint8_t * xptr = pi - > _ptr ;
# endif
mio_dns_brrd_mx_t * mx ;
pi - > _rrdlen + = MIO_SIZEOF ( * mx ) ;
if ( MIO_UNLIKELY ( pi - > _end - pi - > _ptr < 2 ) ) goto oops ;
if ( pi - > _rrdptr )
{
2020-04-30 14:48:39 +00:00
mx = ( mio_dns_brrd_mx_t * ) pi - > _rrdptr ;
2020-04-28 11:27:24 +00:00
pi - > _rrdptr + = MIO_SIZEOF ( * mx ) ;
MIO_MEMCPY ( & mx - > preference , pi - > _ptr , 2 ) ; pi - > _ptr + = 2 ;
mx - > preference = mio_ntoh16 ( mx - > preference ) ;
mx - > exchange = ( mio_bch_t * ) pi - > _rrdptr ;
if ( parse_domain_name ( mio , pi ) < = - 1 ) goto oops ;
}
else
{
pi - > _ptr + = 2 ;
if ( parse_domain_name ( mio , pi ) < = - 1 ) goto oops ;
}
MIO_ASSERT ( mio , pi - > _ptr = = xptr + dlen ) ;
break ;
}
2020-02-17 14:54:21 +00:00
case MIO_DNS_RRT_SOA :
{
# if !defined(MIO_BUILD_RELEASE)
mio_uint8_t * xptr = pi - > _ptr ;
# endif
mio_dns_brrd_soa_t * soa ;
pi - > _rrdlen + = MIO_SIZEOF ( * soa ) ;
if ( pi - > _rrdptr )
{
soa = ( mio_dns_brrd_soa_t * ) pi - > _rrdptr ;
pi - > _rrdptr + = MIO_SIZEOF ( * soa ) ;
soa - > mname = ( mio_bch_t * ) pi - > _rrdptr ;
2020-02-19 14:59:06 +00:00
if ( parse_domain_name ( mio , pi ) < = - 1 ) goto oops ;
2020-02-17 14:54:21 +00:00
soa - > rname = ( mio_bch_t * ) pi - > _rrdptr ;
2020-02-19 14:59:06 +00:00
if ( parse_domain_name ( mio , pi ) < = - 1 ) goto oops ;
2020-02-17 14:54:21 +00:00
2020-04-28 10:27:41 +00:00
if ( MIO_UNLIKELY ( pi - > _end - pi - > _ptr < 20 ) ) goto oops ;
2020-02-17 14:54:21 +00:00
MIO_MEMCPY ( & soa - > serial , pi - > _ptr , 20 ) ;
soa - > serial = mio_ntoh32 ( soa - > serial ) ;
soa - > refresh = mio_ntoh32 ( soa - > refresh ) ;
soa - > retry = mio_ntoh32 ( soa - > retry ) ;
soa - > expire = mio_ntoh32 ( soa - > expire ) ;
soa - > minimum = mio_ntoh32 ( soa - > minimum ) ;
}
else
{
2020-02-19 14:59:06 +00:00
if ( parse_domain_name ( mio , pi ) < = - 1 ) goto oops ;
if ( parse_domain_name ( mio , pi ) < = - 1 ) goto oops ;
2020-04-28 10:27:41 +00:00
if ( MIO_UNLIKELY ( pi - > _end - pi - > _ptr < 20 ) ) goto oops ;
2020-02-17 14:54:21 +00:00
}
2020-02-19 15:42:29 +00:00
pi - > _ptr + = 20 ;
2020-02-17 14:54:21 +00:00
MIO_ASSERT ( mio , pi - > _ptr = = xptr + dlen ) ;
break ;
}
default :
verbatim :
pi - > _ptr + = dlen ;
pi - > _rrdlen + = dlen ;
if ( pi - > _rrdptr )
{
MIO_MEMCPY ( pi - > _rrdptr , rrtr + 1 , dlen ) ; /* copy actual data */
pi - > _rrdptr + = dlen ;
}
2020-02-12 15:23:36 +00:00
}
2020-02-11 15:51:41 +00:00
2020-02-17 14:54:21 +00:00
if ( pi - > _rrdptr )
2020-02-13 09:31:02 +00:00
{
2020-02-17 14:54:21 +00:00
/* store information about the actual record */
mio_dns_brr_t * brr ;
switch ( rr_part )
2020-02-14 10:06:29 +00:00
{
2020-02-17 14:54:21 +00:00
case MIO_DNS_RR_PART_ANSWER : brr = pi - > rr . an ; break ;
case MIO_DNS_RR_PART_AUTHORITY : brr = pi - > rr . ns ; break ;
case MIO_DNS_RR_PART_ADDITIONAL : brr = pi - > rr . ar ; break ;
2020-04-28 11:27:24 +00:00
default : goto oops ;
2020-02-14 10:06:29 +00:00
}
2020-02-17 14:54:21 +00:00
brr [ pos ] . part = rr_part ;
brr [ pos ] . rrname = ( mio_bch_t * ) xrrdptr ;
brr [ pos ] . rrtype = mio_ntoh16 ( rrtr - > rrtype ) ;
brr [ pos ] . rrclass = mio_ntoh16 ( rrtr - > rrclass ) ;
brr [ pos ] . ttl = mio_ntoh32 ( rrtr - > ttl ) ;
brr [ pos ] . dptr = xrrdptr2 ;
/* this length may be different from the length in the header as transformation is performed on some RR data.
* for a domain name , it ' s inclusive of the termining null . */
brr [ pos ] . dlen = pi - > _rrdptr - xrrdptr2 ;
2020-02-13 09:31:02 +00:00
}
return 0 ;
2020-02-11 15:51:41 +00:00
2020-02-12 10:06:05 +00:00
oops :
2020-02-13 09:31:02 +00:00
mio_seterrnum ( mio , MIO_EINVAL ) ;
return - 1 ;
2020-02-11 15:51:41 +00:00
}
2020-02-13 09:31:02 +00:00
mio_dns_pkt_info_t * mio_dns_make_packet_info ( mio_t * mio , const mio_dns_pkt_t * pkt , mio_oow_t len )
2020-02-11 15:51:41 +00:00
{
2020-02-13 09:31:02 +00:00
mio_uint16_t i ;
mio_dns_pkt_info_t pib , * pii ;
2020-02-11 15:51:41 +00:00
MIO_ASSERT ( mio , len > = MIO_SIZEOF ( * pkt ) ) ;
2020-02-13 09:31:02 +00:00
MIO_MEMSET ( & pib , 0 , MIO_SIZEOF ( pib ) ) ;
2020-06-25 03:08:22 +00:00
/* this is used a initial workspace and also indicates that it's the first run.
* at the second run , it is set to a dynamically allocated memory block large enough
* to hold actual data . */
pii = & pib ;
2020-02-13 09:31:02 +00:00
redo :
2020-02-13 16:27:29 +00:00
pii - > _start = ( mio_uint8_t * ) pkt ;
pii - > _end = ( mio_uint8_t * ) pkt + len ;
pii - > _ptr = ( mio_uint8_t * ) ( pkt + 1 ) ;
2020-02-13 09:31:02 +00:00
pii - > hdr . id = mio_ntoh16 ( pkt - > id ) ;
pii - > hdr . qr = pkt - > qr & 0x01 ;
pii - > hdr . opcode = pkt - > opcode & 0x0F ;
pii - > hdr . aa = pkt - > aa & 0x01 ;
pii - > hdr . tc = pkt - > tc & 0x01 ;
pii - > hdr . rd = pkt - > rd & 0x01 ;
pii - > hdr . ra = pkt - > ra & 0x01 ;
pii - > hdr . ad = pkt - > ad & 0x01 ;
pii - > hdr . cd = pkt - > cd & 0x01 ;
pii - > hdr . rcode = pkt - > rcode & 0x0F ;
pii - > qdcount = mio_ntoh16 ( pkt - > qdcount ) ;
pii - > ancount = mio_ntoh16 ( pkt - > ancount ) ;
pii - > nscount = mio_ntoh16 ( pkt - > nscount ) ;
pii - > arcount = mio_ntoh16 ( pkt - > arcount ) ;
for ( i = 0 ; i < pii - > qdcount ; i + + )
{
2020-02-14 10:06:29 +00:00
if ( parse_question_rr ( mio , i , pii ) < = - 1 ) goto oops ;
2020-02-13 09:31:02 +00:00
}
for ( i = 0 ; i < pii - > ancount ; i + + )
{
2020-02-14 10:06:29 +00:00
if ( parse_answer_rr ( mio , MIO_DNS_RR_PART_ANSWER , i , pii ) < = - 1 ) goto oops ;
2020-02-13 09:31:02 +00:00
}
for ( i = 0 ; i < pii - > nscount ; i + + )
{
2020-02-14 10:06:29 +00:00
if ( parse_answer_rr ( mio , MIO_DNS_RR_PART_AUTHORITY , i , pii ) < = - 1 ) goto oops ;
2020-02-13 09:31:02 +00:00
}
2020-02-11 15:51:41 +00:00
2020-02-13 09:31:02 +00:00
for ( i = 0 ; i < pii - > arcount ; i + + )
{
2020-02-14 10:06:29 +00:00
if ( parse_answer_rr ( mio , MIO_DNS_RR_PART_ADDITIONAL , i , pii ) < = - 1 ) goto oops ;
2020-02-13 09:31:02 +00:00
}
2020-02-11 15:51:41 +00:00
2020-02-13 09:31:02 +00:00
if ( pii = = & pib )
{
2020-06-25 03:08:22 +00:00
/* TODO: better buffer management... */
2020-02-13 16:27:29 +00:00
pii = ( mio_dns_pkt_info_t * ) mio_callocmem ( mio , MIO_SIZEOF ( * pii ) + ( MIO_SIZEOF ( mio_dns_bqr_t ) * pib . qdcount ) + ( MIO_SIZEOF ( mio_dns_brr_t ) * ( pib . ancount + pib . nscount + pib . arcount ) ) + pib . _rrdlen ) ;
2020-02-13 09:31:02 +00:00
if ( ! pii ) goto oops ;
pii - > rr . qd = ( mio_dns_bqr_t * ) ( & pii [ 1 ] ) ;
pii - > rr . an = ( mio_dns_brr_t * ) & pii - > rr . qd [ pib . qdcount ] ;
pii - > rr . ns = ( mio_dns_brr_t * ) & pii - > rr . an [ pib . ancount ] ;
pii - > rr . ar = ( mio_dns_brr_t * ) & pii - > rr . ns [ pib . nscount ] ;
2020-02-13 16:27:29 +00:00
pii - > _rrdptr = ( mio_uint8_t * ) & pii - > rr . ar [ pib . arcount ] ;
2020-06-25 03:08:22 +00:00
/* _rrdptr points to the beginning of memory where additional data will
* be held for some RRs . _rrdlen is the length of total additional data .
* the additional data refers to the data that is pointed to by the
* breakdown RRs ( mio_dns_bqr_t / mio_dns_brr_t ) but is not stored in them . */
2020-02-13 09:31:02 +00:00
goto redo ;
}
2020-02-11 15:51:41 +00:00
2020-02-13 09:31:02 +00:00
return pii ;
oops :
if ( pii & & pii ! = & pib ) mio_freemem ( mio , pii ) ;
return MIO_NULL ;
}
void mio_dns_free_packet_info ( mio_t * mio , mio_dns_pkt_info_t * pi )
{
/* TODO: better management */
2020-02-16 11:03:25 +00:00
mio_freemem ( mio , pi ) ;
}
/* ----------------------------------------------------------------------- */
2020-02-25 07:09:20 +00:00
static int encode_rrdata_in_dns_msg ( mio_t * mio , const mio_dns_brr_t * rr , mio_uint16_t * dxlen , void * dptr )
2020-02-16 11:03:25 +00:00
{
2020-03-02 04:58:22 +00:00
mio_oow_t xlen ; /* actual data length after encoding */
2020-02-24 16:39:29 +00:00
2020-02-16 11:03:25 +00:00
switch ( rr - > rrtype )
{
case MIO_DNS_RRT_A :
2020-03-02 04:58:22 +00:00
if ( MIO_UNLIKELY ( rr - > dlen ! = MIO_SIZEOF ( mio_ip4ad_t ) ) ) goto inval ;
2020-02-25 07:09:20 +00:00
goto verbatim ;
2020-02-16 11:03:25 +00:00
2020-02-25 07:09:20 +00:00
case MIO_DNS_RRT_AAAA :
2020-03-02 04:58:22 +00:00
if ( MIO_UNLIKELY ( rr - > dlen ! = MIO_SIZEOF ( mio_ip6ad_t ) ) ) goto inval ;
2020-02-25 07:09:20 +00:00
goto verbatim ;
2020-02-16 11:03:25 +00:00
2020-04-28 11:33:54 +00:00
/*case MIO_DNS_RRT_MB:
case MIO_DNS_RRT_MD :
case MIO_DNS_RRT_MF :
case MIO_DNS_RRT_MG :
case MIO_DNS_RRT_MR : */
2020-02-16 11:03:25 +00:00
case MIO_DNS_RRT_CNAME :
case MIO_DNS_RRT_NS :
case MIO_DNS_RRT_PTR :
/* just a normal domain name */
2020-02-24 16:39:29 +00:00
if ( dptr )
xlen = to_dn ( rr - > dptr , dptr ) ;
else
xlen = to_dn_capa ( rr - > dptr ) ;
2020-03-02 04:58:22 +00:00
if ( MIO_UNLIKELY ( xlen < = 0 ) ) goto inval ;
2020-02-16 11:03:25 +00:00
break ;
#if 0
case MIO_DNS_RRT_HINFO :
/* cpu, os */
break ;
# endif
#if 0
case MIO_DNS_RRT_MINFO :
/* rmailbx, emailbx */
# endif
2020-02-24 16:39:29 +00:00
xlen = rr - > dlen ;
2020-02-16 11:03:25 +00:00
break ;
2020-04-28 11:27:24 +00:00
case MIO_DNS_RRT_MX :
{
mio_dns_brrd_mx_t * mx ;
mio_oow_t tmp ;
if ( MIO_UNLIKELY ( rr - > dlen ! = MIO_SIZEOF ( mio_dns_brrd_mx_t ) ) ) goto inval ;
mx = ( mio_dns_brrd_mx_t * ) rr - > dptr ;
xlen = 0 ;
if ( dptr )
{
mio_uint16_t ti ;
ti = mio_hton16 ( mx - > preference ) ;
MIO_MEMCPY ( ( mio_uint8_t * ) dptr + xlen , & ti , MIO_SIZEOF ( ti ) ) ; xlen + = MIO_SIZEOF ( ti ) ;
tmp = to_dn ( mx - > exchange , ( mio_uint8_t * ) dptr + xlen ) ;
if ( MIO_UNLIKELY ( tmp < = 0 ) ) goto inval ;
xlen + = tmp ;
}
else
{
xlen + = 2 ;
tmp = to_dn_capa ( mx - > exchange ) ;
if ( MIO_UNLIKELY ( tmp < = 0 ) ) goto inval ;
xlen + = tmp ;
}
break ;
}
2020-02-16 11:03:25 +00:00
case MIO_DNS_RRT_SOA :
2020-03-02 04:58:22 +00:00
{
2020-02-16 11:03:25 +00:00
/* soa */
2020-03-02 04:58:22 +00:00
mio_dns_brrd_soa_t * soa ;
mio_oow_t tmp ;
if ( MIO_UNLIKELY ( rr - > dlen ! = MIO_SIZEOF ( mio_dns_brrd_soa_t ) ) ) goto inval ;
soa = ( mio_dns_brrd_soa_t * ) rr - > dptr ;
xlen = 0 ;
if ( dptr )
{
mio_uint32_t ti ;
tmp = to_dn ( soa - > mname , ( mio_uint8_t * ) dptr + xlen ) ;
if ( MIO_UNLIKELY ( tmp < = 0 ) ) goto inval ;
xlen + = tmp ;
tmp = to_dn ( soa - > rname , ( mio_uint8_t * ) dptr + xlen ) ;
if ( MIO_UNLIKELY ( tmp < = 0 ) ) goto inval ;
xlen + = tmp ;
2020-04-28 11:27:24 +00:00
ti = mio_hton32 ( soa - > serial ) ;
2020-03-02 04:58:22 +00:00
MIO_MEMCPY ( ( mio_uint8_t * ) dptr + xlen , & ti , MIO_SIZEOF ( ti ) ) ; xlen + = MIO_SIZEOF ( ti ) ;
2020-04-28 11:27:24 +00:00
ti = mio_hton32 ( soa - > refresh ) ;
2020-03-02 04:58:22 +00:00
MIO_MEMCPY ( ( mio_uint8_t * ) dptr + xlen , & ti , MIO_SIZEOF ( ti ) ) ; xlen + = MIO_SIZEOF ( ti ) ;
2020-04-28 11:27:24 +00:00
ti = mio_hton32 ( soa - > retry ) ;
2020-03-02 04:58:22 +00:00
MIO_MEMCPY ( ( mio_uint8_t * ) dptr + xlen , & ti , MIO_SIZEOF ( ti ) ) ; xlen + = MIO_SIZEOF ( ti ) ;
2020-04-28 11:27:24 +00:00
ti = mio_hton32 ( soa - > expire ) ;
2020-03-02 04:58:22 +00:00
MIO_MEMCPY ( ( mio_uint8_t * ) dptr + xlen , & ti , MIO_SIZEOF ( ti ) ) ; xlen + = MIO_SIZEOF ( ti ) ;
2020-04-28 11:27:24 +00:00
ti = mio_hton32 ( soa - > minimum ) ;
2020-03-02 04:58:22 +00:00
MIO_MEMCPY ( ( mio_uint8_t * ) dptr + xlen , & ti , MIO_SIZEOF ( ti ) ) ; xlen + = MIO_SIZEOF ( ti ) ;
}
else
{
tmp = to_dn_capa ( soa - > mname ) ;
if ( MIO_UNLIKELY ( tmp < = 0 ) ) goto inval ;
xlen + = tmp ;
tmp = to_dn_capa ( soa - > rname ) ;
if ( MIO_UNLIKELY ( tmp < = 0 ) ) goto inval ;
xlen + = tmp ;
xlen + = 20 ;
}
2020-02-16 11:03:25 +00:00
break ;
2020-03-02 04:58:22 +00:00
}
2020-02-16 11:03:25 +00:00
case MIO_DNS_RRT_TXT :
case MIO_DNS_RRT_NULL :
default :
2020-02-25 07:09:20 +00:00
verbatim :
2020-02-16 11:03:25 +00:00
/* TODO: custom transformator? */
2020-02-25 07:09:20 +00:00
if ( dptr ) MIO_MEMCPY ( dptr , rr - > dptr , rr - > dlen ) ;
2020-02-24 16:39:29 +00:00
xlen = rr - > dlen ;
break ;
2020-02-16 11:03:25 +00:00
}
2020-02-25 07:09:20 +00:00
if ( MIO_UNLIKELY ( xlen > MIO_TYPE_MAX ( mio_uint16_t ) ) ) goto inval ;
* dxlen = ( mio_uint16_t ) xlen ;
return 0 ;
inval :
mio_seterrnum ( mio , MIO_EINVAL ) ;
return - 1 ;
2020-02-16 11:03:25 +00:00
}
mio_dns_msg_t * mio_dns_make_msg ( mio_t * mio , mio_dns_bhdr_t * bhdr , mio_dns_bqr_t * qr , mio_oow_t qr_count , mio_dns_brr_t * rr , mio_oow_t rr_count , mio_dns_bedns_t * edns , mio_oow_t xtnsize )
{
mio_oow_t dnlen , msgbufsz , pktlen , i ;
mio_dns_msg_t * msg ;
mio_dns_pkt_t * pkt ;
mio_uint8_t * dn ;
mio_dns_qrtr_t * qrtr ;
mio_dns_rrtr_t * rrtr ;
int rr_sect ;
mio_uint32_t edns_dlen ;
pktlen = MIO_SIZEOF ( * pkt ) ;
for ( i = 0 ; i < qr_count ; i + + )
{
2020-02-25 07:09:20 +00:00
dnlen = to_dn_capa ( qr [ i ] . qname ) ;
if ( MIO_UNLIKELY ( dnlen < = 0 ) )
{
mio_seterrnum ( mio , MIO_EINVAL ) ;
return MIO_NULL ;
}
pktlen + = dnlen + MIO_SIZEOF ( * qrtr ) ;
2020-02-16 11:03:25 +00:00
}
for ( i = 0 ; i < rr_count ; i + + )
{
2020-02-25 07:09:20 +00:00
mio_uint16_t rrdata_len ;
dnlen = to_dn_capa ( rr [ i ] . rrname ) ;
if ( MIO_UNLIKELY ( dnlen < = 0 ) )
2020-02-24 16:39:29 +00:00
{
mio_seterrnum ( mio , MIO_EINVAL ) ;
return MIO_NULL ;
}
2020-02-25 07:09:20 +00:00
if ( MIO_UNLIKELY ( encode_rrdata_in_dns_msg ( mio , & rr [ i ] , & rrdata_len , MIO_NULL ) < = - 1 ) ) return MIO_NULL ;
pktlen + = dnlen + MIO_SIZEOF ( * rrtr ) + rrdata_len ;
2020-02-16 11:03:25 +00:00
}
edns_dlen = 0 ;
if ( edns )
{
mio_dns_beopt_t * beopt ;
pktlen + = 1 + MIO_SIZEOF ( * rrtr ) ; /* edns0 OPT RR - 1 for the root name */
beopt = edns - > beoptr ;
for ( i = 0 ; i < edns - > beonum ; i + + )
{
edns_dlen + = MIO_SIZEOF ( mio_dns_eopt_t ) + beopt - > dlen ;
2020-02-25 07:09:20 +00:00
if ( MIO_UNLIKELY ( edns_dlen > MIO_TYPE_MAX ( mio_uint16_t ) ) )
2020-02-16 11:03:25 +00:00
{
mio_seterrbfmt ( mio , MIO_EINVAL , " edns options too large " ) ;
return MIO_NULL ;
}
beopt + + ;
}
pktlen + = edns_dlen ;
}
else
{
2020-02-25 07:09:20 +00:00
if ( MIO_UNLIKELY ( bhdr - > rcode > 0x0F ) )
2020-02-16 11:03:25 +00:00
{
/* rcode is larger than 4 bits. but edns info is not provided */
mio_seterrbfmt ( mio , MIO_EINVAL , " rcode too large without edns - %d " , bhdr - > rcode ) ;
return MIO_NULL ;
}
}
msgbufsz = MIO_SIZEOF ( * msg ) + MIO_ALIGN_POW2 ( pktlen , MIO_SIZEOF_VOID_P ) + xtnsize ;
/* TODO: msg buffer reuse */
msg = mio_callocmem ( mio , msgbufsz ) ;
2020-07-01 15:54:23 +00:00
if ( MIO_UNLIKELY ( ! msg ) ) return MIO_NULL ;
2020-02-16 11:03:25 +00:00
msg - > msglen = msgbufsz ; /* record the instance size */
msg - > pktalilen = MIO_ALIGN_POW2 ( pktlen , MIO_SIZEOF_VOID_P ) ;
pkt = mio_dns_msg_to_pkt ( msg ) ; /* actual packet data begins after the message structure */
dn = ( mio_uint8_t * ) ( pkt + 1 ) ; /* skip the dns packet header */
for ( i = 0 ; i < qr_count ; i + + )
{
/* dnlen includes the ending <zero> */
2020-02-24 16:39:29 +00:00
dnlen = to_dn ( qr [ i ] . qname , dn ) ;
2020-02-25 07:09:20 +00:00
MIO_ASSERT ( mio , dnlen > 0 ) ;
2020-02-16 11:03:25 +00:00
qrtr = ( mio_dns_qrtr_t * ) ( dn + dnlen ) ;
qrtr - > qtype = mio_hton16 ( qr [ i ] . qtype ) ;
qrtr - > qclass = mio_hton16 ( qr [ i ] . qclass ) ;
dn = ( mio_uint8_t * ) ( qrtr + 1 ) ;
}
for ( rr_sect = MIO_DNS_RR_PART_ANSWER ; rr_sect < = MIO_DNS_RR_PART_ADDITIONAL ; )
{
mio_oow_t match_count = 0 ;
for ( i = 0 ; i < rr_count ; i + + )
{
if ( rr [ i ] . part = = rr_sect )
{
2020-02-25 07:09:20 +00:00
mio_uint16_t rrdata_len ;
2020-02-16 11:03:25 +00:00
2020-02-24 16:39:29 +00:00
dnlen = to_dn ( rr [ i ] . rrname , dn ) ;
2020-02-25 07:09:20 +00:00
MIO_ASSERT ( mio , dnlen > 0 ) ;
2020-02-16 11:03:25 +00:00
rrtr = ( mio_dns_rrtr_t * ) ( dn + dnlen ) ;
rrtr - > rrtype = mio_hton16 ( rr [ i ] . rrtype ) ;
rrtr - > rrclass = mio_hton16 ( rr [ i ] . rrclass ) ;
rrtr - > ttl = mio_hton32 ( rr [ i ] . ttl ) ;
2020-02-25 07:09:20 +00:00
encode_rrdata_in_dns_msg ( mio , & rr [ i ] , & rrdata_len , rrtr + 1 ) ; /* this must succeed */
2020-02-24 16:39:29 +00:00
rrtr - > dlen = mio_hton16 ( rrdata_len ) ;
dn = ( mio_uint8_t * ) ( rrtr + 1 ) + rrdata_len ;
2020-02-16 11:03:25 +00:00
match_count + + ;
}
}
rr_sect = rr_sect + 1 ;
( ( mio_dns_pkt_alt_t * ) pkt ) - > rrcount [ rr_sect ] = mio_hton16 ( match_count ) ;
}
if ( edns )
{
mio_dns_eopt_t * eopt ;
mio_dns_beopt_t * beopt ;
/* add EDNS0 OPT RR */
* dn = 0 ; /* root domain. as if to_dn("") is called */
rrtr = ( mio_dns_rrtr_t * ) ( dn + 1 ) ;
rrtr - > rrtype = MIO_CONST_HTON16 ( MIO_DNS_RRT_OPT ) ;
rrtr - > rrclass = mio_hton16 ( edns - > uplen ) ;
rrtr - > ttl = mio_hton32 ( MIO_DNS_EDNS_MAKE_TTL ( bhdr - > rcode , edns - > version , edns - > dnssecok ) ) ;
rrtr - > dlen = mio_hton16 ( ( mio_uint16_t ) edns_dlen ) ;
dn = ( mio_uint8_t * ) ( rrtr + 1 ) ;
beopt = edns - > beoptr ;
eopt = ( mio_dns_eopt_t * ) dn ;
2020-07-01 15:54:23 +00:00
msg - > ednsrrtroff = ( mio_uint8_t * ) rrtr - ( mio_uint8_t * ) pkt ;
2020-02-16 11:03:25 +00:00
for ( i = 0 ; i < edns - > beonum ; i + + )
{
eopt - > code = mio_hton16 ( beopt - > code ) ;
eopt - > dlen = mio_hton16 ( beopt - > dlen ) ;
MIO_MEMCPY ( + + eopt , beopt - > dptr , beopt - > dlen ) ;
eopt = ( mio_dns_eopt_t * ) ( ( mio_uint8_t * ) eopt + beopt - > dlen ) ;
beopt + + ;
}
pkt - > arcount = mio_hton16 ( ( mio_ntoh16 ( pkt - > arcount ) + 1 ) ) ;
dn + = edns_dlen ;
}
pkt - > qdcount = mio_hton16 ( qr_count ) ;
pkt - > id = mio_hton16 ( ( mio_uint16_t ) bhdr - > id ) ;
/*pkt->qr = (rr_count > 0);
pkt - > opcode = MIO_DNS_OPCODE_QUERY ; */
pkt - > qr = bhdr - > qr & 0x01 ;
pkt - > opcode = bhdr - > opcode & 0x0F ;
pkt - > aa = bhdr - > aa & 0x01 ;
pkt - > tc = bhdr - > tc & 0x01 ;
pkt - > rd = bhdr - > rd & 0x01 ;
pkt - > ra = bhdr - > ra & 0x01 ;
pkt - > ad = bhdr - > ad & 0x01 ;
pkt - > cd = bhdr - > cd & 0x01 ;
pkt - > rcode = bhdr - > rcode & 0x0F ;
msg - > pktlen = dn - ( mio_uint8_t * ) pkt ;
MIO_ASSERT ( mio , msg - > pktlen = = pktlen ) ;
MIO_ASSERT ( mio , msg - > pktalilen = = MIO_ALIGN_POW2 ( pktlen , MIO_SIZEOF_VOID_P ) ) ;
return msg ;
}
void mio_dns_free_msg ( mio_t * mio , mio_dns_msg_t * msg )
{
/* TODO: better management */
mio_freemem ( mio , msg ) ;
2020-02-11 15:51:41 +00:00
}