/** * @file traffic_selector.c * * @brief Implementation of traffic_selector_t. * */ /* * Copyright (C) 2007 Tobias Brunner * Copyright (C) 2005-2007 Martin Willi * Copyright (C) 2005 Jan Hutter * Hochschule fuer Technik Rapperswil * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the * Free Software Foundation; either version 2 of the License, or (at your * option) any later version. See . * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * for more details. */ #include #include #include #include #include #include "traffic_selector.h" #include #include #include ENUM(ts_type_name, TS_IPV4_ADDR_RANGE, TS_IPV6_ADDR_RANGE, "TS_IPV4_ADDR_RANGE", "TS_IPV6_ADDR_RANGE", ); typedef struct private_traffic_selector_t private_traffic_selector_t; /** * Private data of an traffic_selector_t object */ struct private_traffic_selector_t { /** * Public part */ traffic_selector_t public; /** * Type of address */ ts_type_t type; /** * IP protocol (UDP, TCP, ICMP, ...) */ u_int8_t protocol; /** * narrow this traffic selector to hosts external ip * if set, from and to have no meaning until set_address() is called */ bool dynamic; /** * begin of address range, network order */ union { /** dummy char for common address manipulation */ char from[0]; /** IPv4 address */ u_int32_t from4[1]; /** IPv6 address */ u_int32_t from6[4]; }; /** * end of address range, network order */ union { /** dummy char for common address manipulation */ char to[0]; /** IPv4 address */ u_int32_t to4[1]; /** IPv6 address */ u_int32_t to6[4]; }; /** * begin of port range */ u_int16_t from_port; /** * end of port range */ u_int16_t to_port; }; /** * calculate to "to"-address for the "from" address and a subnet size */ static void calc_range(private_traffic_selector_t *this, u_int8_t netbits) { int byte; size_t size = (this->type == TS_IPV4_ADDR_RANGE) ? 4 : 16; /* go through the from address, starting at the tail. While we * have not processed the bits belonging to the host, set them to 1 on * the to address. If we reach the bits for the net, copy them from "from". */ for (byte = size - 1; byte >=0; byte--) { u_char mask = 0x00; int shift; shift = (byte+1) * 8 - netbits; if (shift > 0) { mask = 1 << shift; if (mask != 0xFF) { mask--; } } this->to[byte] = this->from[byte] | mask; } } /** * calculate to subnet size from "to"- and "from"-address */ static u_int8_t calc_netbits(private_traffic_selector_t *this) { int byte, bit; size_t size = (this->type == TS_IPV4_ADDR_RANGE) ? 4 : 16; /* go trough all bits of the addresses, begging in the front. * As longer as they equal, the subnet gets larger */ for (byte = 0; byte < size; byte++) { for (bit = 7; bit >= 0; bit--) { if ((1<from[byte]) != (1<to[byte])) { return ((7 - bit) + (byte * 8)); } } } /* single host, netmask is 32/128 */ return (size * 8); } /** * internal generic constructor */ static private_traffic_selector_t *traffic_selector_create(u_int8_t protocol, ts_type_t type, u_int16_t from_port, u_int16_t to_port); /** * output handler in printf() */ static int print(FILE *stream, const struct printf_info *info, const void *const *args) { private_traffic_selector_t *this = *((private_traffic_selector_t**)(args[0])); char addr_str[INET6_ADDRSTRLEN] = ""; char *serv_proto = NULL; u_int8_t mask; bool has_proto; bool has_ports; size_t written = 0; if (this == NULL) { return fprintf(stream, "(null)"); } if (this->type == TS_IPV4_ADDR_RANGE) { inet_ntop(AF_INET, &this->from4, addr_str, sizeof(addr_str)); } else { inet_ntop(AF_INET6, &this->from6, addr_str, sizeof(addr_str)); } mask = calc_netbits(this); written += fprintf(stream, "%s/%d", addr_str, mask); /* check if we have protocol and/or port selectors */ has_proto = this->protocol != 0; has_ports = !(this->from_port == 0 && this->to_port == 0xFFFF); if (!has_proto && !has_ports) { return written; } written += fprintf(stream, "["); /* build protocol string */ if (has_proto) { struct protoent *proto = getprotobynumber(this->protocol); if (proto) { written += fprintf(stream, "%s", proto->p_name); serv_proto = proto->p_name; } else { written += fprintf(stream, "%d", this->protocol); } } if (has_proto && has_ports) { written += fprintf(stream, "/"); } /* build port string */ if (has_ports) { if (this->from_port == this->to_port) { struct servent *serv = getservbyport(htons(this->from_port), serv_proto); if (serv) { written += fprintf(stream, "%s", serv->s_name); } else { written += fprintf(stream, "%d", this->from_port); } } else { written += fprintf(stream, "%d-%d", this->from_port, this->to_port); } } written += fprintf(stream, "]"); return written; } /** * register printf() handlers */ static void __attribute__ ((constructor))print_register() { register_printf_function(PRINTF_TRAFFIC_SELECTOR, print, arginfo_ptr); } /** * implements traffic_selector_t.get_subset */ static traffic_selector_t *get_subset(private_traffic_selector_t *this, private_traffic_selector_t *other) { if (this->type == other->type && (this->protocol == other->protocol || this->protocol == 0 || other->protocol == 0)) { u_int16_t from_port, to_port; u_char *from, *to; u_int8_t protocol; size_t size; private_traffic_selector_t *new_ts; /* calculate the maximum port range allowed for both */ from_port = max(this->from_port, other->from_port); to_port = min(this->to_port, other->to_port); if (from_port > to_port) { return NULL; } /* select protocol, which is not zero */ protocol = max(this->protocol, other->protocol); switch (this->type) { case TS_IPV4_ADDR_RANGE: size = sizeof(this->from4); break; case TS_IPV6_ADDR_RANGE: size = sizeof(this->from6); break; default: return NULL; } /* get higher from-address */ if (memcmp(this->from, other->from, size) > 0) { from = this->from; } else { from = other->from; } /* get lower to-address */ if (memcmp(this->to, other->to, size) > 0) { to = other->to; } else { to = this->to; } /* if "from" > "to", we don't have a match */ if (memcmp(from, to, size) > 0) { return NULL; } /* we have a match in protocol, port, and address: return it... */ new_ts = traffic_selector_create(protocol, this->type, from_port, to_port); new_ts->type = this->type; memcpy(new_ts->from, from, size); memcpy(new_ts->to, to, size); return &new_ts->public; } return NULL; } /** * implements traffic_selector_t.equals */ static bool equals(private_traffic_selector_t *this, private_traffic_selector_t *other) { if (this->type != other->type) { return FALSE; } if (!(this->from_port == other->from_port && this->to_port == other->to_port && this->protocol == other->protocol)) { return FALSE; } switch (this->type) { case TS_IPV4_ADDR_RANGE: if (memeq(this->from4, other->from4, sizeof(this->from4))) { return TRUE; } break; case TS_IPV6_ADDR_RANGE: if (memeq(this->from6, other->from6, sizeof(this->from6))) { return TRUE; } break; default: break; } return FALSE; } /** * Implements traffic_selector_t.get_from_address. */ static chunk_t get_from_address(private_traffic_selector_t *this) { chunk_t from = chunk_empty; switch (this->type) { case TS_IPV4_ADDR_RANGE: { from.len = sizeof(this->from4); from.ptr = malloc(from.len); memcpy(from.ptr, this->from4, from.len); break; } case TS_IPV6_ADDR_RANGE: { from.len = sizeof(this->from6); from.ptr = malloc(from.len); memcpy(from.ptr, this->from6, from.len); break; } } return from; } /** * Implements traffic_selector_t.get_to_address. */ static chunk_t get_to_address(private_traffic_selector_t *this) { chunk_t to = chunk_empty; switch (this->type) { case TS_IPV4_ADDR_RANGE: { to.len = sizeof(this->to4); to.ptr = malloc(to.len); memcpy(to.ptr, this->to4, to.len); break; } case TS_IPV6_ADDR_RANGE: { to.len = sizeof(this->to6); to.ptr = malloc(to.len); memcpy(to.ptr, this->to6, to.len); break; } } return to; } /** * Implements traffic_selector_t.get_from_port. */ static u_int16_t get_from_port(private_traffic_selector_t *this) { return this->from_port; } /** * Implements traffic_selector_t.get_to_port. */ static u_int16_t get_to_port(private_traffic_selector_t *this) { return this->to_port; } /** * Implements traffic_selector_t.get_type. */ static ts_type_t get_type(private_traffic_selector_t *this) { return this->type; } /** * Implements traffic_selector_t.get_protocol. */ static u_int8_t get_protocol(private_traffic_selector_t *this) { return this->protocol; } /** * Implements traffic_selector_t.is_host. */ static bool is_host(private_traffic_selector_t *this, host_t *host) { if (this->dynamic) { return TRUE; } if (host) { chunk_t addr; int family = host->get_family(host); if ((family == AF_INET && this->type == TS_IPV4_ADDR_RANGE) || (family == AF_INET6 && this->type == TS_IPV6_ADDR_RANGE)) { addr = host->get_address(host); if (memeq(addr.ptr, this->from, addr.len) && memeq(addr.ptr, this->to, addr.len)) { return TRUE; } } } else { size_t length = (this->type == TS_IPV4_ADDR_RANGE) ? 4 : 16; if (memeq(this->from, this->to, length)) { return TRUE; } } return FALSE; } /** * Implements traffic_selector_t.set_address. */ static void set_address(private_traffic_selector_t *this, host_t *host) { if (this->dynamic) { this->type = host->get_family(host) == AF_INET ? TS_IPV4_ADDR_RANGE : TS_IPV6_ADDR_RANGE; chunk_t from = host->get_address(host); memcpy(this->from, from.ptr, from.len); memcpy(this->to, from.ptr, from.len); } } /** * Implements traffic_selector_t.is_contained_in. */ static bool is_contained_in(private_traffic_selector_t *this, private_traffic_selector_t *other) { private_traffic_selector_t *subset; bool contained_in = FALSE; subset = (private_traffic_selector_t*)get_subset(this, other); if (subset) { if (equals(subset, this)) { contained_in = TRUE; } free(subset); } return contained_in; } /** * Implements traffic_selector_t.includes. */ static bool includes(private_traffic_selector_t *this, host_t *host) { chunk_t addr; int family = host->get_family(host); if ((family == AF_INET && this->type == TS_IPV4_ADDR_RANGE) || (family == AF_INET6 && this->type == TS_IPV6_ADDR_RANGE)) { addr = host->get_address(host); return memcmp(this->from, addr.ptr, addr.len) <= 0 && memcmp(this->to, addr.ptr, addr.len) >= 0; } return FALSE; } /** * Implements traffic_selector_t.clone. */ static traffic_selector_t *clone_(private_traffic_selector_t *this) { private_traffic_selector_t *clone; clone = traffic_selector_create(this->protocol, this->type, this->from_port, this->to_port); clone->dynamic = this->dynamic; switch (clone->type) { case TS_IPV4_ADDR_RANGE: { memcpy(clone->from4, this->from4, sizeof(this->from4)); memcpy(clone->to4, this->to4, sizeof(this->to4)); return &clone->public; } case TS_IPV6_ADDR_RANGE: { memcpy(clone->from6, this->from6, sizeof(this->from6)); memcpy(clone->to6, this->to6, sizeof(this->to6)); return &clone->public; } default: { /* unreachable */ return &clone->public; } } } /** * Implements traffic_selector_t.destroy. */ static void destroy(private_traffic_selector_t *this) { free(this); } /* * see header */ traffic_selector_t *traffic_selector_create_from_bytes(u_int8_t protocol, ts_type_t type, chunk_t from, u_int16_t from_port, chunk_t to, u_int16_t to_port) { private_traffic_selector_t *this = traffic_selector_create(protocol, type, from_port, to_port); switch (type) { case TS_IPV4_ADDR_RANGE: { if (from.len != 4 || to.len != 4) { free(this); return NULL; } memcpy(this->from4, from.ptr, from.len); memcpy(this->to4, to.ptr, to.len); break; } case TS_IPV6_ADDR_RANGE: { if (from.len != 16 || to.len != 16) { free(this); return NULL; } memcpy(this->from6, from.ptr, from.len); memcpy(this->to6, to.ptr, to.len); break; } default: { free(this); return NULL; } } return (&this->public); } /* * see header */ traffic_selector_t *traffic_selector_create_from_subnet(host_t *net, u_int8_t netbits, u_int8_t protocol, u_int16_t port) { private_traffic_selector_t *this = traffic_selector_create(protocol, 0, 0, 65535); switch (net->get_family(net)) { case AF_INET: { chunk_t from; this->type = TS_IPV4_ADDR_RANGE; from = net->get_address(net); memcpy(this->from4, from.ptr, from.len); if (this->from4[0] == 0) { /* use /0 for 0.0.0.0 */ this->to4[0] = ~0; } else { calc_range(this, netbits); } break; } case AF_INET6: { chunk_t from; this->type = TS_IPV6_ADDR_RANGE; from = net->get_address(net); memcpy(this->from6, from.ptr, from.len); if (this->from6[0] == 0 && this->from6[1] == 0 && this->from6[2] == 0 && this->from6[3] == 0) { /* use /0 for ::0 */ this->to6[0] = ~0; this->to6[1] = ~0; this->to6[2] = ~0; this->to6[3] = ~0; } else { calc_range(this, netbits); } break; } default: { free(this); return NULL; } } if (port) { this->from_port = port; this->to_port = port; } return (&this->public); } /* * see header */ traffic_selector_t *traffic_selector_create_from_string( u_int8_t protocol, ts_type_t type, char *from_addr, u_int16_t from_port, char *to_addr, u_int16_t to_port) { private_traffic_selector_t *this = traffic_selector_create(protocol, type, from_port, to_port); this->type = type; switch (type) { case TS_IPV4_ADDR_RANGE: { if (inet_pton(AF_INET, from_addr, (struct in_addr*)this->from4) < 0) { free(this); return NULL; } if (inet_pton(AF_INET, to_addr, (struct in_addr*)this->to4) < 0) { free(this); return NULL; } break; } case TS_IPV6_ADDR_RANGE: { if (inet_pton(AF_INET6, from_addr, (struct in6_addr*)this->from6) < 0) { free(this); return NULL; } if (inet_pton(AF_INET6, to_addr, (struct in6_addr*)this->to6) < 0) { free(this); return NULL; } break; } } return (&this->public); } /* * see header */ traffic_selector_t *traffic_selector_create_dynamic( u_int8_t protocol, ts_type_t type, u_int16_t from_port, u_int16_t to_port) { private_traffic_selector_t *this = traffic_selector_create(protocol, type, from_port, to_port); memset(this->from6, 0, sizeof(this->from6)); memset(this->to6, 0xFF, sizeof(this->to6)); this->dynamic = TRUE; return &this->public; } /* * see declaration */ static private_traffic_selector_t *traffic_selector_create(u_int8_t protocol, ts_type_t type, u_int16_t from_port, u_int16_t to_port) { private_traffic_selector_t *this = malloc_thing(private_traffic_selector_t); /* public functions */ this->public.get_subset = (traffic_selector_t*(*)(traffic_selector_t*,traffic_selector_t*))get_subset; this->public.equals = (bool(*)(traffic_selector_t*,traffic_selector_t*))equals; this->public.get_from_address = (chunk_t(*)(traffic_selector_t*))get_from_address; this->public.get_to_address = (chunk_t(*)(traffic_selector_t*))get_to_address; this->public.get_from_port = (u_int16_t(*)(traffic_selector_t*))get_from_port; this->public.get_to_port = (u_int16_t(*)(traffic_selector_t*))get_to_port; this->public.get_type = (ts_type_t(*)(traffic_selector_t*))get_type; this->public.get_protocol = (u_int8_t(*)(traffic_selector_t*))get_protocol; this->public.is_host = (bool(*)(traffic_selector_t*,host_t*))is_host; this->public.is_contained_in = (bool(*)(traffic_selector_t*,traffic_selector_t*))is_contained_in; this->public.includes = (bool(*)(traffic_selector_t*,host_t*))includes; this->public.set_address = (void(*)(traffic_selector_t*,host_t*))set_address; this->public.clone = (traffic_selector_t*(*)(traffic_selector_t*))clone_; this->public.destroy = (void(*)(traffic_selector_t*))destroy; this->from_port = from_port; this->to_port = to_port; this->protocol = protocol; this->type = type; this->dynamic = FALSE; return this; } /* vim: set ts=4 sw=4 noet: */