]>
Commit | Line | Data |
---|---|---|
b9e8b45a | 1 | /* |
eb0b295e | 2 | * Copyright (c) 2009, 2010, 2011, 2012, 2013, 2014, 2015 Nicira, Inc. |
b9e8b45a BP |
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 | ||
17 | #include <config.h> | |
18 | #include "packets.h" | |
d31f1109 | 19 | #include <arpa/inet.h> |
6ca00f6f | 20 | #include <sys/socket.h> |
b9e8b45a | 21 | #include <netinet/in.h> |
bc7a5acd | 22 | #include <netinet/ip6.h> |
00894212 | 23 | #include <netinet/icmp6.h> |
76343538 | 24 | #include <stdlib.h> |
d31f1109 | 25 | #include "byte-order.h" |
c97664b3 | 26 | #include "csum.h" |
c6bcb685 | 27 | #include "crc32c.h" |
12113c39 | 28 | #include "flow.h" |
7d48a4cc | 29 | #include "hmap.h" |
d31f1109 | 30 | #include "dynamic-string.h" |
8c45d00f | 31 | #include "ovs-thread.h" |
b5e7e61a | 32 | #include "odp-util.h" |
cf62fa4c | 33 | #include "dp-packet.h" |
7c457c33 | 34 | #include "unaligned.h" |
b9e8b45a | 35 | |
d31f1109 | 36 | const struct in6_addr in6addr_exact = IN6ADDR_EXACT_INIT; |
06994f87 | 37 | const struct in6_addr in6addr_all_hosts = IN6ADDR_ALL_HOSTS_INIT; |
d31f1109 | 38 | |
093ca5b3 BP |
39 | /* Parses 's' as a 16-digit hexadecimal number representing a datapath ID. On |
40 | * success stores the dpid into '*dpidp' and returns true, on failure stores 0 | |
41 | * into '*dpidp' and returns false. | |
42 | * | |
43 | * Rejects an all-zeros dpid as invalid. */ | |
76343538 BP |
44 | bool |
45 | dpid_from_string(const char *s, uint64_t *dpidp) | |
46 | { | |
b123cc3c | 47 | *dpidp = (strlen(s) == 16 && strspn(s, "0123456789abcdefABCDEF") == 16 |
093ca5b3 | 48 | ? strtoull(s, NULL, 16) |
76343538 BP |
49 | : 0); |
50 | return *dpidp != 0; | |
51 | } | |
52 | ||
7d48a4cc BP |
53 | /* Returns true if 'ea' is a reserved address, that a bridge must never |
54 | * forward, false otherwise. | |
05be4e2c EJ |
55 | * |
56 | * If you change this function's behavior, please update corresponding | |
57 | * documentation in vswitch.xml at the same time. */ | |
58 | bool | |
74ff3298 | 59 | eth_addr_is_reserved(const struct eth_addr ea) |
05be4e2c | 60 | { |
7d48a4cc BP |
61 | struct eth_addr_node { |
62 | struct hmap_node hmap_node; | |
8c45d00f | 63 | const uint64_t ea64; |
05be4e2c EJ |
64 | }; |
65 | ||
7d48a4cc BP |
66 | static struct eth_addr_node nodes[] = { |
67 | /* STP, IEEE pause frames, and other reserved protocols. */ | |
f0ac9da9 BP |
68 | { HMAP_NODE_NULL_INITIALIZER, 0x0180c2000000ULL }, |
69 | { HMAP_NODE_NULL_INITIALIZER, 0x0180c2000001ULL }, | |
70 | { HMAP_NODE_NULL_INITIALIZER, 0x0180c2000002ULL }, | |
71 | { HMAP_NODE_NULL_INITIALIZER, 0x0180c2000003ULL }, | |
72 | { HMAP_NODE_NULL_INITIALIZER, 0x0180c2000004ULL }, | |
73 | { HMAP_NODE_NULL_INITIALIZER, 0x0180c2000005ULL }, | |
74 | { HMAP_NODE_NULL_INITIALIZER, 0x0180c2000006ULL }, | |
75 | { HMAP_NODE_NULL_INITIALIZER, 0x0180c2000007ULL }, | |
76 | { HMAP_NODE_NULL_INITIALIZER, 0x0180c2000008ULL }, | |
77 | { HMAP_NODE_NULL_INITIALIZER, 0x0180c2000009ULL }, | |
78 | { HMAP_NODE_NULL_INITIALIZER, 0x0180c200000aULL }, | |
79 | { HMAP_NODE_NULL_INITIALIZER, 0x0180c200000bULL }, | |
80 | { HMAP_NODE_NULL_INITIALIZER, 0x0180c200000cULL }, | |
81 | { HMAP_NODE_NULL_INITIALIZER, 0x0180c200000dULL }, | |
82 | { HMAP_NODE_NULL_INITIALIZER, 0x0180c200000eULL }, | |
83 | { HMAP_NODE_NULL_INITIALIZER, 0x0180c200000fULL }, | |
7d48a4cc BP |
84 | |
85 | /* Extreme protocols. */ | |
86 | { HMAP_NODE_NULL_INITIALIZER, 0x00e02b000000ULL }, /* EDP. */ | |
87 | { HMAP_NODE_NULL_INITIALIZER, 0x00e02b000004ULL }, /* EAPS. */ | |
88 | { HMAP_NODE_NULL_INITIALIZER, 0x00e02b000006ULL }, /* EAPS. */ | |
89 | ||
90 | /* Cisco protocols. */ | |
91 | { HMAP_NODE_NULL_INITIALIZER, 0x01000c000000ULL }, /* ISL. */ | |
92 | { HMAP_NODE_NULL_INITIALIZER, 0x01000cccccccULL }, /* PAgP, UDLD, CDP, | |
93 | * DTP, VTP. */ | |
94 | { HMAP_NODE_NULL_INITIALIZER, 0x01000ccccccdULL }, /* PVST+. */ | |
95 | { HMAP_NODE_NULL_INITIALIZER, 0x01000ccdcdcdULL }, /* STP Uplink Fast, | |
96 | * FlexLink. */ | |
97 | ||
98 | /* Cisco CFM. */ | |
99 | { HMAP_NODE_NULL_INITIALIZER, 0x01000cccccc0ULL }, | |
100 | { HMAP_NODE_NULL_INITIALIZER, 0x01000cccccc1ULL }, | |
101 | { HMAP_NODE_NULL_INITIALIZER, 0x01000cccccc2ULL }, | |
102 | { HMAP_NODE_NULL_INITIALIZER, 0x01000cccccc3ULL }, | |
103 | { HMAP_NODE_NULL_INITIALIZER, 0x01000cccccc4ULL }, | |
104 | { HMAP_NODE_NULL_INITIALIZER, 0x01000cccccc5ULL }, | |
105 | { HMAP_NODE_NULL_INITIALIZER, 0x01000cccccc6ULL }, | |
106 | { HMAP_NODE_NULL_INITIALIZER, 0x01000cccccc7ULL }, | |
107 | }; | |
05be4e2c | 108 | |
8c45d00f | 109 | static struct ovsthread_once once = OVSTHREAD_ONCE_INITIALIZER; |
7d48a4cc | 110 | struct eth_addr_node *node; |
8c45d00f | 111 | static struct hmap addrs; |
7d48a4cc | 112 | uint64_t ea64; |
05be4e2c | 113 | |
8c45d00f BP |
114 | if (ovsthread_once_start(&once)) { |
115 | hmap_init(&addrs); | |
7d48a4cc | 116 | for (node = nodes; node < &nodes[ARRAY_SIZE(nodes)]; node++) { |
965607c8 | 117 | hmap_insert(&addrs, &node->hmap_node, hash_uint64(node->ea64)); |
7d48a4cc | 118 | } |
8c45d00f | 119 | ovsthread_once_done(&once); |
7d48a4cc | 120 | } |
05be4e2c | 121 | |
7d48a4cc | 122 | ea64 = eth_addr_to_uint64(ea); |
965607c8 | 123 | HMAP_FOR_EACH_IN_BUCKET (node, hmap_node, hash_uint64(ea64), &addrs) { |
7d48a4cc | 124 | if (node->ea64 == ea64) { |
05be4e2c EJ |
125 | return true; |
126 | } | |
127 | } | |
128 | return false; | |
129 | } | |
130 | ||
76343538 | 131 | bool |
74ff3298 | 132 | eth_addr_from_string(const char *s, struct eth_addr *ea) |
76343538 | 133 | { |
74ff3298 | 134 | if (ovs_scan(s, ETH_ADDR_SCAN_FMT, ETH_ADDR_SCAN_ARGS(*ea))) { |
76343538 BP |
135 | return true; |
136 | } else { | |
74ff3298 | 137 | *ea = eth_addr_zero; |
76343538 BP |
138 | return false; |
139 | } | |
140 | } | |
141 | ||
38f7147c | 142 | /* Fills 'b' with a Reverse ARP packet with Ethernet source address 'eth_src'. |
b9e8b45a | 143 | * This function is used by Open vSwitch to compose packets in cases where |
38f7147c EJ |
144 | * context is important but content doesn't (or shouldn't) matter. |
145 | * | |
146 | * The returned packet has enough headroom to insert an 802.1Q VLAN header if | |
147 | * desired. */ | |
b9e8b45a | 148 | void |
74ff3298 | 149 | compose_rarp(struct dp_packet *b, const struct eth_addr eth_src) |
b9e8b45a | 150 | { |
38f7147c | 151 | struct eth_header *eth; |
7cb57d10 | 152 | struct arp_eth_header *arp; |
b9e8b45a | 153 | |
cf62fa4c PS |
154 | dp_packet_clear(b); |
155 | dp_packet_prealloc_tailroom(b, 2 + ETH_HEADER_LEN + VLAN_HEADER_LEN | |
7cb57d10 | 156 | + ARP_ETH_HEADER_LEN); |
cf62fa4c PS |
157 | dp_packet_reserve(b, 2 + VLAN_HEADER_LEN); |
158 | eth = dp_packet_put_uninit(b, sizeof *eth); | |
74ff3298 JR |
159 | eth->eth_dst = eth_addr_broadcast; |
160 | eth->eth_src = eth_src; | |
38f7147c EJ |
161 | eth->eth_type = htons(ETH_TYPE_RARP); |
162 | ||
cf62fa4c | 163 | arp = dp_packet_put_uninit(b, sizeof *arp); |
7cb57d10 EJ |
164 | arp->ar_hrd = htons(ARP_HRD_ETHERNET); |
165 | arp->ar_pro = htons(ARP_PRO_IP); | |
166 | arp->ar_hln = sizeof arp->ar_sha; | |
167 | arp->ar_pln = sizeof arp->ar_spa; | |
168 | arp->ar_op = htons(ARP_OP_RARP); | |
74ff3298 | 169 | arp->ar_sha = eth_src; |
7c457c33 | 170 | put_16aligned_be32(&arp->ar_spa, htonl(0)); |
74ff3298 | 171 | arp->ar_tha = eth_src; |
7c457c33 | 172 | put_16aligned_be32(&arp->ar_tpa, htonl(0)); |
cf3b7538 | 173 | |
82eb5b0a | 174 | dp_packet_reset_offsets(b); |
cf62fa4c | 175 | dp_packet_set_l3(b, arp); |
b9e8b45a | 176 | } |
d31f1109 | 177 | |
d9065a90 | 178 | /* Insert VLAN header according to given TCI. Packet passed must be Ethernet |
2f4ca41b | 179 | * packet. Ignores the CFI bit of 'tci' using 0 instead. |
7c66b273 | 180 | * |
cf3b7538 | 181 | * Also adjusts the layer offsets accordingly. */ |
7c66b273 | 182 | void |
cf62fa4c | 183 | eth_push_vlan(struct dp_packet *packet, ovs_be16 tpid, ovs_be16 tci) |
7c66b273 | 184 | { |
7c66b273 BP |
185 | struct vlan_eth_header *veh; |
186 | ||
d9065a90 | 187 | /* Insert new 802.1Q header. */ |
cf62fa4c | 188 | veh = dp_packet_resize_l2(packet, VLAN_HEADER_LEN); |
437d0d22 JR |
189 | memmove(veh, (char *)veh + VLAN_HEADER_LEN, 2 * ETH_ADDR_LEN); |
190 | veh->veth_type = tpid; | |
191 | veh->veth_tci = tci & htons(~VLAN_CFI); | |
7c66b273 BP |
192 | } |
193 | ||
f4ebc25e BP |
194 | /* Removes outermost VLAN header (if any is present) from 'packet'. |
195 | * | |
d6943394 TH |
196 | * 'packet->l2_5' should initially point to 'packet''s outer-most VLAN header |
197 | * or may be NULL if there are no VLAN headers. */ | |
f4ebc25e | 198 | void |
cf62fa4c | 199 | eth_pop_vlan(struct dp_packet *packet) |
f4ebc25e | 200 | { |
cf62fa4c | 201 | struct vlan_eth_header *veh = dp_packet_l2(packet); |
437d0d22 | 202 | |
cf62fa4c | 203 | if (veh && dp_packet_size(packet) >= sizeof *veh |
d6943394 | 204 | && eth_type_vlan(veh->veth_type)) { |
f4ebc25e | 205 | |
437d0d22 | 206 | memmove((char *)veh + VLAN_HEADER_LEN, veh, 2 * ETH_ADDR_LEN); |
cf62fa4c | 207 | dp_packet_resize_l2(packet, -VLAN_HEADER_LEN); |
f4ebc25e BP |
208 | } |
209 | } | |
210 | ||
b02475c5 | 211 | /* Set ethertype of the packet. */ |
56b02633 | 212 | static void |
cf62fa4c | 213 | set_ethertype(struct dp_packet *packet, ovs_be16 eth_type) |
b02475c5 | 214 | { |
cf62fa4c | 215 | struct eth_header *eh = dp_packet_l2(packet); |
cf3b7538 JR |
216 | |
217 | if (!eh) { | |
218 | return; | |
219 | } | |
b02475c5 | 220 | |
d6943394 | 221 | if (eth_type_vlan(eh->eth_type)) { |
b02475c5 | 222 | ovs_be16 *p; |
cf62fa4c | 223 | char *l2_5 = dp_packet_l2_5(packet); |
437d0d22 | 224 | |
db5a1019 | 225 | p = ALIGNED_CAST(ovs_be16 *, |
cf62fa4c | 226 | (l2_5 ? l2_5 : (char *)dp_packet_l3(packet)) - 2); |
b02475c5 SH |
227 | *p = eth_type; |
228 | } else { | |
229 | eh->eth_type = eth_type; | |
230 | } | |
231 | } | |
232 | ||
cf62fa4c | 233 | static bool is_mpls(struct dp_packet *packet) |
b02475c5 | 234 | { |
437d0d22 | 235 | return packet->l2_5_ofs != UINT16_MAX; |
b02475c5 SH |
236 | } |
237 | ||
238 | /* Set time to live (TTL) of an MPLS label stack entry (LSE). */ | |
b676167a | 239 | void |
b02475c5 SH |
240 | set_mpls_lse_ttl(ovs_be32 *lse, uint8_t ttl) |
241 | { | |
242 | *lse &= ~htonl(MPLS_TTL_MASK); | |
243 | *lse |= htonl((ttl << MPLS_TTL_SHIFT) & MPLS_TTL_MASK); | |
244 | } | |
245 | ||
246 | /* Set traffic class (TC) of an MPLS label stack entry (LSE). */ | |
247 | void | |
248 | set_mpls_lse_tc(ovs_be32 *lse, uint8_t tc) | |
249 | { | |
250 | *lse &= ~htonl(MPLS_TC_MASK); | |
251 | *lse |= htonl((tc << MPLS_TC_SHIFT) & MPLS_TC_MASK); | |
252 | } | |
253 | ||
254 | /* Set label of an MPLS label stack entry (LSE). */ | |
255 | void | |
256 | set_mpls_lse_label(ovs_be32 *lse, ovs_be32 label) | |
257 | { | |
258 | *lse &= ~htonl(MPLS_LABEL_MASK); | |
259 | *lse |= htonl((ntohl(label) << MPLS_LABEL_SHIFT) & MPLS_LABEL_MASK); | |
260 | } | |
261 | ||
262 | /* Set bottom of stack (BoS) bit of an MPLS label stack entry (LSE). */ | |
263 | void | |
264 | set_mpls_lse_bos(ovs_be32 *lse, uint8_t bos) | |
265 | { | |
266 | *lse &= ~htonl(MPLS_BOS_MASK); | |
267 | *lse |= htonl((bos << MPLS_BOS_SHIFT) & MPLS_BOS_MASK); | |
268 | } | |
269 | ||
270 | /* Compose an MPLS label stack entry (LSE) from its components: | |
271 | * label, traffic class (TC), time to live (TTL) and | |
272 | * bottom of stack (BoS) bit. */ | |
273 | ovs_be32 | |
274 | set_mpls_lse_values(uint8_t ttl, uint8_t tc, uint8_t bos, ovs_be32 label) | |
275 | { | |
276 | ovs_be32 lse = htonl(0); | |
277 | set_mpls_lse_ttl(&lse, ttl); | |
278 | set_mpls_lse_tc(&lse, tc); | |
279 | set_mpls_lse_bos(&lse, bos); | |
280 | set_mpls_lse_label(&lse, label); | |
281 | return lse; | |
282 | } | |
283 | ||
b02475c5 SH |
284 | /* Set MPLS label stack entry to outermost MPLS header.*/ |
285 | void | |
cf62fa4c | 286 | set_mpls_lse(struct dp_packet *packet, ovs_be32 mpls_lse) |
b02475c5 | 287 | { |
b02475c5 SH |
288 | /* Packet type should be MPLS to set label stack entry. */ |
289 | if (is_mpls(packet)) { | |
cf62fa4c | 290 | struct mpls_hdr *mh = dp_packet_l2_5(packet); |
437d0d22 | 291 | |
b02475c5 | 292 | /* Update mpls label stack entry. */ |
5fa008d4 | 293 | put_16aligned_be32(&mh->mpls_lse, mpls_lse); |
b02475c5 SH |
294 | } |
295 | } | |
296 | ||
898dcef1 | 297 | /* Push MPLS label stack entry 'lse' onto 'packet' as the outermost MPLS |
b02475c5 SH |
298 | * header. If 'packet' does not already have any MPLS labels, then its |
299 | * Ethertype is changed to 'ethtype' (which must be an MPLS Ethertype). */ | |
300 | void | |
cf62fa4c | 301 | push_mpls(struct dp_packet *packet, ovs_be16 ethtype, ovs_be32 lse) |
b02475c5 | 302 | { |
437d0d22 JR |
303 | char * header; |
304 | size_t len; | |
b02475c5 SH |
305 | |
306 | if (!eth_type_mpls(ethtype)) { | |
307 | return; | |
308 | } | |
309 | ||
310 | if (!is_mpls(packet)) { | |
437d0d22 JR |
311 | /* Set MPLS label stack offset. */ |
312 | packet->l2_5_ofs = packet->l3_ofs; | |
b02475c5 SH |
313 | } |
314 | ||
437d0d22 JR |
315 | set_ethertype(packet, ethtype); |
316 | ||
b02475c5 | 317 | /* Push new MPLS shim header onto packet. */ |
437d0d22 | 318 | len = packet->l2_5_ofs; |
cf62fa4c | 319 | header = dp_packet_resize_l2_5(packet, MPLS_HLEN); |
437d0d22 JR |
320 | memmove(header, header + MPLS_HLEN, len); |
321 | memcpy(header + len, &lse, sizeof lse); | |
b02475c5 SH |
322 | } |
323 | ||
324 | /* If 'packet' is an MPLS packet, removes its outermost MPLS label stack entry. | |
325 | * If the label that was removed was the only MPLS label, changes 'packet''s | |
326 | * Ethertype to 'ethtype' (which ordinarily should not be an MPLS | |
327 | * Ethertype). */ | |
328 | void | |
cf62fa4c | 329 | pop_mpls(struct dp_packet *packet, ovs_be16 ethtype) |
b02475c5 | 330 | { |
b02475c5 | 331 | if (is_mpls(packet)) { |
cf62fa4c | 332 | struct mpls_hdr *mh = dp_packet_l2_5(packet); |
437d0d22 JR |
333 | size_t len = packet->l2_5_ofs; |
334 | ||
799a91bb | 335 | set_ethertype(packet, ethtype); |
5fa008d4 | 336 | if (get_16aligned_be32(&mh->mpls_lse) & htonl(MPLS_BOS_MASK)) { |
cf62fa4c | 337 | dp_packet_set_l2_5(packet, NULL); |
b02475c5 SH |
338 | } |
339 | /* Shift the l2 header forward. */ | |
cf62fa4c PS |
340 | memmove((char*)dp_packet_data(packet) + MPLS_HLEN, dp_packet_data(packet), len); |
341 | dp_packet_resize_l2_5(packet, -MPLS_HLEN); | |
b02475c5 SH |
342 | } |
343 | } | |
344 | ||
e22f1753 BP |
345 | /* Converts hex digits in 'hex' to an Ethernet packet in '*packetp'. The |
346 | * caller must free '*packetp'. On success, returns NULL. On failure, returns | |
bb622f82 BP |
347 | * an error message and stores NULL in '*packetp'. |
348 | * | |
349 | * Aligns the L3 header of '*packetp' on a 32-bit boundary. */ | |
e22f1753 | 350 | const char * |
cf62fa4c | 351 | eth_from_hex(const char *hex, struct dp_packet **packetp) |
e22f1753 | 352 | { |
cf62fa4c | 353 | struct dp_packet *packet; |
e22f1753 | 354 | |
bb622f82 | 355 | /* Use 2 bytes of headroom to 32-bit align the L3 header. */ |
cf62fa4c | 356 | packet = *packetp = dp_packet_new_with_headroom(strlen(hex) / 2, 2); |
e22f1753 | 357 | |
cf62fa4c PS |
358 | if (dp_packet_put_hex(packet, hex, NULL)[0] != '\0') { |
359 | dp_packet_delete(packet); | |
e22f1753 BP |
360 | *packetp = NULL; |
361 | return "Trailing garbage in packet data"; | |
362 | } | |
363 | ||
cf62fa4c PS |
364 | if (dp_packet_size(packet) < ETH_HEADER_LEN) { |
365 | dp_packet_delete(packet); | |
e22f1753 BP |
366 | *packetp = NULL; |
367 | return "Packet data too short for Ethernet"; | |
368 | } | |
369 | ||
370 | return NULL; | |
371 | } | |
372 | ||
3b4d8ad3 | 373 | void |
74ff3298 JR |
374 | eth_format_masked(const struct eth_addr eth, |
375 | const struct eth_addr *mask, struct ds *s) | |
3b4d8ad3 JS |
376 | { |
377 | ds_put_format(s, ETH_ADDR_FMT, ETH_ADDR_ARGS(eth)); | |
74ff3298 JR |
378 | if (mask && !eth_mask_is_exact(*mask)) { |
379 | ds_put_format(s, "/"ETH_ADDR_FMT, ETH_ADDR_ARGS(*mask)); | |
3b4d8ad3 JS |
380 | } |
381 | } | |
382 | ||
aad29cd1 | 383 | /* Given the IP netmask 'netmask', returns the number of bits of the IP address |
c08201d6 BP |
384 | * that it specifies, that is, the number of 1-bits in 'netmask'. |
385 | * | |
386 | * If 'netmask' is not a CIDR netmask (see ip_is_cidr()), the return value will | |
387 | * still be in the valid range but isn't otherwise meaningful. */ | |
aad29cd1 BP |
388 | int |
389 | ip_count_cidr_bits(ovs_be32 netmask) | |
390 | { | |
d578065e | 391 | return 32 - ctz32(ntohl(netmask)); |
aad29cd1 BP |
392 | } |
393 | ||
394 | void | |
395 | ip_format_masked(ovs_be32 ip, ovs_be32 mask, struct ds *s) | |
396 | { | |
ed36537e | 397 | ds_put_format(s, IP_FMT, IP_ARGS(ip)); |
b8266395 | 398 | if (mask != OVS_BE32_MAX) { |
aad29cd1 BP |
399 | if (ip_is_cidr(mask)) { |
400 | ds_put_format(s, "/%d", ip_count_cidr_bits(mask)); | |
401 | } else { | |
ed36537e | 402 | ds_put_format(s, "/"IP_FMT, IP_ARGS(mask)); |
aad29cd1 BP |
403 | } |
404 | } | |
405 | } | |
406 | ||
407 | ||
d31f1109 JP |
408 | /* Stores the string representation of the IPv6 address 'addr' into the |
409 | * character array 'addr_str', which must be at least INET6_ADDRSTRLEN | |
410 | * bytes long. */ | |
411 | void | |
412 | format_ipv6_addr(char *addr_str, const struct in6_addr *addr) | |
413 | { | |
414 | inet_ntop(AF_INET6, addr, addr_str, INET6_ADDRSTRLEN); | |
415 | } | |
416 | ||
417 | void | |
418 | print_ipv6_addr(struct ds *string, const struct in6_addr *addr) | |
419 | { | |
aad29cd1 BP |
420 | char *dst; |
421 | ||
422 | ds_reserve(string, string->length + INET6_ADDRSTRLEN); | |
423 | ||
424 | dst = string->string + string->length; | |
425 | format_ipv6_addr(dst, addr); | |
426 | string->length += strlen(dst); | |
427 | } | |
d31f1109 | 428 | |
964a4d5f TLSC |
429 | void |
430 | print_ipv6_mapped(struct ds *s, const struct in6_addr *addr) | |
431 | { | |
432 | if (IN6_IS_ADDR_V4MAPPED(addr)) { | |
433 | ds_put_format(s, IP_FMT, addr->s6_addr[12], addr->s6_addr[13], | |
434 | addr->s6_addr[14], addr->s6_addr[15]); | |
435 | } else { | |
436 | print_ipv6_addr(s, addr); | |
437 | } | |
438 | } | |
439 | ||
aad29cd1 BP |
440 | void |
441 | print_ipv6_masked(struct ds *s, const struct in6_addr *addr, | |
442 | const struct in6_addr *mask) | |
443 | { | |
444 | print_ipv6_addr(s, addr); | |
445 | if (mask && !ipv6_mask_is_exact(mask)) { | |
446 | if (ipv6_is_cidr(mask)) { | |
447 | int cidr_bits = ipv6_count_cidr_bits(mask); | |
448 | ds_put_format(s, "/%d", cidr_bits); | |
449 | } else { | |
450 | ds_put_char(s, '/'); | |
451 | print_ipv6_addr(s, mask); | |
452 | } | |
453 | } | |
d31f1109 JP |
454 | } |
455 | ||
456 | struct in6_addr ipv6_addr_bitand(const struct in6_addr *a, | |
457 | const struct in6_addr *b) | |
458 | { | |
459 | int i; | |
460 | struct in6_addr dst; | |
461 | ||
462 | #ifdef s6_addr32 | |
463 | for (i=0; i<4; i++) { | |
464 | dst.s6_addr32[i] = a->s6_addr32[i] & b->s6_addr32[i]; | |
465 | } | |
466 | #else | |
467 | for (i=0; i<16; i++) { | |
468 | dst.s6_addr[i] = a->s6_addr[i] & b->s6_addr[i]; | |
469 | } | |
470 | #endif | |
471 | ||
472 | return dst; | |
473 | } | |
474 | ||
475 | /* Returns an in6_addr consisting of 'mask' high-order 1-bits and 128-N | |
476 | * low-order 0-bits. */ | |
477 | struct in6_addr | |
478 | ipv6_create_mask(int mask) | |
479 | { | |
480 | struct in6_addr netmask; | |
481 | uint8_t *netmaskp = &netmask.s6_addr[0]; | |
482 | ||
483 | memset(&netmask, 0, sizeof netmask); | |
484 | while (mask > 8) { | |
485 | *netmaskp = 0xff; | |
486 | netmaskp++; | |
487 | mask -= 8; | |
488 | } | |
489 | ||
490 | if (mask) { | |
491 | *netmaskp = 0xff << (8 - mask); | |
492 | } | |
493 | ||
494 | return netmask; | |
495 | } | |
496 | ||
aad29cd1 BP |
497 | /* Given the IPv6 netmask 'netmask', returns the number of bits of the IPv6 |
498 | * address that it specifies, that is, the number of 1-bits in 'netmask'. | |
ff0b06ee BP |
499 | * 'netmask' must be a CIDR netmask (see ipv6_is_cidr()). |
500 | * | |
501 | * If 'netmask' is not a CIDR netmask (see ipv6_is_cidr()), the return value | |
502 | * will still be in the valid range but isn't otherwise meaningful. */ | |
d31f1109 JP |
503 | int |
504 | ipv6_count_cidr_bits(const struct in6_addr *netmask) | |
505 | { | |
506 | int i; | |
507 | int count = 0; | |
508 | const uint8_t *netmaskp = &netmask->s6_addr[0]; | |
509 | ||
d31f1109 JP |
510 | for (i=0; i<16; i++) { |
511 | if (netmaskp[i] == 0xff) { | |
512 | count += 8; | |
513 | } else { | |
514 | uint8_t nm; | |
515 | ||
516 | for(nm = netmaskp[i]; nm; nm <<= 1) { | |
517 | count++; | |
518 | } | |
519 | break; | |
520 | } | |
521 | ||
522 | } | |
523 | ||
524 | return count; | |
525 | } | |
526 | ||
d31f1109 JP |
527 | /* Returns true if 'netmask' is a CIDR netmask, that is, if it consists of N |
528 | * high-order 1-bits and 128-N low-order 0-bits. */ | |
529 | bool | |
530 | ipv6_is_cidr(const struct in6_addr *netmask) | |
531 | { | |
532 | const uint8_t *netmaskp = &netmask->s6_addr[0]; | |
533 | int i; | |
534 | ||
535 | for (i=0; i<16; i++) { | |
536 | if (netmaskp[i] != 0xff) { | |
537 | uint8_t x = ~netmaskp[i]; | |
538 | if (x & (x + 1)) { | |
539 | return false; | |
540 | } | |
541 | while (++i < 16) { | |
542 | if (netmaskp[i]) { | |
543 | return false; | |
544 | } | |
545 | } | |
546 | } | |
547 | } | |
548 | ||
549 | return true; | |
550 | } | |
c25c91fd | 551 | |
5de1bb5c BP |
552 | /* Populates 'b' with an Ethernet II packet headed with the given 'eth_dst', |
553 | * 'eth_src' and 'eth_type' parameters. A payload of 'size' bytes is allocated | |
554 | * in 'b' and returned. This payload may be populated with appropriate | |
cf3b7538 JR |
555 | * information by the caller. Sets 'b''s 'frame' pointer and 'l3' offset to |
556 | * the Ethernet header and payload respectively. Aligns b->l3 on a 32-bit | |
bb622f82 | 557 | * boundary. |
eda1f38d BP |
558 | * |
559 | * The returned packet has enough headroom to insert an 802.1Q VLAN header if | |
560 | * desired. */ | |
40f78b38 | 561 | void * |
74ff3298 JR |
562 | eth_compose(struct dp_packet *b, const struct eth_addr eth_dst, |
563 | const struct eth_addr eth_src, uint16_t eth_type, | |
5de1bb5c | 564 | size_t size) |
c25c91fd | 565 | { |
40f78b38 | 566 | void *data; |
c25c91fd | 567 | struct eth_header *eth; |
c25c91fd | 568 | |
cf62fa4c | 569 | dp_packet_clear(b); |
c25c91fd | 570 | |
bb622f82 BP |
571 | /* The magic 2 here ensures that the L3 header (when it is added later) |
572 | * will be 32-bit aligned. */ | |
cf62fa4c PS |
573 | dp_packet_prealloc_tailroom(b, 2 + ETH_HEADER_LEN + VLAN_HEADER_LEN + size); |
574 | dp_packet_reserve(b, 2 + VLAN_HEADER_LEN); | |
575 | eth = dp_packet_put_uninit(b, ETH_HEADER_LEN); | |
576 | data = dp_packet_put_uninit(b, size); | |
c25c91fd | 577 | |
74ff3298 JR |
578 | eth->eth_dst = eth_dst; |
579 | eth->eth_src = eth_src; | |
40f78b38 EJ |
580 | eth->eth_type = htons(eth_type); |
581 | ||
82eb5b0a | 582 | dp_packet_reset_offsets(b); |
cf62fa4c | 583 | dp_packet_set_l3(b, data); |
75a4ead1 | 584 | |
40f78b38 | 585 | return data; |
07a6cf77 EJ |
586 | } |
587 | ||
c97664b3 | 588 | static void |
cf62fa4c | 589 | packet_set_ipv4_addr(struct dp_packet *packet, |
7c457c33 | 590 | ovs_16aligned_be32 *addr, ovs_be32 new_addr) |
c97664b3 | 591 | { |
cf62fa4c | 592 | struct ip_header *nh = dp_packet_l3(packet); |
7c457c33 | 593 | ovs_be32 old_addr = get_16aligned_be32(addr); |
cf62fa4c | 594 | size_t l4_size = dp_packet_l4_size(packet); |
c97664b3 | 595 | |
5a51b2cd | 596 | if (nh->ip_proto == IPPROTO_TCP && l4_size >= TCP_HEADER_LEN) { |
cf62fa4c | 597 | struct tcp_header *th = dp_packet_l4(packet); |
c97664b3 | 598 | |
7c457c33 | 599 | th->tcp_csum = recalc_csum32(th->tcp_csum, old_addr, new_addr); |
5a51b2cd | 600 | } else if (nh->ip_proto == IPPROTO_UDP && l4_size >= UDP_HEADER_LEN ) { |
cf62fa4c | 601 | struct udp_header *uh = dp_packet_l4(packet); |
c97664b3 EJ |
602 | |
603 | if (uh->udp_csum) { | |
7c457c33 | 604 | uh->udp_csum = recalc_csum32(uh->udp_csum, old_addr, new_addr); |
c97664b3 EJ |
605 | if (!uh->udp_csum) { |
606 | uh->udp_csum = htons(0xffff); | |
607 | } | |
608 | } | |
609 | } | |
7c457c33 BP |
610 | nh->ip_csum = recalc_csum32(nh->ip_csum, old_addr, new_addr); |
611 | put_16aligned_be32(addr, new_addr); | |
c97664b3 EJ |
612 | } |
613 | ||
bc7a5acd AA |
614 | /* Returns true, if packet contains at least one routing header where |
615 | * segements_left > 0. | |
616 | * | |
437d0d22 | 617 | * This function assumes that L3 and L4 offsets are set in the packet. */ |
bc7a5acd | 618 | static bool |
cf62fa4c | 619 | packet_rh_present(struct dp_packet *packet) |
bc7a5acd | 620 | { |
4528f34f | 621 | const struct ovs_16aligned_ip6_hdr *nh; |
bc7a5acd AA |
622 | int nexthdr; |
623 | size_t len; | |
624 | size_t remaining; | |
cf62fa4c | 625 | uint8_t *data = dp_packet_l3(packet); |
bc7a5acd | 626 | |
437d0d22 | 627 | remaining = packet->l4_ofs - packet->l3_ofs; |
bc7a5acd AA |
628 | |
629 | if (remaining < sizeof *nh) { | |
630 | return false; | |
631 | } | |
4528f34f | 632 | nh = ALIGNED_CAST(struct ovs_16aligned_ip6_hdr *, data); |
bc7a5acd AA |
633 | data += sizeof *nh; |
634 | remaining -= sizeof *nh; | |
635 | nexthdr = nh->ip6_nxt; | |
636 | ||
637 | while (1) { | |
638 | if ((nexthdr != IPPROTO_HOPOPTS) | |
639 | && (nexthdr != IPPROTO_ROUTING) | |
640 | && (nexthdr != IPPROTO_DSTOPTS) | |
641 | && (nexthdr != IPPROTO_AH) | |
642 | && (nexthdr != IPPROTO_FRAGMENT)) { | |
643 | /* It's either a terminal header (e.g., TCP, UDP) or one we | |
644 | * don't understand. In either case, we're done with the | |
645 | * packet, so use it to fill in 'nw_proto'. */ | |
646 | break; | |
647 | } | |
648 | ||
649 | /* We only verify that at least 8 bytes of the next header are | |
650 | * available, but many of these headers are longer. Ensure that | |
651 | * accesses within the extension header are within those first 8 | |
652 | * bytes. All extension headers are required to be at least 8 | |
653 | * bytes. */ | |
654 | if (remaining < 8) { | |
655 | return false; | |
656 | } | |
657 | ||
658 | if (nexthdr == IPPROTO_AH) { | |
659 | /* A standard AH definition isn't available, but the fields | |
660 | * we care about are in the same location as the generic | |
661 | * option header--only the header length is calculated | |
662 | * differently. */ | |
663 | const struct ip6_ext *ext_hdr = (struct ip6_ext *)data; | |
664 | ||
665 | nexthdr = ext_hdr->ip6e_nxt; | |
666 | len = (ext_hdr->ip6e_len + 2) * 4; | |
667 | } else if (nexthdr == IPPROTO_FRAGMENT) { | |
4528f34f BP |
668 | const struct ovs_16aligned_ip6_frag *frag_hdr |
669 | = ALIGNED_CAST(struct ovs_16aligned_ip6_frag *, data); | |
bc7a5acd AA |
670 | |
671 | nexthdr = frag_hdr->ip6f_nxt; | |
672 | len = sizeof *frag_hdr; | |
673 | } else if (nexthdr == IPPROTO_ROUTING) { | |
674 | const struct ip6_rthdr *rh = (struct ip6_rthdr *)data; | |
675 | ||
676 | if (rh->ip6r_segleft > 0) { | |
677 | return true; | |
678 | } | |
679 | ||
680 | nexthdr = rh->ip6r_nxt; | |
681 | len = (rh->ip6r_len + 1) * 8; | |
682 | } else { | |
683 | const struct ip6_ext *ext_hdr = (struct ip6_ext *)data; | |
684 | ||
685 | nexthdr = ext_hdr->ip6e_nxt; | |
686 | len = (ext_hdr->ip6e_len + 1) * 8; | |
687 | } | |
688 | ||
689 | if (remaining < len) { | |
690 | return false; | |
691 | } | |
692 | remaining -= len; | |
693 | data += len; | |
694 | } | |
695 | ||
696 | return false; | |
697 | } | |
698 | ||
699 | static void | |
cf62fa4c | 700 | packet_update_csum128(struct dp_packet *packet, uint8_t proto, |
4528f34f | 701 | ovs_16aligned_be32 addr[4], const ovs_be32 new_addr[4]) |
bc7a5acd | 702 | { |
cf62fa4c | 703 | size_t l4_size = dp_packet_l4_size(packet); |
5a51b2cd JR |
704 | |
705 | if (proto == IPPROTO_TCP && l4_size >= TCP_HEADER_LEN) { | |
cf62fa4c | 706 | struct tcp_header *th = dp_packet_l4(packet); |
bc7a5acd AA |
707 | |
708 | th->tcp_csum = recalc_csum128(th->tcp_csum, addr, new_addr); | |
5a51b2cd | 709 | } else if (proto == IPPROTO_UDP && l4_size >= UDP_HEADER_LEN) { |
cf62fa4c | 710 | struct udp_header *uh = dp_packet_l4(packet); |
bc7a5acd AA |
711 | |
712 | if (uh->udp_csum) { | |
713 | uh->udp_csum = recalc_csum128(uh->udp_csum, addr, new_addr); | |
714 | if (!uh->udp_csum) { | |
715 | uh->udp_csum = htons(0xffff); | |
716 | } | |
717 | } | |
5abf65d0 JG |
718 | } else if (proto == IPPROTO_ICMPV6 && |
719 | l4_size >= sizeof(struct icmp6_header)) { | |
cf62fa4c | 720 | struct icmp6_header *icmp = dp_packet_l4(packet); |
00894212 JG |
721 | |
722 | icmp->icmp6_cksum = recalc_csum128(icmp->icmp6_cksum, addr, new_addr); | |
bc7a5acd AA |
723 | } |
724 | } | |
725 | ||
726 | static void | |
cf62fa4c | 727 | packet_set_ipv6_addr(struct dp_packet *packet, uint8_t proto, |
4068403a | 728 | ovs_16aligned_be32 addr[4], const ovs_be32 new_addr[4], |
bc7a5acd AA |
729 | bool recalculate_csum) |
730 | { | |
731 | if (recalculate_csum) { | |
4528f34f | 732 | packet_update_csum128(packet, proto, addr, new_addr); |
bc7a5acd | 733 | } |
4068403a | 734 | memcpy(addr, new_addr, sizeof(ovs_be32[4])); |
bc7a5acd AA |
735 | } |
736 | ||
737 | static void | |
4528f34f | 738 | packet_set_ipv6_flow_label(ovs_16aligned_be32 *flow_label, ovs_be32 flow_key) |
bc7a5acd | 739 | { |
4528f34f BP |
740 | ovs_be32 old_label = get_16aligned_be32(flow_label); |
741 | ovs_be32 new_label = (old_label & htonl(~IPV6_LABEL_MASK)) | flow_key; | |
742 | put_16aligned_be32(flow_label, new_label); | |
bc7a5acd AA |
743 | } |
744 | ||
745 | static void | |
4528f34f | 746 | packet_set_ipv6_tc(ovs_16aligned_be32 *flow_label, uint8_t tc) |
bc7a5acd | 747 | { |
4528f34f BP |
748 | ovs_be32 old_label = get_16aligned_be32(flow_label); |
749 | ovs_be32 new_label = (old_label & htonl(0xF00FFFFF)) | htonl(tc << 20); | |
750 | put_16aligned_be32(flow_label, new_label); | |
bc7a5acd AA |
751 | } |
752 | ||
c97664b3 EJ |
753 | /* Modifies the IPv4 header fields of 'packet' to be consistent with 'src', |
754 | * 'dst', 'tos', and 'ttl'. Updates 'packet''s L4 checksums as appropriate. | |
755 | * 'packet' must contain a valid IPv4 packet with correctly populated l[347] | |
756 | * markers. */ | |
757 | void | |
cf62fa4c | 758 | packet_set_ipv4(struct dp_packet *packet, ovs_be32 src, ovs_be32 dst, |
c97664b3 EJ |
759 | uint8_t tos, uint8_t ttl) |
760 | { | |
cf62fa4c | 761 | struct ip_header *nh = dp_packet_l3(packet); |
c97664b3 | 762 | |
7c457c33 | 763 | if (get_16aligned_be32(&nh->ip_src) != src) { |
c97664b3 EJ |
764 | packet_set_ipv4_addr(packet, &nh->ip_src, src); |
765 | } | |
766 | ||
7c457c33 | 767 | if (get_16aligned_be32(&nh->ip_dst) != dst) { |
c97664b3 EJ |
768 | packet_set_ipv4_addr(packet, &nh->ip_dst, dst); |
769 | } | |
770 | ||
771 | if (nh->ip_tos != tos) { | |
772 | uint8_t *field = &nh->ip_tos; | |
773 | ||
774 | nh->ip_csum = recalc_csum16(nh->ip_csum, htons((uint16_t) *field), | |
775 | htons((uint16_t) tos)); | |
776 | *field = tos; | |
777 | } | |
778 | ||
779 | if (nh->ip_ttl != ttl) { | |
780 | uint8_t *field = &nh->ip_ttl; | |
781 | ||
782 | nh->ip_csum = recalc_csum16(nh->ip_csum, htons(*field << 8), | |
783 | htons(ttl << 8)); | |
784 | *field = ttl; | |
785 | } | |
786 | } | |
787 | ||
bc7a5acd AA |
788 | /* Modifies the IPv6 header fields of 'packet' to be consistent with 'src', |
789 | * 'dst', 'traffic class', and 'next hop'. Updates 'packet''s L4 checksums as | |
790 | * appropriate. 'packet' must contain a valid IPv6 packet with correctly | |
437d0d22 | 791 | * populated l[34] offsets. */ |
bc7a5acd | 792 | void |
cf62fa4c | 793 | packet_set_ipv6(struct dp_packet *packet, uint8_t proto, const ovs_be32 src[4], |
bc7a5acd AA |
794 | const ovs_be32 dst[4], uint8_t key_tc, ovs_be32 key_fl, |
795 | uint8_t key_hl) | |
796 | { | |
cf62fa4c | 797 | struct ovs_16aligned_ip6_hdr *nh = dp_packet_l3(packet); |
bc7a5acd AA |
798 | |
799 | if (memcmp(&nh->ip6_src, src, sizeof(ovs_be32[4]))) { | |
4528f34f | 800 | packet_set_ipv6_addr(packet, proto, nh->ip6_src.be32, src, true); |
bc7a5acd AA |
801 | } |
802 | ||
803 | if (memcmp(&nh->ip6_dst, dst, sizeof(ovs_be32[4]))) { | |
4528f34f | 804 | packet_set_ipv6_addr(packet, proto, nh->ip6_dst.be32, dst, |
bc7a5acd AA |
805 | !packet_rh_present(packet)); |
806 | } | |
807 | ||
808 | packet_set_ipv6_tc(&nh->ip6_flow, key_tc); | |
809 | ||
810 | packet_set_ipv6_flow_label(&nh->ip6_flow, key_fl); | |
811 | ||
812 | nh->ip6_hlim = key_hl; | |
813 | } | |
814 | ||
c97664b3 EJ |
815 | static void |
816 | packet_set_port(ovs_be16 *port, ovs_be16 new_port, ovs_be16 *csum) | |
817 | { | |
818 | if (*port != new_port) { | |
819 | *csum = recalc_csum16(*csum, *port, new_port); | |
820 | *port = new_port; | |
821 | } | |
822 | } | |
823 | ||
824 | /* Sets the TCP source and destination port ('src' and 'dst' respectively) of | |
825 | * the TCP header contained in 'packet'. 'packet' must be a valid TCP packet | |
437d0d22 | 826 | * with its l4 offset properly populated. */ |
c97664b3 | 827 | void |
cf62fa4c | 828 | packet_set_tcp_port(struct dp_packet *packet, ovs_be16 src, ovs_be16 dst) |
c97664b3 | 829 | { |
cf62fa4c | 830 | struct tcp_header *th = dp_packet_l4(packet); |
c97664b3 EJ |
831 | |
832 | packet_set_port(&th->tcp_src, src, &th->tcp_csum); | |
833 | packet_set_port(&th->tcp_dst, dst, &th->tcp_csum); | |
834 | } | |
835 | ||
836 | /* Sets the UDP source and destination port ('src' and 'dst' respectively) of | |
837 | * the UDP header contained in 'packet'. 'packet' must be a valid UDP packet | |
437d0d22 | 838 | * with its l4 offset properly populated. */ |
c97664b3 | 839 | void |
cf62fa4c | 840 | packet_set_udp_port(struct dp_packet *packet, ovs_be16 src, ovs_be16 dst) |
c97664b3 | 841 | { |
cf62fa4c | 842 | struct udp_header *uh = dp_packet_l4(packet); |
c97664b3 EJ |
843 | |
844 | if (uh->udp_csum) { | |
845 | packet_set_port(&uh->udp_src, src, &uh->udp_csum); | |
846 | packet_set_port(&uh->udp_dst, dst, &uh->udp_csum); | |
847 | ||
848 | if (!uh->udp_csum) { | |
849 | uh->udp_csum = htons(0xffff); | |
850 | } | |
851 | } else { | |
852 | uh->udp_src = src; | |
853 | uh->udp_dst = dst; | |
854 | } | |
855 | } | |
12113c39 | 856 | |
c6bcb685 JS |
857 | /* Sets the SCTP source and destination port ('src' and 'dst' respectively) of |
858 | * the SCTP header contained in 'packet'. 'packet' must be a valid SCTP packet | |
437d0d22 | 859 | * with its l4 offset properly populated. */ |
c6bcb685 | 860 | void |
cf62fa4c | 861 | packet_set_sctp_port(struct dp_packet *packet, ovs_be16 src, ovs_be16 dst) |
c6bcb685 | 862 | { |
cf62fa4c | 863 | struct sctp_header *sh = dp_packet_l4(packet); |
c6bcb685 | 864 | ovs_be32 old_csum, old_correct_csum, new_csum; |
cf62fa4c | 865 | uint16_t tp_len = dp_packet_l4_size(packet); |
c6bcb685 | 866 | |
5fa008d4 BP |
867 | old_csum = get_16aligned_be32(&sh->sctp_csum); |
868 | put_16aligned_be32(&sh->sctp_csum, 0); | |
437d0d22 | 869 | old_correct_csum = crc32c((void *)sh, tp_len); |
c6bcb685 JS |
870 | |
871 | sh->sctp_src = src; | |
872 | sh->sctp_dst = dst; | |
873 | ||
437d0d22 | 874 | new_csum = crc32c((void *)sh, tp_len); |
5fa008d4 | 875 | put_16aligned_be32(&sh->sctp_csum, old_csum ^ old_correct_csum ^ new_csum); |
c6bcb685 JS |
876 | } |
877 | ||
e60e935b | 878 | void |
cf62fa4c | 879 | packet_set_nd(struct dp_packet *packet, const ovs_be32 target[4], |
74ff3298 | 880 | const struct eth_addr sll, const struct eth_addr tll) { |
e60e935b SRCSA |
881 | struct ovs_nd_msg *ns; |
882 | struct ovs_nd_opt *nd_opt; | |
cf62fa4c | 883 | int bytes_remain = dp_packet_l4_size(packet); |
e60e935b SRCSA |
884 | |
885 | if (OVS_UNLIKELY(bytes_remain < sizeof(*ns))) { | |
886 | return; | |
887 | } | |
888 | ||
cf62fa4c | 889 | ns = dp_packet_l4(packet); |
e60e935b SRCSA |
890 | nd_opt = &ns->options[0]; |
891 | bytes_remain -= sizeof(*ns); | |
892 | ||
893 | if (memcmp(&ns->target, target, sizeof(ovs_be32[4]))) { | |
894 | packet_set_ipv6_addr(packet, IPPROTO_ICMPV6, | |
895 | ns->target.be32, | |
896 | target, true); | |
897 | } | |
898 | ||
899 | while (bytes_remain >= ND_OPT_LEN && nd_opt->nd_opt_len != 0) { | |
900 | if (nd_opt->nd_opt_type == ND_OPT_SOURCE_LINKADDR | |
901 | && nd_opt->nd_opt_len == 1) { | |
74ff3298 | 902 | if (!eth_addr_equals(nd_opt->nd_opt_mac, sll)) { |
e60e935b SRCSA |
903 | ovs_be16 *csum = &(ns->icmph.icmp6_cksum); |
904 | ||
74ff3298 JR |
905 | *csum = recalc_csum48(*csum, nd_opt->nd_opt_mac, sll); |
906 | nd_opt->nd_opt_mac = sll; | |
e60e935b SRCSA |
907 | } |
908 | ||
909 | /* A packet can only contain one SLL or TLL option */ | |
910 | break; | |
911 | } else if (nd_opt->nd_opt_type == ND_OPT_TARGET_LINKADDR | |
912 | && nd_opt->nd_opt_len == 1) { | |
74ff3298 | 913 | if (!eth_addr_equals(nd_opt->nd_opt_mac, tll)) { |
e60e935b SRCSA |
914 | ovs_be16 *csum = &(ns->icmph.icmp6_cksum); |
915 | ||
74ff3298 JR |
916 | *csum = recalc_csum48(*csum, nd_opt->nd_opt_mac, tll); |
917 | nd_opt->nd_opt_mac = tll; | |
e60e935b SRCSA |
918 | } |
919 | ||
920 | /* A packet can only contain one SLL or TLL option */ | |
921 | break; | |
922 | } | |
923 | ||
924 | nd_opt += nd_opt->nd_opt_len; | |
925 | bytes_remain -= nd_opt->nd_opt_len * ND_OPT_LEN; | |
926 | } | |
927 | } | |
928 | ||
61bf6666 JR |
929 | const char * |
930 | packet_tcp_flag_to_string(uint32_t flag) | |
931 | { | |
932 | switch (flag) { | |
933 | case TCP_FIN: | |
934 | return "fin"; | |
935 | case TCP_SYN: | |
936 | return "syn"; | |
937 | case TCP_RST: | |
938 | return "rst"; | |
939 | case TCP_PSH: | |
940 | return "psh"; | |
941 | case TCP_ACK: | |
942 | return "ack"; | |
943 | case TCP_URG: | |
944 | return "urg"; | |
945 | case TCP_ECE: | |
946 | return "ece"; | |
947 | case TCP_CWR: | |
948 | return "cwr"; | |
949 | case TCP_NS: | |
950 | return "ns"; | |
951 | case 0x200: | |
952 | return "[200]"; | |
953 | case 0x400: | |
954 | return "[400]"; | |
955 | case 0x800: | |
956 | return "[800]"; | |
957 | default: | |
958 | return NULL; | |
959 | } | |
960 | } | |
961 | ||
7393104d | 962 | /* Appends a string representation of the TCP flags value 'tcp_flags' |
f41b5b3b | 963 | * (e.g. from struct flow.tcp_flags or obtained via TCP_FLAGS) to 's', in the |
7393104d BP |
964 | * format used by tcpdump. */ |
965 | void | |
a66733a8 | 966 | packet_format_tcp_flags(struct ds *s, uint16_t tcp_flags) |
7393104d BP |
967 | { |
968 | if (!tcp_flags) { | |
969 | ds_put_cstr(s, "none"); | |
970 | return; | |
971 | } | |
972 | ||
973 | if (tcp_flags & TCP_SYN) { | |
974 | ds_put_char(s, 'S'); | |
975 | } | |
976 | if (tcp_flags & TCP_FIN) { | |
977 | ds_put_char(s, 'F'); | |
978 | } | |
979 | if (tcp_flags & TCP_PSH) { | |
980 | ds_put_char(s, 'P'); | |
981 | } | |
982 | if (tcp_flags & TCP_RST) { | |
983 | ds_put_char(s, 'R'); | |
984 | } | |
985 | if (tcp_flags & TCP_URG) { | |
986 | ds_put_char(s, 'U'); | |
987 | } | |
988 | if (tcp_flags & TCP_ACK) { | |
989 | ds_put_char(s, '.'); | |
990 | } | |
a66733a8 JR |
991 | if (tcp_flags & TCP_ECE) { |
992 | ds_put_cstr(s, "E"); | |
7393104d | 993 | } |
a66733a8 JR |
994 | if (tcp_flags & TCP_CWR) { |
995 | ds_put_cstr(s, "C"); | |
996 | } | |
997 | if (tcp_flags & TCP_NS) { | |
998 | ds_put_cstr(s, "N"); | |
999 | } | |
1000 | if (tcp_flags & 0x200) { | |
1001 | ds_put_cstr(s, "[200]"); | |
1002 | } | |
1003 | if (tcp_flags & 0x400) { | |
1004 | ds_put_cstr(s, "[400]"); | |
1005 | } | |
1006 | if (tcp_flags & 0x800) { | |
1007 | ds_put_cstr(s, "[800]"); | |
7393104d BP |
1008 | } |
1009 | } | |
a36de779 PS |
1010 | |
1011 | #define ARP_PACKET_SIZE (2 + ETH_HEADER_LEN + VLAN_HEADER_LEN + \ | |
1012 | ARP_ETH_HEADER_LEN) | |
1013 | ||
eb0b295e BP |
1014 | /* Clears 'b' and replaces its contents by an ARP frame with the specified |
1015 | * 'arp_op', 'arp_sha', 'arp_tha', 'arp_spa', and 'arp_tpa'. The outer | |
1016 | * Ethernet frame is initialized with Ethernet source 'arp_sha' and destination | |
1017 | * 'arp_tha', except that destination ff:ff:ff:ff:ff:ff is used instead if | |
1018 | * 'broadcast' is true. */ | |
a36de779 | 1019 | void |
eb0b295e | 1020 | compose_arp(struct dp_packet *b, uint16_t arp_op, |
74ff3298 JR |
1021 | const struct eth_addr arp_sha, const struct eth_addr arp_tha, |
1022 | bool broadcast, ovs_be32 arp_spa, ovs_be32 arp_tpa) | |
a36de779 PS |
1023 | { |
1024 | struct eth_header *eth; | |
1025 | struct arp_eth_header *arp; | |
1026 | ||
cf62fa4c PS |
1027 | dp_packet_clear(b); |
1028 | dp_packet_prealloc_tailroom(b, ARP_PACKET_SIZE); | |
1029 | dp_packet_reserve(b, 2 + VLAN_HEADER_LEN); | |
a36de779 | 1030 | |
cf62fa4c | 1031 | eth = dp_packet_put_uninit(b, sizeof *eth); |
74ff3298 JR |
1032 | eth->eth_dst = broadcast ? eth_addr_broadcast : arp_tha; |
1033 | eth->eth_src = arp_sha; | |
a36de779 PS |
1034 | eth->eth_type = htons(ETH_TYPE_ARP); |
1035 | ||
cf62fa4c | 1036 | arp = dp_packet_put_uninit(b, sizeof *arp); |
a36de779 PS |
1037 | arp->ar_hrd = htons(ARP_HRD_ETHERNET); |
1038 | arp->ar_pro = htons(ARP_PRO_IP); | |
1039 | arp->ar_hln = sizeof arp->ar_sha; | |
1040 | arp->ar_pln = sizeof arp->ar_spa; | |
eb0b295e | 1041 | arp->ar_op = htons(arp_op); |
74ff3298 JR |
1042 | arp->ar_sha = arp_sha; |
1043 | arp->ar_tha = arp_tha; | |
a36de779 | 1044 | |
eb0b295e BP |
1045 | put_16aligned_be32(&arp->ar_spa, arp_spa); |
1046 | put_16aligned_be32(&arp->ar_tpa, arp_tpa); | |
a36de779 | 1047 | |
82eb5b0a | 1048 | dp_packet_reset_offsets(b); |
cf62fa4c | 1049 | dp_packet_set_l3(b, arp); |
a36de779 | 1050 | } |
0292a0c9 JG |
1051 | |
1052 | uint32_t | |
1053 | packet_csum_pseudoheader(const struct ip_header *ip) | |
1054 | { | |
1055 | uint32_t partial = 0; | |
1056 | ||
1057 | partial = csum_add32(partial, get_16aligned_be32(&ip->ip_src)); | |
1058 | partial = csum_add32(partial, get_16aligned_be32(&ip->ip_dst)); | |
1059 | partial = csum_add16(partial, htons(ip->ip_proto)); | |
1060 | partial = csum_add16(partial, htons(ntohs(ip->ip_tot_len) - | |
1061 | IP_IHL(ip->ip_ihl_ver) * 4)); | |
1062 | ||
1063 | return partial; | |
1064 | } | |
07659514 | 1065 |