| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2009-11-09 16:43:47 -08:00
										 |  |  |  * Copyright (c) 2008, 2009, 2010 Nicira Networks. | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-06-15 15:11:30 -07:00
										 |  |  |  * Licensed under the Apache License, Version 2.0 (the "License"); | 
					
						
							|  |  |  |  * you may not use this file except in compliance with the License. | 
					
						
							|  |  |  |  * You may obtain a copy of the License at: | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-06-15 15:11:30 -07:00
										 |  |  |  *     http://www.apache.org/licenses/LICENSE-2.0
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Unless required by applicable law or agreed to in writing, software | 
					
						
							|  |  |  |  * distributed under the License is distributed on an "AS IS" BASIS, | 
					
						
							|  |  |  |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
					
						
							|  |  |  |  * See the License for the specific language governing permissions and | 
					
						
							|  |  |  |  * limitations under the License. | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  */ | 
					
						
							|  |  |  | #ifndef FLOW_H
 | 
					
						
							|  |  |  | #define FLOW_H 1
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-12 12:51:36 -08:00
										 |  |  | #include <sys/types.h>
 | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | #include <netinet/in.h>
 | 
					
						
							|  |  |  | #include <stdbool.h>
 | 
					
						
							|  |  |  | #include <stdint.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							| 
									
										
										
										
											2010-04-12 11:49:16 -04:00
										 |  |  | #include "openflow/nicira-ext.h"
 | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | #include "openflow/openflow.h"
 | 
					
						
							|  |  |  | #include "hash.h"
 | 
					
						
							|  |  |  | #include "openvswitch/datapath-protocol.h"
 | 
					
						
							|  |  |  | #include "util.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ds; | 
					
						
							| 
									
										
										
										
											2010-11-08 10:37:35 -08:00
										 |  |  | struct flow_wildcards; | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | struct ofp_match; | 
					
						
							|  |  |  | struct ofpbuf; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-11 10:41:33 -08:00
										 |  |  | #define FLOW_N_REGS 3
 | 
					
						
							|  |  |  | BUILD_ASSERT_DECL(FLOW_N_REGS <= NXM_NX_MAX_REGS); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-11 13:31:35 -07:00
										 |  |  | struct flow { | 
					
						
							| 
									
										
										
										
											2010-11-11 10:41:33 -08:00
										 |  |  |     uint32_t regs[FLOW_N_REGS]; /* Registers. */ | 
					
						
							| 
									
										
										
										
											2010-10-26 15:24:26 -07:00
										 |  |  |     ovs_be32 tun_id;            /* Encapsulating tunnel ID. */ | 
					
						
							|  |  |  |     ovs_be32 nw_src;            /* IP source address. */ | 
					
						
							|  |  |  |     ovs_be32 nw_dst;            /* IP destination address. */ | 
					
						
							| 
									
										
										
										
											2010-10-11 13:31:35 -07:00
										 |  |  |     uint16_t in_port;           /* Input switch port. */ | 
					
						
							| 
									
										
										
										
											2010-10-26 15:24:26 -07:00
										 |  |  |     ovs_be16 dl_vlan;           /* Input VLAN. */ | 
					
						
							|  |  |  |     ovs_be16 dl_type;           /* Ethernet frame type. */ | 
					
						
							|  |  |  |     ovs_be16 tp_src;            /* TCP/UDP source port. */ | 
					
						
							|  |  |  |     ovs_be16 tp_dst;            /* TCP/UDP destination port. */ | 
					
						
							| 
									
										
										
										
											2010-10-11 13:31:35 -07:00
										 |  |  |     uint8_t dl_src[6];          /* Ethernet source address. */ | 
					
						
							|  |  |  |     uint8_t dl_dst[6];          /* Ethernet destination address. */ | 
					
						
							|  |  |  |     uint8_t nw_proto;           /* IP protocol or low 8 bits of ARP opcode. */ | 
					
						
							|  |  |  |     uint8_t dl_vlan_pcp;        /* Input VLAN priority. */ | 
					
						
							|  |  |  |     uint8_t nw_tos;             /* IP ToS (DSCP field, 6 bits). */ | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Assert that there are FLOW_SIG_SIZE bytes of significant data in "struct
 | 
					
						
							|  |  |  |  * flow", followed by FLOW_PAD_SIZE bytes of padding. */ | 
					
						
							| 
									
										
										
										
											2010-11-11 10:41:33 -08:00
										 |  |  | #define FLOW_SIG_SIZE (37 + FLOW_N_REGS * 4)
 | 
					
						
							| 
									
										
										
										
											2010-10-11 13:31:35 -07:00
										 |  |  | #define FLOW_PAD_SIZE 3
 | 
					
						
							|  |  |  | BUILD_ASSERT_DECL(offsetof(struct flow, nw_tos) == FLOW_SIG_SIZE - 1); | 
					
						
							|  |  |  | BUILD_ASSERT_DECL(sizeof(((struct flow *)0)->nw_tos) == 1); | 
					
						
							|  |  |  | BUILD_ASSERT_DECL(sizeof(struct flow) == FLOW_SIG_SIZE + FLOW_PAD_SIZE); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-26 15:24:26 -07:00
										 |  |  | int flow_extract(struct ofpbuf *, ovs_be32 tun_id, uint16_t in_port, | 
					
						
							| 
									
										
										
										
											2010-09-03 11:30:02 -07:00
										 |  |  |                  struct flow *); | 
					
						
							|  |  |  | void flow_extract_stats(const struct flow *flow, struct ofpbuf *packet, | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |         struct odp_flow_stats *stats); | 
					
						
							| 
									
										
										
										
											2010-11-05 11:10:35 -07:00
										 |  |  | void flow_to_match(const struct flow *, uint32_t wildcards, int flow_format, | 
					
						
							| 
									
										
										
										
											2010-04-12 11:49:16 -04:00
										 |  |  |                    struct ofp_match *); | 
					
						
							| 
									
										
										
										
											2010-11-05 11:10:35 -07:00
										 |  |  | void flow_from_match(const struct ofp_match *, int flow_format, | 
					
						
							| 
									
										
										
										
											2010-11-08 10:37:35 -08:00
										 |  |  |                      ovs_be64 cookie, struct flow *, struct flow_wildcards *); | 
					
						
							| 
									
										
										
										
											2010-09-03 11:30:02 -07:00
										 |  |  | char *flow_to_string(const struct flow *); | 
					
						
							|  |  |  | void flow_format(struct ds *, const struct flow *); | 
					
						
							|  |  |  | void flow_print(FILE *, const struct flow *); | 
					
						
							|  |  |  | static inline int flow_compare(const struct flow *, const struct flow *); | 
					
						
							|  |  |  | static inline bool flow_equal(const struct flow *, const struct flow *); | 
					
						
							|  |  |  | static inline size_t flow_hash(const struct flow *, uint32_t basis); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | static inline int | 
					
						
							| 
									
										
										
										
											2010-09-03 11:30:02 -07:00
										 |  |  | flow_compare(const struct flow *a, const struct flow *b) | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-10-11 13:31:35 -07:00
										 |  |  |     return memcmp(a, b, FLOW_SIG_SIZE); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline bool | 
					
						
							| 
									
										
										
										
											2010-09-03 11:30:02 -07:00
										 |  |  | flow_equal(const struct flow *a, const struct flow *b) | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | { | 
					
						
							|  |  |  |     return !flow_compare(a, b); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline size_t | 
					
						
							| 
									
										
										
										
											2010-09-03 11:30:02 -07:00
										 |  |  | flow_hash(const struct flow *flow, uint32_t basis) | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-10-11 13:31:35 -07:00
										 |  |  |     return hash_bytes(flow, FLOW_SIG_SIZE, basis); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-11 10:41:33 -08:00
										 |  |  | /* Set to 1 in the 'wildcards' member of struct flow_wildcards if any bits in
 | 
					
						
							|  |  |  |  * any of the reg_masks are wildcarded.  This maintains the invariant that | 
					
						
							|  |  |  |  * 'wildcards' is nonzero if and only if any bits are wildcarded. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This is used only internally to Open vSwitch--it never appears in the wire | 
					
						
							|  |  |  |  * protocol. */ | 
					
						
							|  |  |  | #define FWW_REGS (1u << 31)
 | 
					
						
							|  |  |  | BUILD_ASSERT_DECL(!(FWW_REGS & OVSFW_ALL)); /* Avoid collisions. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-03 11:00:58 -07:00
										 |  |  | /* Information on wildcards for a flow, as a supplement to "struct flow".
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The flow_wildcards_*() functions below both depend on and maintain the | 
					
						
							|  |  |  |  * following important invariants: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 1. 'wildcards' is nonzero if and only if at least one bit or field is | 
					
						
							|  |  |  |  *    wildcarded. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-11-11 10:41:33 -08:00
										 |  |  |  * 2. Bits in 'wildcards' not included in OVSFW_ALL or FWW_REGS are set to 0. | 
					
						
							|  |  |  |  *    (This is a corollary to invariant #1.) | 
					
						
							| 
									
										
										
										
											2010-11-03 11:00:58 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * 3. The fields in 'wildcards' masked by OFPFW_NW_SRC_MASK and | 
					
						
							|  |  |  |  *    OFPFW_NW_DST_MASK have values between 0 and 32, inclusive. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 4. The fields masked by OFPFW_NW_SRC_MASK and OFPFW_NW_DST_MASK correspond | 
					
						
							|  |  |  |  *    correctly to the masks in 'nw_src_mask' and 'nw_dst_mask', respectively. | 
					
						
							| 
									
										
										
										
											2010-11-11 10:41:33 -08:00
										 |  |  |  * | 
					
						
							|  |  |  |  * 5. FWW_REGS is set to 1 in 'wildcards' if and only if at least one bit in | 
					
						
							|  |  |  |  *    'reg_masks[]' is nonzero.  (This allows wildcarded 'reg_masks[]' to | 
					
						
							|  |  |  |  *    satisfy invariant #1.) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 6. If FWW_REGS is set to 0 in 'wildcards', then the values of all of the | 
					
						
							|  |  |  |  *    other members can be correctly predicted based on 'wildcards' alone. | 
					
						
							| 
									
										
										
										
											2010-11-03 11:00:58 -07:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | struct flow_wildcards { | 
					
						
							| 
									
										
										
										
											2010-10-26 15:24:26 -07:00
										 |  |  |     uint32_t wildcards;         /* enum ofp_flow_wildcards. */ | 
					
						
							| 
									
										
										
										
											2010-11-11 10:41:33 -08:00
										 |  |  |     uint32_t reg_masks[FLOW_N_REGS]; /* 1-bit in each significant regs bit. */ | 
					
						
							| 
									
										
										
										
											2010-10-26 15:24:26 -07:00
										 |  |  |     ovs_be32 nw_src_mask;       /* 1-bit in each significant nw_src bit. */ | 
					
						
							|  |  |  |     ovs_be32 nw_dst_mask;       /* 1-bit in each significant nw_dst bit. */ | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-20 16:33:10 -07:00
										 |  |  | ovs_be32 flow_nw_bits_to_mask(uint32_t wildcards, int shift); | 
					
						
							|  |  |  | void flow_wildcards_init(struct flow_wildcards *, uint32_t wildcards); | 
					
						
							| 
									
										
										
										
											2010-10-27 20:15:56 -07:00
										 |  |  | void flow_wildcards_init_exact(struct flow_wildcards *); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool flow_wildcards_set_nw_src_mask(struct flow_wildcards *, ovs_be32); | 
					
						
							|  |  |  | bool flow_wildcards_set_nw_dst_mask(struct flow_wildcards *, ovs_be32); | 
					
						
							| 
									
										
										
										
											2010-11-11 10:41:33 -08:00
										 |  |  | void flow_wildcards_set_reg_mask(struct flow_wildcards *, | 
					
						
							|  |  |  |                                  int idx, uint32_t mask); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-03 11:00:58 -07:00
										 |  |  | void flow_wildcards_combine(struct flow_wildcards *dst, | 
					
						
							|  |  |  |                             const struct flow_wildcards *src1, | 
					
						
							|  |  |  |                             const struct flow_wildcards *src2); | 
					
						
							|  |  |  | bool flow_wildcards_has_extra(const struct flow_wildcards *, | 
					
						
							|  |  |  |                               const struct flow_wildcards *); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | uint32_t flow_wildcards_hash(const struct flow_wildcards *); | 
					
						
							|  |  |  | bool flow_wildcards_equal(const struct flow_wildcards *, | 
					
						
							|  |  |  |                           const struct flow_wildcards *); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | #endif /* flow.h */
 |