mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02:00 
			
		
		
		
	openvswitch: Add support for unique flow IDs.
Previously, flows were manipulated by userspace specifying a full, unmasked flow key. This adds significant burden onto flow serialization/deserialization, particularly when dumping flows. This patch adds an alternative way to refer to flows using a variable-length "unique flow identifier" (UFID). At flow setup time, userspace may specify a UFID for a flow, which is stored with the flow and inserted into a separate table for lookup, in addition to the standard flow table. Flows created using a UFID must be fetched or deleted using the UFID. All flow dump operations may now be made more terse with OVS_UFID_F_* flags. For example, the OVS_UFID_F_OMIT_KEY flag allows responses to omit the flow key from a datapath operation if the flow has a corresponding UFID. This significantly reduces the time spent assembling and transacting netlink messages. With all OVS_UFID_F_OMIT_* flags enabled, the datapath only returns the UFID and statistics for each flow during flow dump, increasing ovs-vswitchd revalidator performance by 40% or more. Signed-off-by: Joe Stringer <joestringer@nicira.com> Acked-by: Pravin B Shelar <pshelar@nicira.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
		
							parent
							
								
									7b1883cefc
								
							
						
					
					
						commit
						74ed7ab926
					
				
					 8 changed files with 448 additions and 91 deletions
				
			
		| 
						 | 
					@ -131,6 +131,19 @@ performs best-effort detection of overlapping wildcarded flows and may reject
 | 
				
			||||||
some but not all of them. However, this behavior may change in future versions.
 | 
					some but not all of them. However, this behavior may change in future versions.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Unique flow identifiers
 | 
				
			||||||
 | 
					-----------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					An alternative to using the original match portion of a key as the handle for
 | 
				
			||||||
 | 
					flow identification is a unique flow identifier, or "UFID". UFIDs are optional
 | 
				
			||||||
 | 
					for both the kernel and user space program.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					User space programs that support UFID are expected to provide it during flow
 | 
				
			||||||
 | 
					setup in addition to the flow, then refer to the flow using the UFID for all
 | 
				
			||||||
 | 
					future operations. The kernel is not required to index flows by the original
 | 
				
			||||||
 | 
					flow key if a UFID is specified.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
Basic rule for evolving flow keys
 | 
					Basic rule for evolving flow keys
 | 
				
			||||||
