summaryrefslogtreecommitdiff
path: root/trafgen_l7.c
diff options
context:
space:
mode:
authorVadim Kochan <vadim4j@gmail.com>2017-06-01 13:12:28 +0300
committerTobias Klauser <tklauser@distanz.ch>2017-06-02 09:15:27 +0200
commit9d5e303a998e4618d492c9d0befb09e3cd3d746b (patch)
tree8ed2c72a573cad5acf92b62f4f0048c9cff251c4 /trafgen_l7.c
parenta2524f2bc9c0e8651221ba88b6013db55cec0190 (diff)
trafgen: l7: Add DNS header generation API
Add trafgen_l7.c module with DNS proto header generation with support of filling DNS query/answer/authority/additional sections as sub headers. Introcuded new concept as 'sub header' which is needed to easy handle DNS sections which might be added on-demand, and to simplify using sub-header as regular header with a fields, offset, etc. There is a parent header which contains array of pointers of sub-headers, and the array is ordered as they are located in the parent header. The sub-headers mostly encapsulated by the parent header which 'knows' the semantic of them. The new proto_hdr->push_sub_header(...) callback was added to tell the parent header to push the sub-header's fields, sub-header also may have proto_ops which must be filled by the parent. This sub-header concept might be used in the future if it will be needed to support DHCP, WLAN headers. There are 4 kinds of DNS sub-headers - query, answer, authority, additional. 'id' of each sub-header is used to only differentiate these types of sections. These sections have strict order inside DNS header, and there was added the proto_hdr_move_sub_header(...) to sort them in required order. Actually there are only 2 proto_hdr's which describes 4 DNS sections - query & rrecord, because rrecord covers another 3 - answer, auhority, additional which have the same layout. Signed-off-by: Vadim Kochan <vadim4j@gmail.com> Signed-off-by: Tobias Klauser <tklauser@distanz.ch>
Diffstat (limited to 'trafgen_l7.c')
-rw-r--r--trafgen_l7.c170
1 files changed, 170 insertions, 0 deletions
diff --git a/trafgen_l7.c b/trafgen_l7.c
new file mode 100644
index 0000000..f801677
--- /dev/null
+++ b/trafgen_l7.c
@@ -0,0 +1,170 @@
+/*
+ * netsniff-ng - the packet sniffing beast
+ * Subject to the GPL, version 2.
+ */
+
+#include <string.h>
+
+#include "str.h"
+#include "xmalloc.h"
+#include "built_in.h"
+#include "trafgen_l7.h"
+#include "trafgen_proto.h"
+
+static struct proto_field dns_fields[] = {
+ { .id = DNS_ID, .len = 2, .offset = 0 },
+ { .id = DNS_QR, .len = 2, .offset = 2, .shift = 15, .mask = 0x8000 },
+ { .id = DNS_OPCODE, .len = 2, .offset = 2, .shift = 11, .mask = 0x7800 },
+ { .id = DNS_AA, .len = 2, .offset = 2, .shift = 10, .mask = 0x0400 },
+ { .id = DNS_TC, .len = 2, .offset = 2, .shift = 9, .mask = 0x0200 },
+ { .id = DNS_RD, .len = 2, .offset = 2, .shift = 8, .mask = 0x0100 },
+ { .id = DNS_RA, .len = 2, .offset = 2, .shift = 7, .mask = 0x80 },
+ { .id = DNS_ZERO, .len = 2, .offset = 2, .shift = 4, .mask = 0x30 },
+ { .id = DNS_RCODE, .len = 2, .offset = 2, .shift = 0, .mask = 0xf },
+ { .id = DNS_QD_COUNT, .len = 2, .offset = 4, },
+ { .id = DNS_AN_COUNT, .len = 2, .offset = 6, },
+ { .id = DNS_NS_COUNT, .len = 2, .offset = 8, },
+ { .id = DNS_AR_COUNT, .len = 2, .offset = 10, },
+};
+
+static struct proto_field dns_query_fields[] = {
+ { .id = DNS_QUERY_NAME, .len = 0, .offset = 0 },
+ { .id = DNS_QUERY_TYPE, .len = 2, .offset = 0 },
+ { .id = DNS_QUERY_CLASS, .len = 2, .offset = 2 },
+};
+
+static void dns_query_header_init(struct proto_hdr *hdr)
+{
+ proto_header_fields_add(hdr, dns_query_fields, array_size(dns_query_fields));
+}
+
+static void dns_query_header_finish(struct proto_hdr *hdr)
+{
+ proto_hdr_field_set_default_string(hdr, DNS_QUERY_NAME, "www.netsniff-ng.com");
+ proto_hdr_field_set_default_be16(hdr, DNS_QUERY_CLASS, 1);
+ proto_hdr_field_set_default_be16(hdr, DNS_QUERY_TYPE, 1);
+}
+
+static const struct proto_ops dns_proto_query_ops = {
+ .header_init = dns_query_header_init,
+ .header_finish = dns_query_header_finish,
+};
+
+static struct proto_field dns_rrecord_fields[] = {
+ { .id = DNS_RRECORD_NAME, .len = 0, .offset = 0 },
+ { .id = DNS_RRECORD_TYPE, .len = 2, .offset = 0 },
+ { .id = DNS_RRECORD_CLASS, .len = 2, .offset = 2 },
+ { .id = DNS_RRECORD_TTL, .len = 4, .offset = 4 },
+ { .id = DNS_RRECORD_LEN, .len = 2, .offset = 8 },
+ { .id = DNS_RRECORD_DATA, .len = 0, .offset = 10 },
+};
+
+static void dns_rrecord_header_init(struct proto_hdr *hdr)
+{
+ proto_header_fields_add(hdr, dns_rrecord_fields, array_size(dns_rrecord_fields));
+}
+
+static void dns_rrecord_header_finish(struct proto_hdr *hdr)
+{
+ struct proto_field *data = proto_hdr_field_by_id(hdr, DNS_RRECORD_DATA);
+
+ proto_hdr_field_set_default_be32(hdr, DNS_RRECORD_TTL, 1);
+ proto_hdr_field_set_default_be16(hdr, DNS_RRECORD_CLASS, 1);
+ proto_hdr_field_set_default_be16(hdr, DNS_RRECORD_LEN, data->len);
+}
+
+static const struct proto_ops dns_proto_rrecord_ops = {
+ .header_init = dns_rrecord_header_init,
+ .header_finish = dns_rrecord_header_finish,
+};
+
+static void dns_header_init(struct proto_hdr *hdr)
+{
+ proto_lower_default_add(hdr, PROTO_UDP);
+
+ proto_header_fields_add(hdr, dns_fields, array_size(dns_fields));
+}
+
+static void dns_sort_headers(struct proto_hdr *hdr, uint32_t id, int index)
+{
+ int i;
+
+ for (i = index; i < hdr->sub_headers_count; i++) {
+ struct proto_hdr *sub_hdr = hdr->sub_headers[i];
+
+ if (sub_hdr->id == id && sub_hdr->index != index) {
+ proto_hdr_move_sub_header(hdr, sub_hdr, hdr->sub_headers[index]);
+ index++;
+ }
+ }
+}
+
+static void dns_header_finish(struct proto_hdr *hdr)
+{
+ size_t ar_count = 0;
+ size_t ns_count = 0;
+ size_t qd_count = 0;
+ size_t an_count = 0;
+ int i;
+
+ for (i = 0; i < hdr->sub_headers_count; i++) {
+ struct proto_hdr *sub_hdr = hdr->sub_headers[i];
+
+ switch (sub_hdr->id) {
+ case DNS_QUERY_HDR:
+ qd_count++;
+ break;
+ case DNS_ANSWER_HDR:
+ an_count++;
+ break;
+ case DNS_AUTH_HDR:
+ ns_count++;
+ break;
+ case DNS_ADD_HDR:
+ ar_count++;
+ break;
+ }
+ }
+
+ dns_sort_headers(hdr, DNS_QUERY_HDR, 0);
+ dns_sort_headers(hdr, DNS_ANSWER_HDR, qd_count);
+ dns_sort_headers(hdr, DNS_AUTH_HDR, qd_count + an_count);
+ dns_sort_headers(hdr, DNS_ADD_HDR, qd_count + an_count + ns_count);
+
+ proto_hdr_field_set_default_be16(hdr, DNS_QD_COUNT, qd_count);
+ proto_hdr_field_set_default_be16(hdr, DNS_AN_COUNT, an_count);
+ proto_hdr_field_set_default_be16(hdr, DNS_NS_COUNT, ns_count);
+ proto_hdr_field_set_default_be16(hdr, DNS_AR_COUNT, ar_count);
+
+ if (an_count)
+ proto_hdr_field_set_default_be16(hdr, DNS_QR, 1);
+}
+
+static void dns_push_sub_header(struct proto_hdr *hdr, struct proto_hdr *sub_hdr)
+{
+ switch (sub_hdr->id) {
+ case DNS_QUERY_HDR:
+ sub_hdr->ops = &dns_proto_query_ops;
+ break;
+ case DNS_ANSWER_HDR:
+ case DNS_AUTH_HDR:
+ case DNS_ADD_HDR:
+ sub_hdr->ops = &dns_proto_rrecord_ops;
+ break;
+ default:
+ bug();
+ }
+}
+
+static const struct proto_ops dns_proto_ops = {
+ .id = PROTO_DNS,
+ .layer = PROTO_L7,
+ .header_init = dns_header_init,
+ .header_finish = dns_header_finish,
+ .push_sub_header = dns_push_sub_header,
+};
+
+void protos_l7_init(void)
+{
+ proto_ops_register(&dns_proto_ops);
+}