hammer/examples/dns.c

137 lines
3.6 KiB
C
Raw Normal View History

#include "../src/hammer.h"
#define false 0
#define true 1
2012-05-22 00:57:33 +02:00
bool is_zero(parse_result_t *p) {
if (TT_UINT != p->ast->token_type)
return false;
return (0 == p->ast->uint);
}
2012-05-22 00:57:33 +02:00
/**
* A label can't be more than 63 characters.
*/
bool validate_label(parse_result_t *p) {
if (TT_SEQUENCE != p->ast->token_type)
return false;
return (64 > p->ast->seq->used);
}
/**
* Every DNS message should have QDCOUNT entries in the question
* section, and ANCOUNT+NSCOUNT+ARCOUNT resource records.
*
*/
bool validate_dns(parse_result_t *p) {
if (TT_SEQUENCE != p->ast->token_type)
return false;
// The header holds the counts as its last 4 elements.
parsed_token_t *header = p->ast->seq->elements[0];
size_t qd = header->seq->elements[8]->uint;
size_t an = header->seq->elements[9]->uint;
size_t ns = header->seq->elements[10]->uint;
size_t ar = header->seq->elements[11]->uint;
parsed_token_t *questions = p->ast->seq->elements[1];
if (questions->seq->used != qd)
return false;
parsed_token_t *rrs = p->ast->seq->elements[2];
if (an+ns+ar != rrs->seq->used)
return false;
return true;
2012-05-22 00:57:33 +02:00
}
parser_t* init_parser() {
static parser_t *dns_message = NULL;
if (dns_message)
return dns_message;
2012-05-22 00:57:33 +02:00
const parser_t *dns_header = sequence(bits(16, false), // ID
bits(1, false), // QR
bits(4, false), // opcode
bits(1, false), // AA
bits(1, false), // TC
bits(1, false), // RD
bits(1, false), // RA
ignore(attr_bool(bits(3, false), is_zero)), // Z
bits(4, false), // RCODE
2012-05-22 00:57:33 +02:00
uint16(), // QDCOUNT
uint16(), // ANCOUNT
uint16(), // NSCOUNT
uint16(), // ARCOUNT
NULL);
const parser_t *type = int_range(uint16(), 1, 16);
const parser_t *qtype = choice(type,
int_range(uint16(), 252, 255),
NULL);
const parser_t *class = int_range(uint16(), 1, 4);
const parser_t *qclass = choice(class,
int_range(uint16(), 255, 255),
NULL);
const parser_t *dns_question = sequence(length_value(uint8(), uint8()), // QNAME
qtype, // QTYPE
qclass, // QCLASS
NULL);
const parser_t *letter = choice(ch_range('a', 'z'),
ch_range('A', 'Z'),
NULL);
const parser_t *let_dig = choice(letter,
ch_range('0', '9'),
NULL);
const parser_t *ldh_str = many1(choice(let_dig,
ch('-'),
NULL));
const parser_t *label = attr_bool(sequence(letter,
optional(sequence(optional(ldh_str),
let_dig,
NULL)),
NULL),
validate_label);
2012-05-22 00:57:33 +02:00
/**
* You could write it like this ...
* parser_t *indirect_subdomain = indirect();
* const parser_t *subdomain = choice(label,
* sequence(indirect_subdomain,
* ch('.'),
* label,
* NULL),
* NULL);
* bind_indirect(indirect_subdomain, subdomain);
*
* ... but this is easier and equivalent
*/
const parser_t *subdomain = sepBy1(label, ch('.'));
const parser_t *domain = choice(subdomain,
ch(' '),
NULL);
const parser_t *dns_rr = sequence(domain, // NAME
uint16(), // TYPE
uint16(), // CLASS
uint32(), // TTL
length_value(uint16(), uint8()), // RDLENGTH+RDATA
NULL);
2012-05-22 00:57:33 +02:00
dns_message = (parser_t*)attr_bool(sequence(dns_header,
many(dns_question),
many(dns_rr),
end_p(),
NULL),
validate_dns);
return dns_message;
}