---------------------------------
 | 
					---------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -459,6 +459,14 @@ struct ovs_key_nd {
 | 
				
			||||||
 * a wildcarded match. Omitting attribute is treated as wildcarding all
 | 
					 * a wildcarded match. Omitting attribute is treated as wildcarding all
 | 
				
			||||||
 * corresponding fields. Optional for all requests. If not present,
 | 
					 * corresponding fields. Optional for all requests. If not present,
 | 
				
			||||||
 * all flow key bits are exact match bits.
 | 
					 * all flow key bits are exact match bits.
 | 
				
			||||||
 | 
					 * @OVS_FLOW_ATTR_UFID: A value between 1-16 octets specifying a unique
 | 
				
			||||||
 | 
					 * identifier for the flow. Causes the flow to be indexed by this value rather
 | 
				
			||||||
 | 
					 * than the value of the %OVS_FLOW_ATTR_KEY attribute. Optional for all
 | 
				
			||||||
 | 
					 * requests. Present in notifications if the flow was created with this
 | 
				
			||||||
 | 
					 * attribute.
 | 
				
			||||||
 | 
					 * @OVS_FLOW_ATTR_UFID_FLAGS: A 32-bit value of OR'd %OVS_UFID_F_*
 | 
				
			||||||
 | 
					 * flags that provide alternative semantics for flow installation and
 | 
				
			||||||
 | 
					 * retrieval. Optional for all requests.
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * These attributes follow the &struct ovs_header within the Generic Netlink
 | 
					 * These attributes follow the &struct ovs_header within the Generic Netlink
 | 
				
			||||||
 * payload for %OVS_FLOW_* commands.
 | 
					 * payload for %OVS_FLOW_* commands.
 | 
				
			||||||
| 
						 | 
					@ -474,11 +482,23 @@ enum ovs_flow_attr {
 | 
				
			||||||
	OVS_FLOW_ATTR_MASK,      /* Sequence of OVS_KEY_ATTR_* attributes. */
 | 
						OVS_FLOW_ATTR_MASK,      /* Sequence of OVS_KEY_ATTR_* attributes. */
 | 
				
			||||||
	OVS_FLOW_ATTR_PROBE,     /* Flow operation is a feature probe, error
 | 
						OVS_FLOW_ATTR_PROBE,     /* Flow operation is a feature probe, error
 | 
				
			||||||
				  * logging should be suppressed. */
 | 
									  * logging should be suppressed. */
 | 
				
			||||||
 | 
						OVS_FLOW_ATTR_UFID,      /* Variable length unique flow identifier. */
 | 
				
			||||||
 | 
						OVS_FLOW_ATTR_UFID_FLAGS,/* u32 of OVS_UFID_F_*. */
 | 
				
			||||||
	__OVS_FLOW_ATTR_MAX
 | 
						__OVS_FLOW_ATTR_MAX
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define OVS_FLOW_ATTR_MAX (__OVS_FLOW_ATTR_MAX - 1)
 | 
					#define OVS_FLOW_ATTR_MAX (__OVS_FLOW_ATTR_MAX - 1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * Omit attributes for notifications.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * If a datapath request contains an %OVS_UFID_F_OMIT_* flag, then the datapath
 | 
				
			||||||
 | 
					 * may omit the corresponding %OVS_FLOW_ATTR_* from the response.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					#define OVS_UFID_F_OMIT_KEY      (1 << 0)
 | 
				
			||||||
 | 
					#define OVS_UFID_F_OMIT_MASK     (1 << 1)
 | 
				
			||||||
 | 
					#define OVS_UFID_F_OMIT_ACTIONS  (1 << 2)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
 * enum ovs_sample_attr - Attributes for %OVS_ACTION_ATTR_SAMPLE action.
 | 
					 * enum ovs_sample_attr - Attributes for %OVS_ACTION_ATTR_SAMPLE action.
 | 
				
			||||||
 * @OVS_SAMPLE_ATTR_PROBABILITY: 32-bit fraction of packets to sample with
 | 
					 * @OVS_SAMPLE_ATTR_PROBABILITY: 32-bit fraction of packets to sample with
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -65,6 +65,8 @@ static struct genl_family dp_packet_genl_family;
 | 
				
			||||||
static struct genl_family dp_flow_genl_family;
 | 
					static struct genl_family dp_flow_genl_family;
 | 
				
			||||||
static struct genl_family dp_datapath_genl_family;
 | 
					static struct genl_family dp_datapath_genl_family;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static const struct nla_policy flow_policy[];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static const struct genl_multicast_group ovs_dp_flow_multicast_group = {
 | 
					static const struct genl_multicast_group ovs_dp_flow_multicast_group = {
 | 
				
			||||||
	.name = OVS_FLOW_MCGROUP,
 | 
						.name = OVS_FLOW_MCGROUP,
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
| 
						 | 
					@ -662,15 +664,48 @@ static void get_dp_stats(const struct datapath *dp, struct ovs_dp_stats *stats,
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static size_t ovs_flow_cmd_msg_size(const struct sw_flow_actions *acts)
 | 
					static bool should_fill_key(const struct sw_flow_id *sfid, uint32_t ufid_flags)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	return NLMSG_ALIGN(sizeof(struct ovs_header))
 | 
						return ovs_identifier_is_ufid(sfid) &&
 | 
				
			||||||
		+ nla_total_size(ovs_key_attr_size()) /* OVS_FLOW_ATTR_KEY */
 | 
						       !(ufid_flags & OVS_UFID_F_OMIT_KEY);
 | 
				
			||||||
		+ nla_total_size(ovs_key_attr_size()) /* OVS_FLOW_ATTR_MASK */
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static bool should_fill_mask(uint32_t ufid_flags)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						return !(ufid_flags & OVS_UFID_F_OMIT_MASK);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static bool should_fill_actions(uint32_t ufid_flags)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						return !(ufid_flags & OVS_UFID_F_OMIT_ACTIONS);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static size_t ovs_flow_cmd_msg_size(const struct sw_flow_actions *acts,
 | 
				
			||||||
 | 
									    const struct sw_flow_id *sfid,
 | 
				
			||||||
 | 
									    uint32_t ufid_flags)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						size_t len = NLMSG_ALIGN(sizeof(struct ovs_header));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* OVS_FLOW_ATTR_UFID */
 | 
				
			||||||
 | 
						if (sfid && ovs_identifier_is_ufid(sfid))
 | 
				
			||||||
 | 
							len += nla_total_size(sfid->ufid_len);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* OVS_FLOW_ATTR_KEY */
 | 
				
			||||||
 | 
						if (!sfid || should_fill_key(sfid, ufid_flags))
 | 
				
			||||||
 | 
							len += nla_total_size(ovs_key_attr_size());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* OVS_FLOW_ATTR_MASK */
 | 
				
			||||||
 | 
						if (should_fill_mask(ufid_flags))
 | 
				
			||||||
 | 
							len += nla_total_size(ovs_key_attr_size());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* OVS_FLOW_ATTR_ACTIONS */
 | 
				
			||||||
 | 
						if (should_fill_actions(ufid_flags))
 | 
				
			||||||
 | 
							len += nla_total_size(acts->actions_len);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return len
 | 
				
			||||||
		+ nla_total_size(sizeof(struct ovs_flow_stats)) /* OVS_FLOW_ATTR_STATS */
 | 
							+ nla_total_size(sizeof(struct ovs_flow_stats)) /* OVS_FLOW_ATTR_STATS */
 | 
				
			||||||
		+ nla_total_size(1) /* OVS_FLOW_ATTR_TCP_FLAGS */
 | 
							+ nla_total_size(1) /* OVS_FLOW_ATTR_TCP_FLAGS */
 | 
				
			||||||
		+ nla_total_size(8) /* OVS_FLOW_ATTR_USED */
 | 
							+ nla_total_size(8); /* OVS_FLOW_ATTR_USED */
 | 
				
			||||||
		+ nla_total_size(acts->actions_len); /* OVS_FLOW_ATTR_ACTIONS */
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Called with ovs_mutex or RCU read lock. */
 | 
					/* Called with ovs_mutex or RCU read lock. */
 | 
				
			||||||
| 
						 | 
					@ -741,7 +776,7 @@ static int ovs_flow_cmd_fill_actions(const struct sw_flow *flow,
 | 
				
			||||||
/* Called with ovs_mutex or RCU read lock. */
 | 
					/* Called with ovs_mutex or RCU read lock. */
 | 
				
			||||||
static int ovs_flow_cmd_fill_info(const struct sw_flow *flow, int dp_ifindex,
 | 
					static int ovs_flow_cmd_fill_info(const struct sw_flow *flow, int dp_ifindex,
 | 
				
			||||||
				  struct sk_buff *skb, u32 portid,
 | 
									  struct sk_buff *skb, u32 portid,
 | 
				
			||||||
				  u32 seq, u32 flags, u8 cmd)
 | 
									  u32 seq, u32 flags, u8 cmd, u32 ufid_flags)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	const int skb_orig_len = skb->len;
 | 
						const int skb_orig_len = skb->len;
 | 
				
			||||||
	struct ovs_header *ovs_header;
 | 
						struct ovs_header *ovs_header;
 | 
				
			||||||
| 
						 | 
					@ -754,21 +789,31 @@ static int ovs_flow_cmd_fill_info(const struct sw_flow *flow, int dp_ifindex,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	ovs_header->dp_ifindex = dp_ifindex;
 | 
						ovs_header->dp_ifindex = dp_ifindex;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	err = ovs_nla_put_unmasked_key(flow, skb);
 | 
						err = ovs_nla_put_identifier(flow, skb);
 | 
				
			||||||
	if (err)
 | 
						if (err)
 | 
				
			||||||
		goto error;
 | 
							goto error;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	err = ovs_nla_put_mask(flow, skb);
 | 
						if (should_fill_key(&flow->id, ufid_flags)) {
 | 
				
			||||||
	if (err)
 | 
							err = ovs_nla_put_masked_key(flow, skb);
 | 
				
			||||||
		goto error;
 | 
							if (err)
 | 
				
			||||||
 | 
								goto error;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (should_fill_mask(ufid_flags)) {
 | 
				
			||||||
 | 
							err = ovs_nla_put_mask(flow, skb);
 | 
				
			||||||
 | 
							if (err)
 | 
				
			||||||
 | 
								goto error;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	err = ovs_flow_cmd_fill_stats(flow, skb);
 | 
						err = ovs_flow_cmd_fill_stats(flow, skb);
 | 
				
			||||||
	if (err)
 | 
						if (err)
 | 
				
			||||||
		goto error;
 | 
							goto error;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	err = ovs_flow_cmd_fill_actions(flow, skb, skb_orig_len);
 | 
						if (should_fill_actions(ufid_flags)) {
 | 
				
			||||||
	if (err)
 | 
							err = ovs_flow_cmd_fill_actions(flow, skb, skb_orig_len);
 | 
				
			||||||
		goto error;
 | 
							if (err)
 | 
				
			||||||
 | 
								goto error;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	genlmsg_end(skb, ovs_header);
 | 
						genlmsg_end(skb, ovs_header);
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
| 
						 | 
					@ -780,15 +825,19 @@ static int ovs_flow_cmd_fill_info(const struct sw_flow *flow, int dp_ifindex,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* May not be called with RCU read lock. */
 | 
					/* May not be called with RCU read lock. */
 | 
				
			||||||
static struct sk_buff *ovs_flow_cmd_alloc_info(const struct sw_flow_actions *acts,
 | 
					static struct sk_buff *ovs_flow_cmd_alloc_info(const struct sw_flow_actions *acts,
 | 
				
			||||||
 | 
										       const struct sw_flow_id *sfid,
 | 
				
			||||||
					       struct genl_info *info,
 | 
										       struct genl_info *info,
 | 
				
			||||||
					       bool always)
 | 
										       bool always,
 | 
				
			||||||
 | 
										       uint32_t ufid_flags)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct sk_buff *skb;
 | 
						struct sk_buff *skb;
 | 
				
			||||||
 | 
						size_t len;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (!always && !ovs_must_notify(&dp_flow_genl_family, info, 0))
 | 
						if (!always && !ovs_must_notify(&dp_flow_genl_family, info, 0))
 | 
				
			||||||
		return NULL;
 | 
							return NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	skb = genlmsg_new_unicast(ovs_flow_cmd_msg_size(acts), info, GFP_KERNEL);
 | 
						len = ovs_flow_cmd_msg_size(acts, sfid, ufid_flags);
 | 
				
			||||||
 | 
						skb = genlmsg_new_unicast(len, info, GFP_KERNEL);
 | 
				
			||||||
	if (!skb)
 | 
						if (!skb)
 | 
				
			||||||
		return ERR_PTR(-ENOMEM);
 | 
							return ERR_PTR(-ENOMEM);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -799,19 +848,19 @@ static struct sk_buff *ovs_flow_cmd_alloc_info(const struct sw_flow_actions *act
 | 
				
			||||||
static struct sk_buff *ovs_flow_cmd_build_info(const struct sw_flow *flow,
 | 
					static struct sk_buff *ovs_flow_cmd_build_info(const struct sw_flow *flow,
 | 
				
			||||||
					       int dp_ifindex,
 | 
										       int dp_ifindex,
 | 
				
			||||||
					       struct genl_info *info, u8 cmd,
 | 
										       struct genl_info *info, u8 cmd,
 | 
				
			||||||
					       bool always)
 | 
										       bool always, u32 ufid_flags)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct sk_buff *skb;
 | 
						struct sk_buff *skb;
 | 
				
			||||||
	int retval;
 | 
						int retval;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	skb = ovs_flow_cmd_alloc_info(ovsl_dereference(flow->sf_acts), info,
 | 
						skb = ovs_flow_cmd_alloc_info(ovsl_dereference(flow->sf_acts),
 | 
				
			||||||
				      always);
 | 
									      &flow->id, info, always, ufid_flags);
 | 
				
			||||||
	if (IS_ERR_OR_NULL(skb))
 | 
						if (IS_ERR_OR_NULL(skb))
 | 
				
			||||||
		return skb;
 | 
							return skb;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	retval = ovs_flow_cmd_fill_info(flow, dp_ifindex, skb,
 | 
						retval = ovs_flow_cmd_fill_info(flow, dp_ifindex, skb,
 | 
				
			||||||
					info->snd_portid, info->snd_seq, 0,
 | 
										info->snd_portid, info->snd_seq, 0,
 | 
				
			||||||
					cmd);
 | 
										cmd, ufid_flags);
 | 
				
			||||||
	BUG_ON(retval < 0);
 | 
						BUG_ON(retval < 0);
 | 
				
			||||||
	return skb;
 | 
						return skb;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -820,12 +869,14 @@ static int ovs_flow_cmd_new(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct nlattr **a = info->attrs;
 | 
						struct nlattr **a = info->attrs;
 | 
				
			||||||
	struct ovs_header *ovs_header = info->userhdr;
 | 
						struct ovs_header *ovs_header = info->userhdr;
 | 
				
			||||||
	struct sw_flow *flow, *new_flow;
 | 
						struct sw_flow *flow = NULL, *new_flow;
 | 
				
			||||||
	struct sw_flow_mask mask;
 | 
						struct sw_flow_mask mask;
 | 
				
			||||||
	struct sk_buff *reply;
 | 
						struct sk_buff *reply;
 | 
				
			||||||
	struct datapath *dp;
 | 
						struct datapath *dp;
 | 
				
			||||||
 | 
						struct sw_flow_key key;
 | 
				
			||||||
	struct sw_flow_actions *acts;
 | 
						struct sw_flow_actions *acts;
 | 
				
			||||||
	struct sw_flow_match match;
 | 
						struct sw_flow_match match;
 | 
				
			||||||
 | 
						u32 ufid_flags = ovs_nla_get_ufid_flags(a[OVS_FLOW_ATTR_UFID_FLAGS]);
 | 
				
			||||||
	int error;
 | 
						int error;
 | 
				
			||||||
	bool log = !a[OVS_FLOW_ATTR_PROBE];
 | 
						bool log = !a[OVS_FLOW_ATTR_PROBE];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -850,13 +901,19 @@ static int ovs_flow_cmd_new(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* Extract key. */
 | 
						/* Extract key. */
 | 
				
			||||||
	ovs_match_init(&match, &new_flow->unmasked_key, &mask);
 | 
						ovs_match_init(&match, &key, &mask);
 | 
				
			||||||
	error = ovs_nla_get_match(&match, a[OVS_FLOW_ATTR_KEY],
 | 
						error = ovs_nla_get_match(&match, a[OVS_FLOW_ATTR_KEY],
 | 
				
			||||||
				  a[OVS_FLOW_ATTR_MASK], log);
 | 
									  a[OVS_FLOW_ATTR_MASK], log);
 | 
				
			||||||
	if (error)
 | 
						if (error)
 | 
				
			||||||
		goto err_kfree_flow;
 | 
							goto err_kfree_flow;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	ovs_flow_mask_key(&new_flow->key, &new_flow->unmasked_key, &mask);
 | 
						ovs_flow_mask_key(&new_flow->key, &key, &mask);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Extract flow identifier. */
 | 
				
			||||||
 | 
						error = ovs_nla_get_identifier(&new_flow->id, a[OVS_FLOW_ATTR_UFID],
 | 
				
			||||||
 | 
									       &key, log);
 | 
				
			||||||
 | 
						if (error)
 | 
				
			||||||
 | 
							goto err_kfree_flow;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* Validate actions. */
 | 
						/* Validate actions. */
 | 
				
			||||||
	error = ovs_nla_copy_actions(a[OVS_FLOW_ATTR_ACTIONS], &new_flow->key,
 | 
						error = ovs_nla_copy_actions(a[OVS_FLOW_ATTR_ACTIONS], &new_flow->key,
 | 
				
			||||||
| 
						 | 
					@ -866,7 +923,8 @@ static int ovs_flow_cmd_new(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
		goto err_kfree_flow;
 | 
							goto err_kfree_flow;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	reply = ovs_flow_cmd_alloc_info(acts, info, false);
 | 
						reply = ovs_flow_cmd_alloc_info(acts, &new_flow->id, info, false,
 | 
				
			||||||
 | 
										ufid_flags);
 | 
				
			||||||
	if (IS_ERR(reply)) {
 | 
						if (IS_ERR(reply)) {
 | 
				
			||||||
		error = PTR_ERR(reply);
 | 
							error = PTR_ERR(reply);
 | 
				
			||||||
		goto err_kfree_acts;
 | 
							goto err_kfree_acts;
 | 
				
			||||||
| 
						 | 
					@ -878,8 +936,12 @@ static int ovs_flow_cmd_new(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
		error = -ENODEV;
 | 
							error = -ENODEV;
 | 
				
			||||||
		goto err_unlock_ovs;
 | 
							goto err_unlock_ovs;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* Check if this is a duplicate flow */
 | 
						/* Check if this is a duplicate flow */
 | 
				
			||||||
	flow = ovs_flow_tbl_lookup(&dp->table, &new_flow->unmasked_key);
 | 
						if (ovs_identifier_is_ufid(&new_flow->id))
 | 
				
			||||||
 | 
							flow = ovs_flow_tbl_lookup_ufid(&dp->table, &new_flow->id);
 | 
				
			||||||
 | 
						if (!flow)
 | 
				
			||||||
 | 
							flow = ovs_flow_tbl_lookup(&dp->table, &key);
 | 
				
			||||||
	if (likely(!flow)) {
 | 
						if (likely(!flow)) {
 | 
				
			||||||
		rcu_assign_pointer(new_flow->sf_acts, acts);
 | 
							rcu_assign_pointer(new_flow->sf_acts, acts);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -895,7 +957,8 @@ static int ovs_flow_cmd_new(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
						       ovs_header->dp_ifindex,
 | 
											       ovs_header->dp_ifindex,
 | 
				
			||||||
						       reply, info->snd_portid,
 | 
											       reply, info->snd_portid,
 | 
				
			||||||
						       info->snd_seq, 0,
 | 
											       info->snd_seq, 0,
 | 
				
			||||||
						       OVS_FLOW_CMD_NEW);
 | 
											       OVS_FLOW_CMD_NEW,
 | 
				
			||||||
 | 
											       ufid_flags);
 | 
				
			||||||
			BUG_ON(error < 0);
 | 
								BUG_ON(error < 0);
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		ovs_unlock();
 | 
							ovs_unlock();
 | 
				
			||||||
| 
						 | 
					@ -913,10 +976,15 @@ static int ovs_flow_cmd_new(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
			error = -EEXIST;
 | 
								error = -EEXIST;
 | 
				
			||||||
			goto err_unlock_ovs;
 | 
								goto err_unlock_ovs;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		/* The unmasked key has to be the same for flow updates. */
 | 
							/* The flow identifier has to be the same for flow updates.
 | 
				
			||||||
		if (unlikely(!ovs_flow_cmp_unmasked_key(flow, &match))) {
 | 
							 * Look for any overlapping flow.
 | 
				
			||||||
			/* Look for any overlapping flow. */
 | 
							 */
 | 
				
			||||||
			flow = ovs_flow_tbl_lookup_exact(&dp->table, &match);
 | 
							if (unlikely(!ovs_flow_cmp(flow, &match))) {
 | 
				
			||||||
 | 
								if (ovs_identifier_is_key(&flow->id))
 | 
				
			||||||
 | 
									flow = ovs_flow_tbl_lookup_exact(&dp->table,
 | 
				
			||||||
 | 
													 &match);
 | 
				
			||||||
 | 
								else /* UFID matches but key is different */
 | 
				
			||||||
 | 
									flow = NULL;
 | 
				
			||||||
			if (!flow) {
 | 
								if (!flow) {
 | 
				
			||||||
				error = -ENOENT;
 | 
									error = -ENOENT;
 | 
				
			||||||
				goto err_unlock_ovs;
 | 
									goto err_unlock_ovs;
 | 
				
			||||||
| 
						 | 
					@ -931,7 +999,8 @@ static int ovs_flow_cmd_new(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
						       ovs_header->dp_ifindex,
 | 
											       ovs_header->dp_ifindex,
 | 
				
			||||||
						       reply, info->snd_portid,
 | 
											       reply, info->snd_portid,
 | 
				
			||||||
						       info->snd_seq, 0,
 | 
											       info->snd_seq, 0,
 | 
				
			||||||
						       OVS_FLOW_CMD_NEW);
 | 
											       OVS_FLOW_CMD_NEW,
 | 
				
			||||||
 | 
											       ufid_flags);
 | 
				
			||||||
			BUG_ON(error < 0);
 | 
								BUG_ON(error < 0);
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		ovs_unlock();
 | 
							ovs_unlock();
 | 
				
			||||||
| 
						 | 
					@ -987,8 +1056,11 @@ static int ovs_flow_cmd_set(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
	struct datapath *dp;
 | 
						struct datapath *dp;
 | 
				
			||||||
	struct sw_flow_actions *old_acts = NULL, *acts = NULL;
 | 
						struct sw_flow_actions *old_acts = NULL, *acts = NULL;
 | 
				
			||||||
	struct sw_flow_match match;
 | 
						struct sw_flow_match match;
 | 
				
			||||||
 | 
						struct sw_flow_id sfid;
 | 
				
			||||||
 | 
						u32 ufid_flags = ovs_nla_get_ufid_flags(a[OVS_FLOW_ATTR_UFID_FLAGS]);
 | 
				
			||||||
	int error;
 | 
						int error;
 | 
				
			||||||
	bool log = !a[OVS_FLOW_ATTR_PROBE];
 | 
						bool log = !a[OVS_FLOW_ATTR_PROBE];
 | 
				
			||||||
 | 
						bool ufid_present;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* Extract key. */
 | 
						/* Extract key. */
 | 
				
			||||||
	error = -EINVAL;
 | 
						error = -EINVAL;
 | 
				
			||||||
| 
						 | 
					@ -997,6 +1069,7 @@ static int ovs_flow_cmd_set(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
		goto error;
 | 
							goto error;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ufid_present = ovs_nla_get_ufid(&sfid, a[OVS_FLOW_ATTR_UFID], log);
 | 
				
			||||||
	ovs_match_init(&match, &key, &mask);
 | 
						ovs_match_init(&match, &key, &mask);
 | 
				
			||||||
	error = ovs_nla_get_match(&match, a[OVS_FLOW_ATTR_KEY],
 | 
						error = ovs_nla_get_match(&match, a[OVS_FLOW_ATTR_KEY],
 | 
				
			||||||
				  a[OVS_FLOW_ATTR_MASK], log);
 | 
									  a[OVS_FLOW_ATTR_MASK], log);
 | 
				
			||||||
| 
						 | 
					@ -1013,7 +1086,8 @@ static int ovs_flow_cmd_set(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		/* Can allocate before locking if have acts. */
 | 
							/* Can allocate before locking if have acts. */
 | 
				
			||||||
		reply = ovs_flow_cmd_alloc_info(acts, info, false);
 | 
							reply = ovs_flow_cmd_alloc_info(acts, &sfid, info, false,
 | 
				
			||||||
 | 
											ufid_flags);
 | 
				
			||||||
		if (IS_ERR(reply)) {
 | 
							if (IS_ERR(reply)) {
 | 
				
			||||||
			error = PTR_ERR(reply);
 | 
								error = PTR_ERR(reply);
 | 
				
			||||||
			goto err_kfree_acts;
 | 
								goto err_kfree_acts;
 | 
				
			||||||
| 
						 | 
					@ -1027,7 +1101,10 @@ static int ovs_flow_cmd_set(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
		goto err_unlock_ovs;
 | 
							goto err_unlock_ovs;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	/* Check that the flow exists. */
 | 
						/* Check that the flow exists. */
 | 
				
			||||||
	flow = ovs_flow_tbl_lookup_exact(&dp->table, &match);
 | 
						if (ufid_present)
 | 
				
			||||||
 | 
							flow = ovs_flow_tbl_lookup_ufid(&dp->table, &sfid);
 | 
				
			||||||
 | 
						else
 | 
				
			||||||
 | 
							flow = ovs_flow_tbl_lookup_exact(&dp->table, &match);
 | 
				
			||||||
	if (unlikely(!flow)) {
 | 
						if (unlikely(!flow)) {
 | 
				
			||||||
		error = -ENOENT;
 | 
							error = -ENOENT;
 | 
				
			||||||
		goto err_unlock_ovs;
 | 
							goto err_unlock_ovs;
 | 
				
			||||||
| 
						 | 
					@ -1043,13 +1120,16 @@ static int ovs_flow_cmd_set(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
						       ovs_header->dp_ifindex,
 | 
											       ovs_header->dp_ifindex,
 | 
				
			||||||
						       reply, info->snd_portid,
 | 
											       reply, info->snd_portid,
 | 
				
			||||||
						       info->snd_seq, 0,
 | 
											       info->snd_seq, 0,
 | 
				
			||||||
						       OVS_FLOW_CMD_NEW);
 | 
											       OVS_FLOW_CMD_NEW,
 | 
				
			||||||
 | 
											       ufid_flags);
 | 
				
			||||||
			BUG_ON(error < 0);
 | 
								BUG_ON(error < 0);
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	} else {
 | 
						} else {
 | 
				
			||||||
		/* Could not alloc without acts before locking. */
 | 
							/* Could not alloc without acts before locking. */
 | 
				
			||||||
		reply = ovs_flow_cmd_build_info(flow, ovs_header->dp_ifindex,
 | 
							reply = ovs_flow_cmd_build_info(flow, ovs_header->dp_ifindex,
 | 
				
			||||||
						info, OVS_FLOW_CMD_NEW, false);
 | 
											info, OVS_FLOW_CMD_NEW, false,
 | 
				
			||||||
 | 
											ufid_flags);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		if (unlikely(IS_ERR(reply))) {
 | 
							if (unlikely(IS_ERR(reply))) {
 | 
				
			||||||
			error = PTR_ERR(reply);
 | 
								error = PTR_ERR(reply);
 | 
				
			||||||
			goto err_unlock_ovs;
 | 
								goto err_unlock_ovs;
 | 
				
			||||||
| 
						 | 
					@ -1086,17 +1166,22 @@ static int ovs_flow_cmd_get(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
	struct sw_flow *flow;
 | 
						struct sw_flow *flow;
 | 
				
			||||||
	struct datapath *dp;
 | 
						struct datapath *dp;
 | 
				
			||||||
	struct sw_flow_match match;
 | 
						struct sw_flow_match match;
 | 
				
			||||||
	int err;
 | 
						struct sw_flow_id ufid;
 | 
				
			||||||
 | 
						u32 ufid_flags = ovs_nla_get_ufid_flags(a[OVS_FLOW_ATTR_UFID_FLAGS]);
 | 
				
			||||||
 | 
						int err = 0;
 | 
				
			||||||
	bool log = !a[OVS_FLOW_ATTR_PROBE];
 | 
						bool log = !a[OVS_FLOW_ATTR_PROBE];
 | 
				
			||||||
 | 
						bool ufid_present;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (!a[OVS_FLOW_ATTR_KEY]) {
 | 
						ufid_present = ovs_nla_get_ufid(&ufid, a[OVS_FLOW_ATTR_UFID], log);
 | 
				
			||||||
 | 
						if (a[OVS_FLOW_ATTR_KEY]) {
 | 
				
			||||||
 | 
							ovs_match_init(&match, &key, NULL);
 | 
				
			||||||
 | 
							err = ovs_nla_get_match(&match, a[OVS_FLOW_ATTR_KEY], NULL,
 | 
				
			||||||
 | 
										log);
 | 
				
			||||||
 | 
						} else if (!ufid_present) {
 | 
				
			||||||
		OVS_NLERR(log,
 | 
							OVS_NLERR(log,
 | 
				
			||||||
			  "Flow get message rejected, Key attribute missing.");
 | 
								  "Flow get message rejected, Key attribute missing.");
 | 
				
			||||||
		return -EINVAL;
 | 
							err = -EINVAL;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					 | 
				
			||||||
	ovs_match_init(&match, &key, NULL);
 | 
					 | 
				
			||||||
	err = ovs_nla_get_match(&match, a[OVS_FLOW_ATTR_KEY], NULL, log);
 | 
					 | 
				
			||||||
	if (err)
 | 
						if (err)
 | 
				
			||||||
		return err;
 | 
							return err;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1107,14 +1192,17 @@ static int ovs_flow_cmd_get(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
		goto unlock;
 | 
							goto unlock;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	flow = ovs_flow_tbl_lookup_exact(&dp->table, &match);
 | 
						if (ufid_present)
 | 
				
			||||||
 | 
							flow = ovs_flow_tbl_lookup_ufid(&dp->table, &ufid);
 | 
				
			||||||
 | 
						else
 | 
				
			||||||
 | 
							flow = ovs_flow_tbl_lookup_exact(&dp->table, &match);
 | 
				
			||||||
	if (!flow) {
 | 
						if (!flow) {
 | 
				
			||||||
		err = -ENOENT;
 | 
							err = -ENOENT;
 | 
				
			||||||
		goto unlock;
 | 
							goto unlock;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	reply = ovs_flow_cmd_build_info(flow, ovs_header->dp_ifindex, info,
 | 
						reply = ovs_flow_cmd_build_info(flow, ovs_header->dp_ifindex, info,
 | 
				
			||||||
					OVS_FLOW_CMD_NEW, true);
 | 
										OVS_FLOW_CMD_NEW, true, ufid_flags);
 | 
				
			||||||
	if (IS_ERR(reply)) {
 | 
						if (IS_ERR(reply)) {
 | 
				
			||||||
		err = PTR_ERR(reply);
 | 
							err = PTR_ERR(reply);
 | 
				
			||||||
		goto unlock;
 | 
							goto unlock;
 | 
				
			||||||
| 
						 | 
					@ -1133,13 +1221,17 @@ static int ovs_flow_cmd_del(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
	struct ovs_header *ovs_header = info->userhdr;
 | 
						struct ovs_header *ovs_header = info->userhdr;
 | 
				
			||||||
	struct sw_flow_key key;
 | 
						struct sw_flow_key key;
 | 
				
			||||||
	struct sk_buff *reply;
 | 
						struct sk_buff *reply;
 | 
				
			||||||
	struct sw_flow *flow;
 | 
						struct sw_flow *flow = NULL;
 | 
				
			||||||
	struct datapath *dp;
 | 
						struct datapath *dp;
 | 
				
			||||||
	struct sw_flow_match match;
 | 
						struct sw_flow_match match;
 | 
				
			||||||
 | 
						struct sw_flow_id ufid;
 | 
				
			||||||
 | 
						u32 ufid_flags = ovs_nla_get_ufid_flags(a[OVS_FLOW_ATTR_UFID_FLAGS]);
 | 
				
			||||||
	int err;
 | 
						int err;
 | 
				
			||||||
	bool log = !a[OVS_FLOW_ATTR_PROBE];
 | 
						bool log = !a[OVS_FLOW_ATTR_PROBE];
 | 
				
			||||||
 | 
						bool ufid_present;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (likely(a[OVS_FLOW_ATTR_KEY])) {
 | 
						ufid_present = ovs_nla_get_ufid(&ufid, a[OVS_FLOW_ATTR_UFID], log);
 | 
				
			||||||
 | 
						if (a[OVS_FLOW_ATTR_KEY]) {
 | 
				
			||||||
		ovs_match_init(&match, &key, NULL);
 | 
							ovs_match_init(&match, &key, NULL);
 | 
				
			||||||
		err = ovs_nla_get_match(&match, a[OVS_FLOW_ATTR_KEY], NULL,
 | 
							err = ovs_nla_get_match(&match, a[OVS_FLOW_ATTR_KEY], NULL,
 | 
				
			||||||
					log);
 | 
										log);
 | 
				
			||||||
| 
						 | 
					@ -1154,12 +1246,15 @@ static int ovs_flow_cmd_del(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
		goto unlock;
 | 
							goto unlock;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (unlikely(!a[OVS_FLOW_ATTR_KEY])) {
 | 
						if (unlikely(!a[OVS_FLOW_ATTR_KEY] && !ufid_present)) {
 | 
				
			||||||
		err = ovs_flow_tbl_flush(&dp->table);
 | 
							err = ovs_flow_tbl_flush(&dp->table);
 | 
				
			||||||
		goto unlock;
 | 
							goto unlock;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	flow = ovs_flow_tbl_lookup_exact(&dp->table, &match);
 | 
						if (ufid_present)
 | 
				
			||||||
 | 
							flow = ovs_flow_tbl_lookup_ufid(&dp->table, &ufid);
 | 
				
			||||||
 | 
						else
 | 
				
			||||||
 | 
							flow = ovs_flow_tbl_lookup_exact(&dp->table, &match);
 | 
				
			||||||
	if (unlikely(!flow)) {
 | 
						if (unlikely(!flow)) {
 | 
				
			||||||
		err = -ENOENT;
 | 
							err = -ENOENT;
 | 
				
			||||||
		goto unlock;
 | 
							goto unlock;
 | 
				
			||||||
| 
						 | 
					@ -1169,14 +1264,15 @@ static int ovs_flow_cmd_del(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
	ovs_unlock();
 | 
						ovs_unlock();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	reply = ovs_flow_cmd_alloc_info((const struct sw_flow_actions __force *) flow->sf_acts,
 | 
						reply = ovs_flow_cmd_alloc_info((const struct sw_flow_actions __force *) flow->sf_acts,
 | 
				
			||||||
					info, false);
 | 
										&flow->id, info, false, ufid_flags);
 | 
				
			||||||
	if (likely(reply)) {
 | 
						if (likely(reply)) {
 | 
				
			||||||
		if (likely(!IS_ERR(reply))) {
 | 
							if (likely(!IS_ERR(reply))) {
 | 
				
			||||||
			rcu_read_lock();	/*To keep RCU checker happy. */
 | 
								rcu_read_lock();	/*To keep RCU checker happy. */
 | 
				
			||||||
			err = ovs_flow_cmd_fill_info(flow, ovs_header->dp_ifindex,
 | 
								err = ovs_flow_cmd_fill_info(flow, ovs_header->dp_ifindex,
 | 
				
			||||||
						     reply, info->snd_portid,
 | 
											     reply, info->snd_portid,
 | 
				
			||||||
						     info->snd_seq, 0,
 | 
											     info->snd_seq, 0,
 | 
				
			||||||
						     OVS_FLOW_CMD_DEL);
 | 
											     OVS_FLOW_CMD_DEL,
 | 
				
			||||||
 | 
											     ufid_flags);
 | 
				
			||||||
			rcu_read_unlock();
 | 
								rcu_read_unlock();
 | 
				
			||||||
			BUG_ON(err < 0);
 | 
								BUG_ON(err < 0);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1195,9 +1291,18 @@ static int ovs_flow_cmd_del(struct sk_buff *skb, struct genl_info *info)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int ovs_flow_cmd_dump(struct sk_buff *skb, struct netlink_callback *cb)
 | 
					static int ovs_flow_cmd_dump(struct sk_buff *skb, struct netlink_callback *cb)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
 | 
						struct nlattr *a[__OVS_FLOW_ATTR_MAX];
 | 
				
			||||||
	struct ovs_header *ovs_header = genlmsg_data(nlmsg_data(cb->nlh));
 | 
						struct ovs_header *ovs_header = genlmsg_data(nlmsg_data(cb->nlh));
 | 
				
			||||||
	struct table_instance *ti;
 | 
						struct table_instance *ti;
 | 
				
			||||||
	struct datapath *dp;
 | 
						struct datapath *dp;
 | 
				
			||||||
 | 
						u32 ufid_flags;
 | 
				
			||||||
 | 
						int err;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						err = genlmsg_parse(cb->nlh, &dp_flow_genl_family, a,
 | 
				
			||||||
 | 
								    OVS_FLOW_ATTR_MAX, flow_policy);
 | 
				
			||||||
 | 
						if (err)
 | 
				
			||||||
 | 
							return err;
 | 
				
			||||||
 | 
						ufid_flags = ovs_nla_get_ufid_flags(a[OVS_FLOW_ATTR_UFID_FLAGS]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	rcu_read_lock();
 | 
						rcu_read_lock();
 | 
				
			||||||
	dp = get_dp_rcu(sock_net(skb->sk), ovs_header->dp_ifindex);
 | 
						dp = get_dp_rcu(sock_net(skb->sk), ovs_header->dp_ifindex);
 | 
				
			||||||
| 
						 | 
					@ -1220,7 +1325,7 @@ static int ovs_flow_cmd_dump(struct sk_buff *skb, struct netlink_callback *cb)
 | 
				
			||||||
		if (ovs_flow_cmd_fill_info(flow, ovs_header->dp_ifindex, skb,
 | 
							if (ovs_flow_cmd_fill_info(flow, ovs_header->dp_ifindex, skb,
 | 
				
			||||||
					   NETLINK_CB(cb->skb).portid,
 | 
										   NETLINK_CB(cb->skb).portid,
 | 
				
			||||||
					   cb->nlh->nlmsg_seq, NLM_F_MULTI,
 | 
										   cb->nlh->nlmsg_seq, NLM_F_MULTI,
 | 
				
			||||||
					   OVS_FLOW_CMD_NEW) < 0)
 | 
										   OVS_FLOW_CMD_NEW, ufid_flags) < 0)
 | 
				
			||||||
			break;
 | 
								break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		cb->args[0] = bucket;
 | 
							cb->args[0] = bucket;
 | 
				
			||||||
| 
						 | 
					@ -1236,6 +1341,8 @@ static const struct nla_policy flow_policy[OVS_FLOW_ATTR_MAX + 1] = {
 | 
				
			||||||
	[OVS_FLOW_ATTR_ACTIONS] = { .type = NLA_NESTED },
 | 
						[OVS_FLOW_ATTR_ACTIONS] = { .type = NLA_NESTED },
 | 
				
			||||||
	[OVS_FLOW_ATTR_CLEAR] = { .type = NLA_FLAG },
 | 
						[OVS_FLOW_ATTR_CLEAR] = { .type = NLA_FLAG },
 | 
				
			||||||
	[OVS_FLOW_ATTR_PROBE] = { .type = NLA_FLAG },
 | 
						[OVS_FLOW_ATTR_PROBE] = { .type = NLA_FLAG },
 | 
				
			||||||
 | 
						[OVS_FLOW_ATTR_UFID] = { .type = NLA_UNSPEC, .len = 1 },
 | 
				
			||||||
 | 
						[OVS_FLOW_ATTR_UFID_FLAGS] = { .type = NLA_U32 },
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static const struct genl_ops dp_flow_genl_ops[] = {
 | 
					static const struct genl_ops dp_flow_genl_ops[] = {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -197,6 +197,16 @@ struct sw_flow_match {
 | 
				
			||||||
	struct sw_flow_mask *mask;
 | 
						struct sw_flow_mask *mask;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define MAX_UFID_LENGTH 16 /* 128 bits */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					struct sw_flow_id {
 | 
				
			||||||
 | 
						u32 ufid_len;
 | 
				
			||||||
 | 
						union {
 | 
				
			||||||
 | 
							u32 ufid[MAX_UFID_LENGTH / 4];
 | 
				
			||||||
 | 
							struct sw_flow_key *unmasked_key;
 | 
				
			||||||
 | 
						};
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct sw_flow_actions {
 | 
					struct sw_flow_actions {
 | 
				
			||||||
	struct rcu_head rcu;
 | 
						struct rcu_head rcu;
 | 
				
			||||||
	u32 actions_len;
 | 
						u32 actions_len;
 | 
				
			||||||
| 
						 | 
					@ -213,13 +223,15 @@ struct flow_stats {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct sw_flow {
 | 
					struct sw_flow {
 | 
				
			||||||
	struct rcu_head rcu;
 | 
						struct rcu_head rcu;
 | 
				
			||||||
	struct hlist_node hash_node[2];
 | 
						struct {
 | 
				
			||||||
	u32 hash;
 | 
							struct hlist_node node[2];
 | 
				
			||||||
 | 
							u32 hash;
 | 
				
			||||||
 | 
						} flow_table, ufid_table;
 | 
				
			||||||
	int stats_last_writer;		/* NUMA-node id of the last writer on
 | 
						int stats_last_writer;		/* NUMA-node id of the last writer on
 | 
				
			||||||
					 * 'stats[0]'.
 | 
										 * 'stats[0]'.
 | 
				
			||||||
					 */
 | 
										 */
 | 
				
			||||||
	struct sw_flow_key key;
 | 
						struct sw_flow_key key;
 | 
				
			||||||
	struct sw_flow_key unmasked_key;
 | 
						struct sw_flow_id id;
 | 
				
			||||||
	struct sw_flow_mask *mask;
 | 
						struct sw_flow_mask *mask;
 | 
				
			||||||
	struct sw_flow_actions __rcu *sf_acts;
 | 
						struct sw_flow_actions __rcu *sf_acts;
 | 
				
			||||||
	struct flow_stats __rcu *stats[]; /* One for each NUMA node.  First one
 | 
						struct flow_stats __rcu *stats[]; /* One for each NUMA node.  First one
 | 
				
			||||||
| 
						 | 
					@ -243,6 +255,16 @@ struct arp_eth_header {
 | 
				
			||||||
	unsigned char       ar_tip[4];		/* target IP address        */
 | 
						unsigned char       ar_tip[4];		/* target IP address        */
 | 
				
			||||||
} __packed;
 | 
					} __packed;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static inline bool ovs_identifier_is_ufid(const struct sw_flow_id *sfid)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						return sfid->ufid_len;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static inline bool ovs_identifier_is_key(const struct sw_flow_id *sfid)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						return !ovs_identifier_is_ufid(sfid);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void ovs_flow_stats_update(struct sw_flow *, __be16 tcp_flags,
 | 
					void ovs_flow_stats_update(struct sw_flow *, __be16 tcp_flags,
 | 
				
			||||||
			   const struct sk_buff *);
 | 
								   const struct sk_buff *);
 | 
				
			||||||
void ovs_flow_stats_get(const struct sw_flow *, struct ovs_flow_stats *,
 | 
					void ovs_flow_stats_get(const struct sw_flow *, struct ovs_flow_stats *,
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1180,6 +1180,59 @@ int ovs_nla_get_match(struct sw_flow_match *match,
 | 
				
			||||||
	return err;
 | 
						return err;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static size_t get_ufid_len(const struct nlattr *attr, bool log)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						size_t len;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (!attr)
 | 
				
			||||||
 | 
							return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						len = nla_len(attr);
 | 
				
			||||||
 | 
						if (len < 1 || len > MAX_UFID_LENGTH) {
 | 
				
			||||||
 | 
							OVS_NLERR(log, "ufid size %u bytes exceeds the range (1, %d)",
 | 
				
			||||||
 | 
								  nla_len(attr), MAX_UFID_LENGTH);
 | 
				
			||||||
 | 
							return 0;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return len;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Initializes 'flow->ufid', returning true if 'attr' contains a valid UFID,
 | 
				
			||||||
 | 
					 * or false otherwise.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					bool ovs_nla_get_ufid(struct sw_flow_id *sfid, const struct nlattr *attr,
 | 
				
			||||||
 | 
							      bool log)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						sfid->ufid_len = get_ufid_len(attr, log);
 | 
				
			||||||
 | 
						if (sfid->ufid_len)
 | 
				
			||||||
 | 
							memcpy(sfid->ufid, nla_data(attr), sfid->ufid_len);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return sfid->ufid_len;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					int ovs_nla_get_identifier(struct sw_flow_id *sfid, const struct nlattr *ufid,
 | 
				
			||||||
 | 
								   const struct sw_flow_key *key, bool log)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						struct sw_flow_key *new_key;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (ovs_nla_get_ufid(sfid, ufid, log))
 | 
				
			||||||
 | 
							return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* If UFID was not provided, use unmasked key. */
 | 
				
			||||||
 | 
						new_key = kmalloc(sizeof(*new_key), GFP_KERNEL);
 | 
				
			||||||
 | 
						if (!new_key)
 | 
				
			||||||
 | 
							return -ENOMEM;
 | 
				
			||||||
 | 
						memcpy(new_key, key, sizeof(*key));
 | 
				
			||||||
 | 
						sfid->unmasked_key = new_key;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					u32 ovs_nla_get_ufid_flags(const struct nlattr *attr)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						return attr ? nla_get_u32(attr) : 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
 * ovs_nla_get_flow_metadata - parses Netlink attributes into a flow key.
 | 
					 * ovs_nla_get_flow_metadata - parses Netlink attributes into a flow key.
 | 
				
			||||||
 * @key: Receives extracted in_port, priority, tun_key and skb_mark.
 | 
					 * @key: Receives extracted in_port, priority, tun_key and skb_mark.
 | 
				
			||||||
| 
						 | 
					@ -1450,9 +1503,20 @@ int ovs_nla_put_key(const struct sw_flow_key *swkey,
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Called with ovs_mutex or RCU read lock. */
 | 
					/* Called with ovs_mutex or RCU read lock. */
 | 
				
			||||||
int ovs_nla_put_unmasked_key(const struct sw_flow *flow, struct sk_buff *skb)
 | 
					int ovs_nla_put_identifier(const struct sw_flow *flow, struct sk_buff *skb)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	return ovs_nla_put_key(&flow->unmasked_key, &flow->unmasked_key,
 | 
						if (ovs_identifier_is_ufid(&flow->id))
 | 
				
			||||||
 | 
							return nla_put(skb, OVS_FLOW_ATTR_UFID, flow->id.ufid_len,
 | 
				
			||||||
 | 
								       flow->id.ufid);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return ovs_nla_put_key(flow->id.unmasked_key, flow->id.unmasked_key,
 | 
				
			||||||
 | 
								       OVS_FLOW_ATTR_KEY, false, skb);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Called with ovs_mutex or RCU read lock. */
 | 
				
			||||||
 | 
					int ovs_nla_put_masked_key(const struct sw_flow *flow, struct sk_buff *skb)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						return ovs_nla_put_key(&flow->mask->key, &flow->key,
 | 
				
			||||||
				OVS_FLOW_ATTR_KEY, false, skb);
 | 
									OVS_FLOW_ATTR_KEY, false, skb);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -48,7 +48,8 @@ int ovs_nla_put_key(const struct sw_flow_key *, const struct sw_flow_key *,
 | 
				
			||||||
int ovs_nla_get_flow_metadata(const struct nlattr *, struct sw_flow_key *,
 | 
					int ovs_nla_get_flow_metadata(const struct nlattr *, struct sw_flow_key *,
 | 
				
			||||||
			      bool log);
 | 
								      bool log);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
int ovs_nla_put_unmasked_key(const struct sw_flow *flow, struct sk_buff *skb);
 | 
					int ovs_nla_put_identifier(const struct sw_flow *flow, struct sk_buff *skb);
 | 
				
			||||||
 | 
					int ovs_nla_put_masked_key(const struct sw_flow *flow, struct sk_buff *skb);
 | 
				
			||||||
int ovs_nla_put_mask(const struct sw_flow *flow, struct sk_buff *skb);
 | 
					int ovs_nla_put_mask(const struct sw_flow *flow, struct sk_buff *skb);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
int ovs_nla_get_match(struct sw_flow_match *, const struct nlattr *key,
 | 
					int ovs_nla_get_match(struct sw_flow_match *, const struct nlattr *key,
 | 
				
			||||||
| 
						 | 
					@ -56,6 +57,11 @@ int ovs_nla_get_match(struct sw_flow_match *, const struct nlattr *key,
 | 
				
			||||||
int ovs_nla_put_egress_tunnel_key(struct sk_buff *,
 | 
					int ovs_nla_put_egress_tunnel_key(struct sk_buff *,
 | 
				
			||||||
				  const struct ovs_tunnel_info *);
 | 
									  const struct ovs_tunnel_info *);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					bool ovs_nla_get_ufid(struct sw_flow_id *, const struct nlattr *, bool log);
 | 
				
			||||||
 | 
					int ovs_nla_get_identifier(struct sw_flow_id *sfid, const struct nlattr *ufid,
 | 
				
			||||||
 | 
								   const struct sw_flow_key *key, bool log);
 | 
				
			||||||
 | 
					u32 ovs_nla_get_ufid_flags(const struct nlattr *attr);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
int ovs_nla_copy_actions(const struct nlattr *attr,
 | 
					int ovs_nla_copy_actions(const struct nlattr *attr,
 | 
				
			||||||
			 const struct sw_flow_key *key,
 | 
								 const struct sw_flow_key *key,
 | 
				
			||||||
			 struct sw_flow_actions **sfa, bool log);
 | 
								 struct sw_flow_actions **sfa, bool log);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -139,6 +139,8 @@ static void flow_free(struct sw_flow *flow)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	int node;
 | 
						int node;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (ovs_identifier_is_key(&flow->id))
 | 
				
			||||||
 | 
							kfree(flow->id.unmasked_key);
 | 
				
			||||||
	kfree((struct sw_flow_actions __force *)flow->sf_acts);
 | 
						kfree((struct sw_flow_actions __force *)flow->sf_acts);
 | 
				
			||||||
	for_each_node(node)
 | 
						for_each_node(node)
 | 
				
			||||||
		if (flow->stats[node])
 | 
							if (flow->stats[node])
 | 
				
			||||||
| 
						 | 
					@ -200,18 +202,28 @@ static struct table_instance *table_instance_alloc(int new_size)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
int ovs_flow_tbl_init(struct flow_table *table)
 | 
					int ovs_flow_tbl_init(struct flow_table *table)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct table_instance *ti;
 | 
						struct table_instance *ti, *ufid_ti;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	ti = table_instance_alloc(TBL_MIN_BUCKETS);
 | 
						ti = table_instance_alloc(TBL_MIN_BUCKETS);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (!ti)
 | 
						if (!ti)
 | 
				
			||||||
		return -ENOMEM;
 | 
							return -ENOMEM;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ufid_ti = table_instance_alloc(TBL_MIN_BUCKETS);
 | 
				
			||||||
 | 
						if (!ufid_ti)
 | 
				
			||||||
 | 
							goto free_ti;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	rcu_assign_pointer(table->ti, ti);
 | 
						rcu_assign_pointer(table->ti, ti);
 | 
				
			||||||
 | 
						rcu_assign_pointer(table->ufid_ti, ufid_ti);
 | 
				
			||||||
	INIT_LIST_HEAD(&table->mask_list);
 | 
						INIT_LIST_HEAD(&table->mask_list);
 | 
				
			||||||
	table->last_rehash = jiffies;
 | 
						table->last_rehash = jiffies;
 | 
				
			||||||
	table->count = 0;
 | 
						table->count = 0;
 | 
				
			||||||
 | 
						table->ufid_count = 0;
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					free_ti:
 | 
				
			||||||
 | 
						__table_instance_destroy(ti);
 | 
				
			||||||
 | 
						return -ENOMEM;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void flow_tbl_destroy_rcu_cb(struct rcu_head *rcu)
 | 
					static void flow_tbl_destroy_rcu_cb(struct rcu_head *rcu)
 | 
				
			||||||
| 
						 | 
					@ -221,13 +233,16 @@ static void flow_tbl_destroy_rcu_cb(struct rcu_head *rcu)
 | 
				
			||||||
	__table_instance_destroy(ti);
 | 
						__table_instance_destroy(ti);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void table_instance_destroy(struct table_instance *ti, bool deferred)
 | 
					static void table_instance_destroy(struct table_instance *ti,
 | 
				
			||||||
 | 
									   struct table_instance *ufid_ti,
 | 
				
			||||||
 | 
									   bool deferred)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	int i;
 | 
						int i;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (!ti)
 | 
						if (!ti)
 | 
				
			||||||
		return;
 | 
							return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						BUG_ON(!ufid_ti);
 | 
				
			||||||
	if (ti->keep_flows)
 | 
						if (ti->keep_flows)
 | 
				
			||||||
		goto skip_flows;
 | 
							goto skip_flows;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -236,18 +251,24 @@ static void table_instance_destroy(struct table_instance *ti, bool deferred)
 | 
				
			||||||
		struct hlist_head *head = flex_array_get(ti->buckets, i);
 | 
							struct hlist_head *head = flex_array_get(ti->buckets, i);
 | 
				
			||||||
		struct hlist_node *n;
 | 
							struct hlist_node *n;
 | 
				
			||||||
		int ver = ti->node_ver;
 | 
							int ver = ti->node_ver;
 | 
				
			||||||
 | 
							int ufid_ver = ufid_ti->node_ver;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		hlist_for_each_entry_safe(flow, n, head, hash_node[ver]) {
 | 
							hlist_for_each_entry_safe(flow, n, head, flow_table.node[ver]) {
 | 
				
			||||||
			hlist_del_rcu(&flow->hash_node[ver]);
 | 
								hlist_del_rcu(&flow->flow_table.node[ver]);
 | 
				
			||||||
 | 
								if (ovs_identifier_is_ufid(&flow->id))
 | 
				
			||||||
 | 
									hlist_del_rcu(&flow->ufid_table.node[ufid_ver]);
 | 
				
			||||||
			ovs_flow_free(flow, deferred);
 | 
								ovs_flow_free(flow, deferred);
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
skip_flows:
 | 
					skip_flows:
 | 
				
			||||||
	if (deferred)
 | 
						if (deferred) {
 | 
				
			||||||
		call_rcu(&ti->rcu, flow_tbl_destroy_rcu_cb);
 | 
							call_rcu(&ti->rcu, flow_tbl_destroy_rcu_cb);
 | 
				
			||||||
	else
 | 
							call_rcu(&ufid_ti->rcu, flow_tbl_destroy_rcu_cb);
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
		__table_instance_destroy(ti);
 | 
							__table_instance_destroy(ti);
 | 
				
			||||||
 | 
							__table_instance_destroy(ufid_ti);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* No need for locking this function is called from RCU callback or
 | 
					/* No need for locking this function is called from RCU callback or
 | 
				
			||||||
| 
						 | 
					@ -256,8 +277,9 @@ static void table_instance_destroy(struct table_instance *ti, bool deferred)
 | 
				
			||||||
void ovs_flow_tbl_destroy(struct flow_table *table)
 | 
					void ovs_flow_tbl_destroy(struct flow_table *table)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct table_instance *ti = rcu_dereference_raw(table->ti);
 | 
						struct table_instance *ti = rcu_dereference_raw(table->ti);
 | 
				
			||||||
 | 
						struct table_instance *ufid_ti = rcu_dereference_raw(table->ufid_ti);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	table_instance_destroy(ti, false);
 | 
						table_instance_destroy(ti, ufid_ti, false);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct sw_flow *ovs_flow_tbl_dump_next(struct table_instance *ti,
 | 
					struct sw_flow *ovs_flow_tbl_dump_next(struct table_instance *ti,
 | 
				
			||||||
| 
						 | 
					@ -272,7 +294,7 @@ struct sw_flow *ovs_flow_tbl_dump_next(struct table_instance *ti,
 | 
				
			||||||
	while (*bucket < ti->n_buckets) {
 | 
						while (*bucket < ti->n_buckets) {
 | 
				
			||||||
		i = 0;
 | 
							i = 0;
 | 
				
			||||||
		head = flex_array_get(ti->buckets, *bucket);
 | 
							head = flex_array_get(ti->buckets, *bucket);
 | 
				
			||||||
		hlist_for_each_entry_rcu(flow, head, hash_node[ver]) {
 | 
							hlist_for_each_entry_rcu(flow, head, flow_table.node[ver]) {
 | 
				
			||||||
			if (i < *last) {
 | 
								if (i < *last) {
 | 
				
			||||||
				i++;
 | 
									i++;
 | 
				
			||||||
				continue;
 | 
									continue;
 | 
				
			||||||
| 
						 | 
					@ -294,16 +316,26 @@ static struct hlist_head *find_bucket(struct table_instance *ti, u32 hash)
 | 
				
			||||||
				(hash & (ti->n_buckets - 1)));
 | 
									(hash & (ti->n_buckets - 1)));
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void table_instance_insert(struct table_instance *ti, struct sw_flow *flow)
 | 
					static void table_instance_insert(struct table_instance *ti,
 | 
				
			||||||
 | 
									  struct sw_flow *flow)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct hlist_head *head;
 | 
						struct hlist_head *head;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	head = find_bucket(ti, flow->hash);
 | 
						head = find_bucket(ti, flow->flow_table.hash);
 | 
				
			||||||
	hlist_add_head_rcu(&flow->hash_node[ti->node_ver], head);
 | 
						hlist_add_head_rcu(&flow->flow_table.node[ti->node_ver], head);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void ufid_table_instance_insert(struct table_instance *ti,
 | 
				
			||||||
 | 
									       struct sw_flow *flow)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						struct hlist_head *head;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						head = find_bucket(ti, flow->ufid_table.hash);
 | 
				
			||||||
 | 
						hlist_add_head_rcu(&flow->ufid_table.node[ti->node_ver], head);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void flow_table_copy_flows(struct table_instance *old,
 | 
					static void flow_table_copy_flows(struct table_instance *old,
 | 
				
			||||||
				  struct table_instance *new)
 | 
									  struct table_instance *new, bool ufid)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	int old_ver;
 | 
						int old_ver;
 | 
				
			||||||
	int i;
 | 
						int i;
 | 
				
			||||||
| 
						 | 
					@ -318,15 +350,21 @@ static void flow_table_copy_flows(struct table_instance *old,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		head = flex_array_get(old->buckets, i);
 | 
							head = flex_array_get(old->buckets, i);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		hlist_for_each_entry(flow, head, hash_node[old_ver])
 | 
							if (ufid)
 | 
				
			||||||
			table_instance_insert(new, flow);
 | 
								hlist_for_each_entry(flow, head,
 | 
				
			||||||
 | 
										     ufid_table.node[old_ver])
 | 
				
			||||||
 | 
									ufid_table_instance_insert(new, flow);
 | 
				
			||||||
 | 
							else
 | 
				
			||||||
 | 
								hlist_for_each_entry(flow, head,
 | 
				
			||||||
 | 
										     flow_table.node[old_ver])
 | 
				
			||||||
 | 
									table_instance_insert(new, flow);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	old->keep_flows = true;
 | 
						old->keep_flows = true;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static struct table_instance *table_instance_rehash(struct table_instance *ti,
 | 
					static struct table_instance *table_instance_rehash(struct table_instance *ti,
 | 
				
			||||||
					    int n_buckets)
 | 
											    int n_buckets, bool ufid)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct table_instance *new_ti;
 | 
						struct table_instance *new_ti;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -334,27 +372,38 @@ static struct table_instance *table_instance_rehash(struct table_instance *ti,
 | 
				
			||||||
	if (!new_ti)
 | 
						if (!new_ti)
 | 
				
			||||||
		return NULL;
 | 
							return NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	flow_table_copy_flows(ti, new_ti);
 | 
						flow_table_copy_flows(ti, new_ti, ufid);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return new_ti;
 | 
						return new_ti;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
int ovs_flow_tbl_flush(struct flow_table *flow_table)
 | 
					int ovs_flow_tbl_flush(struct flow_table *flow_table)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct table_instance *old_ti;
 | 
						struct table_instance *old_ti, *new_ti;
 | 
				
			||||||
	struct table_instance *new_ti;
 | 
						struct table_instance *old_ufid_ti, *new_ufid_ti;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	old_ti = ovsl_dereference(flow_table->ti);
 | 
					 | 
				
			||||||
	new_ti = table_instance_alloc(TBL_MIN_BUCKETS);
 | 
						new_ti = table_instance_alloc(TBL_MIN_BUCKETS);
 | 
				
			||||||
	if (!new_ti)
 | 
						if (!new_ti)
 | 
				
			||||||
		return -ENOMEM;
 | 
							return -ENOMEM;
 | 
				
			||||||
 | 
						new_ufid_ti = table_instance_alloc(TBL_MIN_BUCKETS);
 | 
				
			||||||
 | 
						if (!new_ufid_ti)
 | 
				
			||||||
 | 
							goto err_free_ti;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						old_ti = ovsl_dereference(flow_table->ti);
 | 
				
			||||||
 | 
						old_ufid_ti = ovsl_dereference(flow_table->ufid_ti);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	rcu_assign_pointer(flow_table->ti, new_ti);
 | 
						rcu_assign_pointer(flow_table->ti, new_ti);
 | 
				
			||||||
 | 
						rcu_assign_pointer(flow_table->ufid_ti, new_ufid_ti);
 | 
				
			||||||
	flow_table->last_rehash = jiffies;
 | 
						flow_table->last_rehash = jiffies;
 | 
				
			||||||
	flow_table->count = 0;
 | 
						flow_table->count = 0;
 | 
				
			||||||
 | 
						flow_table->ufid_count = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	table_instance_destroy(old_ti, true);
 | 
						table_instance_destroy(old_ti, old_ufid_ti, true);
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					err_free_ti:
 | 
				
			||||||
 | 
						__table_instance_destroy(new_ti);
 | 
				
			||||||
 | 
						return -ENOMEM;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static u32 flow_hash(const struct sw_flow_key *key,
 | 
					static u32 flow_hash(const struct sw_flow_key *key,
 | 
				
			||||||
| 
						 | 
					@ -402,14 +451,15 @@ static bool flow_cmp_masked_key(const struct sw_flow *flow,
 | 
				
			||||||
	return cmp_key(&flow->key, key, range->start, range->end);
 | 
						return cmp_key(&flow->key, key, range->start, range->end);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
bool ovs_flow_cmp_unmasked_key(const struct sw_flow *flow,
 | 
					static bool ovs_flow_cmp_unmasked_key(const struct sw_flow *flow,
 | 
				
			||||||
			       const struct sw_flow_match *match)
 | 
									      const struct sw_flow_match *match)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct sw_flow_key *key = match->key;
 | 
						struct sw_flow_key *key = match->key;
 | 
				
			||||||
	int key_start = flow_key_start(key);
 | 
						int key_start = flow_key_start(key);
 | 
				
			||||||
	int key_end = match->range.end;
 | 
						int key_end = match->range.end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return cmp_key(&flow->unmasked_key, key, key_start, key_end);
 | 
						BUG_ON(ovs_identifier_is_ufid(&flow->id));
 | 
				
			||||||
 | 
						return cmp_key(flow->id.unmasked_key, key, key_start, key_end);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static struct sw_flow *masked_flow_lookup(struct table_instance *ti,
 | 
					static struct sw_flow *masked_flow_lookup(struct table_instance *ti,
 | 
				
			||||||
| 
						 | 
					@ -424,8 +474,8 @@ static struct sw_flow *masked_flow_lookup(struct table_instance *ti,
 | 
				
			||||||
	ovs_flow_mask_key(&masked_key, unmasked, mask);
 | 
						ovs_flow_mask_key(&masked_key, unmasked, mask);
 | 
				
			||||||
	hash = flow_hash(&masked_key, &mask->range);
 | 
						hash = flow_hash(&masked_key, &mask->range);
 | 
				
			||||||
	head = find_bucket(ti, hash);
 | 
						head = find_bucket(ti, hash);
 | 
				
			||||||
	hlist_for_each_entry_rcu(flow, head, hash_node[ti->node_ver]) {
 | 
						hlist_for_each_entry_rcu(flow, head, flow_table.node[ti->node_ver]) {
 | 
				
			||||||
		if (flow->mask == mask && flow->hash == hash &&
 | 
							if (flow->mask == mask && flow->flow_table.hash == hash &&
 | 
				
			||||||
		    flow_cmp_masked_key(flow, &masked_key, &mask->range))
 | 
							    flow_cmp_masked_key(flow, &masked_key, &mask->range))
 | 
				
			||||||
			return flow;
 | 
								return flow;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
| 
						 | 
					@ -468,7 +518,48 @@ struct sw_flow *ovs_flow_tbl_lookup_exact(struct flow_table *tbl,
 | 
				
			||||||
	/* Always called under ovs-mutex. */
 | 
						/* Always called under ovs-mutex. */
 | 
				
			||||||
	list_for_each_entry(mask, &tbl->mask_list, list) {
 | 
						list_for_each_entry(mask, &tbl->mask_list, list) {
 | 
				
			||||||
		flow = masked_flow_lookup(ti, match->key, mask);
 | 
							flow = masked_flow_lookup(ti, match->key, mask);
 | 
				
			||||||
		if (flow && ovs_flow_cmp_unmasked_key(flow, match))  /* Found */
 | 
							if (flow && ovs_identifier_is_key(&flow->id) &&
 | 
				
			||||||
 | 
							    ovs_flow_cmp_unmasked_key(flow, match))
 | 
				
			||||||
 | 
								return flow;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return NULL;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static u32 ufid_hash(const struct sw_flow_id *sfid)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						return jhash(sfid->ufid, sfid->ufid_len, 0);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static bool ovs_flow_cmp_ufid(const struct sw_flow *flow,
 | 
				
			||||||
 | 
								      const struct sw_flow_id *sfid)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						if (flow->id.ufid_len != sfid->ufid_len)
 | 
				
			||||||
 | 
							return false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return !memcmp(flow->id.ufid, sfid->ufid, sfid->ufid_len);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					bool ovs_flow_cmp(const struct sw_flow *flow, const struct sw_flow_match *match)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						if (ovs_identifier_is_ufid(&flow->id))
 | 
				
			||||||
 | 
							return flow_cmp_masked_key(flow, match->key, &match->range);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return ovs_flow_cmp_unmasked_key(flow, match);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					struct sw_flow *ovs_flow_tbl_lookup_ufid(struct flow_table *tbl,
 | 
				
			||||||
 | 
										 const struct sw_flow_id *ufid)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						struct table_instance *ti = rcu_dereference_ovsl(tbl->ufid_ti);
 | 
				
			||||||
 | 
						struct sw_flow *flow;
 | 
				
			||||||
 | 
						struct hlist_head *head;
 | 
				
			||||||
 | 
						u32 hash;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						hash = ufid_hash(ufid);
 | 
				
			||||||
 | 
						head = find_bucket(ti, hash);
 | 
				
			||||||
 | 
						hlist_for_each_entry_rcu(flow, head, ufid_table.node[ti->node_ver]) {
 | 
				
			||||||
 | 
							if (flow->ufid_table.hash == hash &&
 | 
				
			||||||
 | 
							    ovs_flow_cmp_ufid(flow, ufid))
 | 
				
			||||||
			return flow;
 | 
								return flow;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	return NULL;
 | 
						return NULL;
 | 
				
			||||||
| 
						 | 
					@ -485,9 +576,10 @@ int ovs_flow_tbl_num_masks(const struct flow_table *table)
 | 
				
			||||||
	return num;
 | 
						return num;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static struct table_instance *table_instance_expand(struct table_instance *ti)
 | 
					static struct table_instance *table_instance_expand(struct table_instance *ti,
 | 
				
			||||||
 | 
											    bool ufid)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	return table_instance_rehash(ti, ti->n_buckets * 2);
 | 
						return table_instance_rehash(ti, ti->n_buckets * 2, ufid);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Remove 'mask' from the mask list, if it is not needed any more. */
 | 
					/* Remove 'mask' from the mask list, if it is not needed any more. */
 | 
				
			||||||
| 
						 | 
					@ -512,10 +604,15 @@ static void flow_mask_remove(struct flow_table *tbl, struct sw_flow_mask *mask)
 | 
				
			||||||
void ovs_flow_tbl_remove(struct flow_table *table, struct sw_flow *flow)
 | 
					void ovs_flow_tbl_remove(struct flow_table *table, struct sw_flow *flow)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct table_instance *ti = ovsl_dereference(table->ti);
 | 
						struct table_instance *ti = ovsl_dereference(table->ti);
 | 
				
			||||||
 | 
						struct table_instance *ufid_ti = ovsl_dereference(table->ufid_ti);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	BUG_ON(table->count == 0);
 | 
						BUG_ON(table->count == 0);
 | 
				
			||||||
	hlist_del_rcu(&flow->hash_node[ti->node_ver]);
 | 
						hlist_del_rcu(&flow->flow_table.node[ti->node_ver]);
 | 
				
			||||||
	table->count--;
 | 
						table->count--;
 | 
				
			||||||
 | 
						if (ovs_identifier_is_ufid(&flow->id)) {
 | 
				
			||||||
 | 
							hlist_del_rcu(&flow->ufid_table.node[ufid_ti->node_ver]);
 | 
				
			||||||
 | 
							table->ufid_count--;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* RCU delete the mask. 'flow->mask' is not NULLed, as it should be
 | 
						/* RCU delete the mask. 'flow->mask' is not NULLed, as it should be
 | 
				
			||||||
	 * accessible as long as the RCU read lock is held.
 | 
						 * accessible as long as the RCU read lock is held.
 | 
				
			||||||
| 
						 | 
					@ -589,24 +686,46 @@ static void flow_key_insert(struct flow_table *table, struct sw_flow *flow)
 | 
				
			||||||
	struct table_instance *new_ti = NULL;
 | 
						struct table_instance *new_ti = NULL;
 | 
				
			||||||
	struct table_instance *ti;
 | 
						struct table_instance *ti;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	flow->hash = flow_hash(&flow->key, &flow->mask->range);
 | 
						flow->flow_table.hash = flow_hash(&flow->key, &flow->mask->range);
 | 
				
			||||||
	ti = ovsl_dereference(table->ti);
 | 
						ti = ovsl_dereference(table->ti);
 | 
				
			||||||
	table_instance_insert(ti, flow);
 | 
						table_instance_insert(ti, flow);
 | 
				
			||||||
	table->count++;
 | 
						table->count++;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* Expand table, if necessary, to make room. */
 | 
						/* Expand table, if necessary, to make room. */
 | 
				
			||||||
	if (table->count > ti->n_buckets)
 | 
						if (table->count > ti->n_buckets)
 | 
				
			||||||
		new_ti = table_instance_expand(ti);
 | 
							new_ti = table_instance_expand(ti, false);
 | 
				
			||||||
	else if (time_after(jiffies, table->last_rehash + REHASH_INTERVAL))
 | 
						else if (time_after(jiffies, table->last_rehash + REHASH_INTERVAL))
 | 
				
			||||||
		new_ti = table_instance_rehash(ti, ti->n_buckets);
 | 
							new_ti = table_instance_rehash(ti, ti->n_buckets, false);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (new_ti) {
 | 
						if (new_ti) {
 | 
				
			||||||
		rcu_assign_pointer(table->ti, new_ti);
 | 
							rcu_assign_pointer(table->ti, new_ti);
 | 
				
			||||||
		table_instance_destroy(ti, true);
 | 
							call_rcu(&ti->rcu, flow_tbl_destroy_rcu_cb);
 | 
				
			||||||
		table->last_rehash = jiffies;
 | 
							table->last_rehash = jiffies;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Must be called with OVS mutex held. */
 | 
				
			||||||
 | 
					static void flow_ufid_insert(struct flow_table *table, struct sw_flow *flow)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						struct table_instance *ti;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						flow->ufid_table.hash = ufid_hash(&flow->id);
 | 
				
			||||||
 | 
						ti = ovsl_dereference(table->ufid_ti);
 | 
				
			||||||
 | 
						ufid_table_instance_insert(ti, flow);
 | 
				
			||||||
 | 
						table->ufid_count++;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Expand table, if necessary, to make room. */
 | 
				
			||||||
 | 
						if (table->ufid_count > ti->n_buckets) {
 | 
				
			||||||
 | 
							struct table_instance *new_ti;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							new_ti = table_instance_expand(ti, true);
 | 
				
			||||||
 | 
							if (new_ti) {
 | 
				
			||||||
 | 
								rcu_assign_pointer(table->ufid_ti, new_ti);
 | 
				
			||||||
 | 
								call_rcu(&ti->rcu, flow_tbl_destroy_rcu_cb);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Must be called with OVS mutex held. */
 | 
					/* Must be called with OVS mutex held. */
 | 
				
			||||||
int ovs_flow_tbl_insert(struct flow_table *table, struct sw_flow *flow,
 | 
					int ovs_flow_tbl_insert(struct flow_table *table, struct sw_flow *flow,
 | 
				
			||||||
			const struct sw_flow_mask *mask)
 | 
								const struct sw_flow_mask *mask)
 | 
				
			||||||
| 
						 | 
					@ -617,6 +736,8 @@ int ovs_flow_tbl_insert(struct flow_table *table, struct sw_flow *flow,
 | 
				
			||||||
	if (err)
 | 
						if (err)
 | 
				
			||||||
		return err;
 | 
							return err;
 | 
				
			||||||
	flow_key_insert(table, flow);
 | 
						flow_key_insert(table, flow);
 | 
				
			||||||
 | 
						if (ovs_identifier_is_ufid(&flow->id))
 | 
				
			||||||
 | 
							flow_ufid_insert(table, flow);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -47,9 +47,11 @@ struct table_instance {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct flow_table {
 | 
					struct flow_table {
 | 
				
			||||||
	struct table_instance __rcu *ti;
 | 
						struct table_instance __rcu *ti;
 | 
				
			||||||
 | 
						struct table_instance __rcu *ufid_ti;
 | 
				
			||||||
	struct list_head mask_list;
 | 
						struct list_head mask_list;
 | 
				
			||||||
	unsigned long last_rehash;
 | 
						unsigned long last_rehash;
 | 
				
			||||||
	unsigned int count;
 | 
						unsigned int count;
 | 
				
			||||||
 | 
						unsigned int ufid_count;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
extern struct kmem_cache *flow_stats_cache;
 | 
					extern struct kmem_cache *flow_stats_cache;
 | 
				
			||||||
| 
						 | 
					@ -78,8 +80,10 @@ struct sw_flow *ovs_flow_tbl_lookup(struct flow_table *,
 | 
				
			||||||
				    const struct sw_flow_key *);
 | 
									    const struct sw_flow_key *);
 | 
				
			||||||
struct sw_flow *ovs_flow_tbl_lookup_exact(struct flow_table *tbl,
 | 
					struct sw_flow *ovs_flow_tbl_lookup_exact(struct flow_table *tbl,
 | 
				
			||||||
					  const struct sw_flow_match *match);
 | 
										  const struct sw_flow_match *match);
 | 
				
			||||||
bool ovs_flow_cmp_unmasked_key(const struct sw_flow *flow,
 | 
					struct sw_flow *ovs_flow_tbl_lookup_ufid(struct flow_table *,
 | 
				
			||||||
			       const struct sw_flow_match *match);
 | 
										 const struct sw_flow_id *);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					bool ovs_flow_cmp(const struct sw_flow *, const struct sw_flow_match *);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void ovs_flow_mask_key(struct sw_flow_key *dst, const struct sw_flow_key *src,
 | 
					void ovs_flow_mask_key(struct sw_flow_key *dst, const struct sw_flow_key *src,
 | 
				
			||||||
		       const struct sw_flow_mask *mask);
 | 
							       const struct sw_flow_mask *mask);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue