X-Git-Url: https://pintos-os.org/cgi-bin/gitweb.cgi?a=blobdiff_plain;f=lib%2Fpackets.c;h=8fb7f6b0320fac2a7fabcb02f6bd820420a91e8a;hb=8017806940cfeaba963d6a6f51ae7573cbea6792;hp=b9f37bbb25aad436e68fbf827435d4b8ea18ee97;hpb=e22f17535efae89cf02e4fe7e6ef65507115a684;p=openvswitch diff --git a/lib/packets.c b/lib/packets.c index b9f37bbb..8fb7f6b0 100644 --- a/lib/packets.c +++ b/lib/packets.c @@ -1,5 +1,5 @@ /* - * Copyright (c) 2009, 2010, 2011 Nicira Networks. + * Copyright (c) 2009, 2010, 2011, 2012 Nicira Networks. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,6 +22,8 @@ #include #include #include "byte-order.h" +#include "csum.h" +#include "flow.h" #include "dynamic-string.h" #include "ofpbuf.h" @@ -77,7 +79,7 @@ compose_benign_packet(struct ofpbuf *b, const char *tag, uint16_t snap_type, } /* Insert VLAN header according to given TCI. Packet passed must be Ethernet - * packet. + * packet. Ignores the CFI bit of 'tci' using 0 instead. * * Also sets 'packet->l2' to point to the new Ethernet header. */ void @@ -91,7 +93,7 @@ eth_push_vlan(struct ofpbuf *packet, ovs_be16 tci) memcpy(tmp.veth_dst, eh->eth_dst, ETH_ADDR_LEN); memcpy(tmp.veth_src, eh->eth_src, ETH_ADDR_LEN); tmp.veth_type = htons(ETH_TYPE_VLAN); - tmp.veth_tci = tci; + tmp.veth_tci = tci & htons(~VLAN_CFI); tmp.veth_next_type = eh->eth_type; veh = ofpbuf_push_uninit(packet, VLAN_HEADER_LEN); @@ -377,3 +379,120 @@ snap_compose(struct ofpbuf *b, const uint8_t eth_dst[ETH_ADDR_LEN], return payload; } + +static void +packet_set_ipv4_addr(struct ofpbuf *packet, ovs_be32 *addr, ovs_be32 new_addr) +{ + struct ip_header *nh = packet->l3; + + if (nh->ip_proto == IPPROTO_TCP && packet->l7) { + struct tcp_header *th = packet->l4; + + th->tcp_csum = recalc_csum32(th->tcp_csum, *addr, new_addr); + } else if (nh->ip_proto == IPPROTO_UDP && packet->l7) { + struct udp_header *uh = packet->l4; + + if (uh->udp_csum) { + uh->udp_csum = recalc_csum32(uh->udp_csum, *addr, new_addr); + if (!uh->udp_csum) { + uh->udp_csum = htons(0xffff); + } + } + } + nh->ip_csum = recalc_csum32(nh->ip_csum, *addr, new_addr); + *addr = new_addr; +} + +/* Modifies the IPv4 header fields of 'packet' to be consistent with 'src', + * 'dst', 'tos', and 'ttl'. Updates 'packet''s L4 checksums as appropriate. + * 'packet' must contain a valid IPv4 packet with correctly populated l[347] + * markers. */ +void +packet_set_ipv4(struct ofpbuf *packet, ovs_be32 src, ovs_be32 dst, + uint8_t tos, uint8_t ttl) +{ + struct ip_header *nh = packet->l3; + + if (nh->ip_src != src) { + packet_set_ipv4_addr(packet, &nh->ip_src, src); + } + + if (nh->ip_dst != dst) { + packet_set_ipv4_addr(packet, &nh->ip_dst, dst); + } + + if (nh->ip_tos != tos) { + uint8_t *field = &nh->ip_tos; + + nh->ip_csum = recalc_csum16(nh->ip_csum, htons((uint16_t) *field), + htons((uint16_t) tos)); + *field = tos; + } + + if (nh->ip_ttl != ttl) { + uint8_t *field = &nh->ip_ttl; + + nh->ip_csum = recalc_csum16(nh->ip_csum, htons(*field << 8), + htons(ttl << 8)); + *field = ttl; + } +} + +static void +packet_set_port(ovs_be16 *port, ovs_be16 new_port, ovs_be16 *csum) +{ + if (*port != new_port) { + *csum = recalc_csum16(*csum, *port, new_port); + *port = new_port; + } +} + +/* Sets the TCP source and destination port ('src' and 'dst' respectively) of + * the TCP header contained in 'packet'. 'packet' must be a valid TCP packet + * with its l4 marker properly populated. */ +void +packet_set_tcp_port(struct ofpbuf *packet, ovs_be16 src, ovs_be16 dst) +{ + struct tcp_header *th = packet->l4; + + packet_set_port(&th->tcp_src, src, &th->tcp_csum); + packet_set_port(&th->tcp_dst, dst, &th->tcp_csum); +} + +/* Sets the UDP source and destination port ('src' and 'dst' respectively) of + * the UDP header contained in 'packet'. 'packet' must be a valid UDP packet + * with its l4 marker properly populated. */ +void +packet_set_udp_port(struct ofpbuf *packet, ovs_be16 src, ovs_be16 dst) +{ + struct udp_header *uh = packet->l4; + + if (uh->udp_csum) { + packet_set_port(&uh->udp_src, src, &uh->udp_csum); + packet_set_port(&uh->udp_dst, dst, &uh->udp_csum); + + if (!uh->udp_csum) { + uh->udp_csum = htons(0xffff); + } + } else { + uh->udp_src = src; + uh->udp_dst = dst; + } +} + +/* If 'packet' is a TCP packet, returns the TCP flags. Otherwise, returns 0. + * + * 'flow' must be the flow corresponding to 'packet' and 'packet''s header + * pointers must be properly initialized (e.g. with flow_extract()). */ +uint8_t +packet_get_tcp_flags(const struct ofpbuf *packet, const struct flow *flow) +{ + if ((flow->dl_type == htons(ETH_TYPE_IP) || + flow->dl_type == htons(ETH_TYPE_IPV6)) && + flow->nw_proto == IPPROTO_TCP && packet->l7) { + const struct tcp_header *tcp = packet->l4; + return TCP_FLAGS(tcp->tcp_ctl); + } else { + return 0; + } +}