Daniel Drown | a45056e | 2012-03-23 10:42:54 -0500 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2011 Daniel Drown |
| 3 | * |
| 4 | * Licensed under the Apache License, Version 2.0 (the "License"); |
| 5 | * you may not use this file except in compliance with the License. |
| 6 | * You may obtain a copy of the License at |
| 7 | * |
| 8 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | * |
| 10 | * Unless required by applicable law or agreed to in writing, software |
| 11 | * distributed under the License is distributed on an "AS IS" BASIS, |
| 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 13 | * See the License for the specific language governing permissions and |
| 14 | * limitations under the License. |
| 15 | * |
| 16 | * translate.h - translate from one version of ip to another |
| 17 | */ |
| 18 | #ifndef __TRANSLATE_H__ |
| 19 | #define __TRANSLATE_H__ |
| 20 | |
Lorenzo Colitti | f939060 | 2014-02-13 12:53:35 +0900 | [diff] [blame] | 21 | #include <netinet/in.h> |
| 22 | #include <netinet/ip.h> |
| 23 | #include <netinet/ip_icmp.h> |
| 24 | #include <netinet/udp.h> |
| 25 | #include <netinet/tcp.h> |
| 26 | #include <netinet/ip6.h> |
| 27 | #include <netinet/icmp6.h> |
| 28 | #include <linux/icmp.h> |
Lorenzo Colitti | d908418 | 2013-03-22 00:42:21 +0900 | [diff] [blame] | 29 | #include <linux/if_tun.h> |
Daniel Drown | a45056e | 2012-03-23 10:42:54 -0500 | [diff] [blame] | 30 | |
Lorenzo Colitti | f939060 | 2014-02-13 12:53:35 +0900 | [diff] [blame] | 31 | #include "clatd.h" |
| 32 | |
Lorenzo Colitti | d908418 | 2013-03-22 00:42:21 +0900 | [diff] [blame] | 33 | #define MAX_TCP_HDR (15 * 4) // Data offset field is 4 bits and counts in 32-bit words. |
Daniel Drown | a45056e | 2012-03-23 10:42:54 -0500 | [diff] [blame] | 34 | |
Lorenzo Colitti | d908418 | 2013-03-22 00:42:21 +0900 | [diff] [blame] | 35 | // A clat_packet is an array of iovec structures representing a packet that we are translating. |
Lorenzo Colitti | ee80ca6 | 2013-04-10 16:52:22 +0900 | [diff] [blame] | 36 | // The CLAT_POS_XXX constants represent the array indices within the clat_packet that contain |
| 37 | // specific parts of the packet. The packet_* functions operate on all the packet segments past a |
| 38 | // given position. |
Lorenzo Colitti | a4454bf | 2014-02-25 09:27:31 +0900 | [diff] [blame^] | 39 | typedef enum { |
Lorenzo Colitti | 57d480d | 2014-02-09 10:35:38 +0900 | [diff] [blame] | 40 | CLAT_POS_TUNHDR, CLAT_POS_IPHDR, CLAT_POS_FRAGHDR, CLAT_POS_TRANSPORTHDR, |
| 41 | CLAT_POS_ICMPERR_IPHDR, CLAT_POS_ICMPERR_FRAGHDR, CLAT_POS_ICMPERR_TRANSPORTHDR, |
| 42 | CLAT_POS_PAYLOAD, CLAT_POS_MAX |
Lorenzo Colitti | a4454bf | 2014-02-25 09:27:31 +0900 | [diff] [blame^] | 43 | } clat_packet_index; |
Lorenzo Colitti | ee80ca6 | 2013-04-10 16:52:22 +0900 | [diff] [blame] | 44 | typedef struct iovec clat_packet[CLAT_POS_MAX]; |
Lorenzo Colitti | d908418 | 2013-03-22 00:42:21 +0900 | [diff] [blame] | 45 | |
Lorenzo Colitti | ee80ca6 | 2013-04-10 16:52:22 +0900 | [diff] [blame] | 46 | // Calculates the checksum over all the packet components starting from pos. |
Lorenzo Colitti | a4454bf | 2014-02-25 09:27:31 +0900 | [diff] [blame^] | 47 | uint16_t packet_checksum(uint32_t checksum, clat_packet packet, clat_packet_index pos); |
Lorenzo Colitti | ee80ca6 | 2013-04-10 16:52:22 +0900 | [diff] [blame] | 48 | |
| 49 | // Returns the total length of the packet components after pos. |
Lorenzo Colitti | a4454bf | 2014-02-25 09:27:31 +0900 | [diff] [blame^] | 50 | uint16_t packet_length(clat_packet packet, clat_packet_index pos); |
Lorenzo Colitti | ee80ca6 | 2013-04-10 16:52:22 +0900 | [diff] [blame] | 51 | |
| 52 | // Returns true iff the given IPv6 address is in the plat subnet. |
| 53 | int is_in_plat_subnet(const struct in6_addr *addr6); |
Lorenzo Colitti | d908418 | 2013-03-22 00:42:21 +0900 | [diff] [blame] | 54 | |
| 55 | // Functions to create tun, IPv4, and IPv6 headers. |
| 56 | void fill_tun_header(struct tun_pi *tun_header, uint16_t proto); |
| 57 | void fill_ip_header(struct iphdr *ip_targ, uint16_t payload_len, uint8_t protocol, |
| 58 | const struct ip6_hdr *old_header); |
| 59 | void fill_ip6_header(struct ip6_hdr *ip6, uint16_t payload_len, uint8_t protocol, |
| 60 | const struct iphdr *old_header); |
| 61 | |
Lorenzo Colitti | f939060 | 2014-02-13 12:53:35 +0900 | [diff] [blame] | 62 | // Translate and send packets. |
Brian Carlstrom | fcac410 | 2014-02-24 20:03:01 -0800 | [diff] [blame] | 63 | void translate_packet(const struct tun_data *tunnel, struct tun_pi *tun_header, |
| 64 | const uint8_t *packet, size_t packetsize); |
Lorenzo Colitti | f939060 | 2014-02-13 12:53:35 +0900 | [diff] [blame] | 65 | |
Lorenzo Colitti | 07fc761 | 2013-11-18 13:33:30 +0900 | [diff] [blame] | 66 | // Translate IPv4 and IPv6 packets. |
Lorenzo Colitti | a4454bf | 2014-02-25 09:27:31 +0900 | [diff] [blame^] | 67 | int ipv4_packet(clat_packet out, clat_packet_index pos, const uint8_t *packet, size_t len); |
| 68 | int ipv6_packet(clat_packet out, clat_packet_index pos, const uint8_t *packet, size_t len); |
Lorenzo Colitti | 07fc761 | 2013-11-18 13:33:30 +0900 | [diff] [blame] | 69 | |
Lorenzo Colitti | 57d480d | 2014-02-09 10:35:38 +0900 | [diff] [blame] | 70 | // Deal with fragmented packets. |
| 71 | size_t maybe_fill_frag_header(struct ip6_frag *frag_hdr, struct ip6_hdr *ip6_targ, |
| 72 | const struct iphdr *old_header); |
| 73 | uint8_t parse_frag_header(const struct ip6_frag *frag_hdr, struct iphdr *ip_targ); |
| 74 | |
Lorenzo Colitti | d908418 | 2013-03-22 00:42:21 +0900 | [diff] [blame] | 75 | // Translate ICMP packets. |
Lorenzo Colitti | a4454bf | 2014-02-25 09:27:31 +0900 | [diff] [blame^] | 76 | int icmp_to_icmp6(clat_packet out, clat_packet_index pos, const struct icmphdr *icmp, |
| 77 | uint32_t checksum, const uint8_t *payload, size_t payload_size); |
| 78 | int icmp6_to_icmp(clat_packet out, clat_packet_index pos, const struct icmp6_hdr *icmp6, |
Brian Carlstrom | fcac410 | 2014-02-24 20:03:01 -0800 | [diff] [blame] | 79 | const uint8_t *payload, size_t payload_size); |
Lorenzo Colitti | d908418 | 2013-03-22 00:42:21 +0900 | [diff] [blame] | 80 | |
Lorenzo Colitti | c9f4c89 | 2013-11-18 12:59:44 +0900 | [diff] [blame] | 81 | // Translate generic IP packets. |
Lorenzo Colitti | a4454bf | 2014-02-25 09:27:31 +0900 | [diff] [blame^] | 82 | int generic_packet(clat_packet out, clat_packet_index pos, const uint8_t *payload, size_t len); |
Lorenzo Colitti | c9f4c89 | 2013-11-18 12:59:44 +0900 | [diff] [blame] | 83 | |
Lorenzo Colitti | d908418 | 2013-03-22 00:42:21 +0900 | [diff] [blame] | 84 | // Translate TCP and UDP packets. |
Lorenzo Colitti | a4454bf | 2014-02-25 09:27:31 +0900 | [diff] [blame^] | 85 | int tcp_packet(clat_packet out, clat_packet_index pos, const struct tcphdr *tcp, |
Lorenzo Colitti | 5a50c02 | 2014-02-10 09:20:05 +0900 | [diff] [blame] | 86 | uint32_t old_sum, uint32_t new_sum, size_t len); |
Lorenzo Colitti | a4454bf | 2014-02-25 09:27:31 +0900 | [diff] [blame^] | 87 | int udp_packet(clat_packet out, clat_packet_index pos, const struct udphdr *udp, |
Lorenzo Colitti | 5a50c02 | 2014-02-10 09:20:05 +0900 | [diff] [blame] | 88 | uint32_t old_sum, uint32_t new_sum, size_t len); |
Lorenzo Colitti | d908418 | 2013-03-22 00:42:21 +0900 | [diff] [blame] | 89 | |
Lorenzo Colitti | a4454bf | 2014-02-25 09:27:31 +0900 | [diff] [blame^] | 90 | int tcp_translate(clat_packet out, clat_packet_index pos, const struct tcphdr *tcp, |
| 91 | size_t header_size, uint32_t old_sum, uint32_t new_sum, |
Brian Carlstrom | fcac410 | 2014-02-24 20:03:01 -0800 | [diff] [blame] | 92 | const uint8_t *payload, size_t payload_size); |
Lorenzo Colitti | a4454bf | 2014-02-25 09:27:31 +0900 | [diff] [blame^] | 93 | int udp_translate(clat_packet out, clat_packet_index pos, const struct udphdr *udp, |
Brian Carlstrom | fcac410 | 2014-02-24 20:03:01 -0800 | [diff] [blame] | 94 | uint32_t old_sum, uint32_t new_sum, |
| 95 | const uint8_t *payload, size_t payload_size); |
Daniel Drown | a45056e | 2012-03-23 10:42:54 -0500 | [diff] [blame] | 96 | |
| 97 | #endif /* __TRANSLATE_H__ */ |