forked from mirrors/linux
		
	ethtool: add ethtool_rx_flow_spec to flow_rule structure translator
This patch adds a function to translate the ethtool_rx_flow_spec structure to the flow_rule representation. This allows us to reuse code from the driver side given that both flower and ethtool_rx_flow interfaces use the same representation. This patch also includes support for the flow type flags FLOW_EXT, FLOW_MAC_EXT and FLOW_RSS. The ethtool_rx_flow_spec_input wrapper structure is used to convey the rss_context field, that is away from the ethtool_rx_flow_spec structure, and the ethtool_rx_flow_spec structure. Signed-off-by: Pablo Neira Ayuso <pablo@netfilter.org> Acked-by: Jiri Pirko <jiri@mellanox.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
		
							parent
							
								
									8bec2833fb
								
							
						
					
					
						commit
						eca4205f9e
					
				
					 2 changed files with 256 additions and 0 deletions
				
			
		| 
						 | 
					@ -400,4 +400,19 @@ struct ethtool_ops {
 | 
				
			||||||
	void	(*get_ethtool_phy_stats)(struct net_device *,
 | 
						void	(*get_ethtool_phy_stats)(struct net_device *,
 | 
				
			||||||
					 struct ethtool_stats *, u64 *);
 | 
										 struct ethtool_stats *, u64 *);
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					struct ethtool_rx_flow_rule {
 | 
				
			||||||
 | 
						struct flow_rule	*rule;
 | 
				
			||||||
 | 
						unsigned long		priv[0];
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					struct ethtool_rx_flow_spec_input {
 | 
				
			||||||
 | 
						const struct ethtool_rx_flow_spec	*fs;
 | 
				
			||||||
 | 
						u32					rss_ctx;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					struct ethtool_rx_flow_rule *
 | 
				
			||||||
 | 
					ethtool_rx_flow_rule_create(const struct ethtool_rx_flow_spec_input *input);
 | 
				
			||||||
 | 
					void ethtool_rx_flow_rule_destroy(struct ethtool_rx_flow_rule *rule);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#endif /* _LINUX_ETHTOOL_H */
 | 
					#endif /* _LINUX_ETHTOOL_H */
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -29,6 +29,7 @@
 | 
				
			||||||
#include <linux/net.h>
 | 
					#include <linux/net.h>
 | 
				
			||||||
#include <net/devlink.h>
 | 
					#include <net/devlink.h>
 | 
				
			||||||
#include <net/xdp_sock.h>
 | 
					#include <net/xdp_sock.h>
 | 
				
			||||||
 | 
					#include <net/flow_offload.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/*
 | 
					/*
 | 
				
			||||||
 * Some useful ethtool_ops methods that're device independent.
 | 
					 * Some useful ethtool_ops methods that're device independent.
 | 
				
			||||||
| 
						 | 
					@ -2820,3 +2821,243 @@ int dev_ethtool(struct net *net, struct ifreq *ifr)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return rc;
 | 
						return rc;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					struct ethtool_rx_flow_key {
 | 
				
			||||||
 | 
						struct flow_dissector_key_basic			basic;
 | 
				
			||||||
 | 
						union {
 | 
				
			||||||
 | 
							struct flow_dissector_key_ipv4_addrs	ipv4;
 | 
				
			||||||
 | 
							struct flow_dissector_key_ipv6_addrs	ipv6;
 | 
				
			||||||
 | 
						};
 | 
				
			||||||
 | 
						struct flow_dissector_key_ports			tp;
 | 
				
			||||||
 | 
						struct flow_dissector_key_ip			ip;
 | 
				
			||||||
 | 
						struct flow_dissector_key_vlan			vlan;
 | 
				
			||||||
 | 
						struct flow_dissector_key_eth_addrs		eth_addrs;
 | 
				
			||||||
 | 
					} __aligned(BITS_PER_LONG / 8); /* Ensure that we can do comparisons as longs. */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					struct ethtool_rx_flow_match {
 | 
				
			||||||
 | 
						struct flow_dissector		dissector;
 | 
				
			||||||
 | 
						struct ethtool_rx_flow_key	key;
 | 
				
			||||||
 | 
						struct ethtool_rx_flow_key	mask;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					struct ethtool_rx_flow_rule *
 | 
				
			||||||
 | 
					ethtool_rx_flow_rule_create(const struct ethtool_rx_flow_spec_input *input)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						const struct ethtool_rx_flow_spec *fs = input->fs;
 | 
				
			||||||
 | 
						static struct in6_addr zero_addr = {};
 | 
				
			||||||
 | 
						struct ethtool_rx_flow_match *match;
 | 
				
			||||||
 | 
						struct ethtool_rx_flow_rule *flow;
 | 
				
			||||||
 | 
						struct flow_action_entry *act;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						flow = kzalloc(sizeof(struct ethtool_rx_flow_rule) +
 | 
				
			||||||
 | 
							       sizeof(struct ethtool_rx_flow_match), GFP_KERNEL);
 | 
				
			||||||
 | 
						if (!flow)
 | 
				
			||||||
 | 
							return ERR_PTR(-ENOMEM);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* ethtool_rx supports only one single action per rule. */
 | 
				
			||||||
 | 
						flow->rule = flow_rule_alloc(1);
 | 
				
			||||||
 | 
						if (!flow->rule) {
 | 
				
			||||||
 | 
							kfree(flow);
 | 
				
			||||||
 | 
							return ERR_PTR(-ENOMEM);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						match = (struct ethtool_rx_flow_match *)flow->priv;
 | 
				
			||||||
 | 
						flow->rule->match.dissector	= &match->dissector;
 | 
				
			||||||
 | 
						flow->rule->match.mask		= &match->mask;
 | 
				
			||||||
 | 
						flow->rule->match.key		= &match->key;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						match->mask.basic.n_proto = htons(0xffff);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						switch (fs->flow_type & ~(FLOW_EXT | FLOW_MAC_EXT | FLOW_RSS)) {
 | 
				
			||||||
 | 
						case TCP_V4_FLOW:
 | 
				
			||||||
 | 
						case UDP_V4_FLOW: {
 | 
				
			||||||
 | 
							const struct ethtool_tcpip4_spec *v4_spec, *v4_m_spec;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							match->key.basic.n_proto = htons(ETH_P_IP);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							v4_spec = &fs->h_u.tcp_ip4_spec;
 | 
				
			||||||
 | 
							v4_m_spec = &fs->m_u.tcp_ip4_spec;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (v4_m_spec->ip4src) {
 | 
				
			||||||
 | 
								match->key.ipv4.src = v4_spec->ip4src;
 | 
				
			||||||
 | 
								match->mask.ipv4.src = v4_m_spec->ip4src;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if (v4_m_spec->ip4dst) {
 | 
				
			||||||
 | 
								match->key.ipv4.dst = v4_spec->ip4dst;
 | 
				
			||||||
 | 
								match->mask.ipv4.dst = v4_m_spec->ip4dst;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if (v4_m_spec->ip4src ||
 | 
				
			||||||
 | 
							    v4_m_spec->ip4dst) {
 | 
				
			||||||
 | 
								match->dissector.used_keys |=
 | 
				
			||||||
 | 
									BIT(FLOW_DISSECTOR_KEY_IPV4_ADDRS);
 | 
				
			||||||
 | 
								match->dissector.offset[FLOW_DISSECTOR_KEY_IPV4_ADDRS] =
 | 
				
			||||||
 | 
									offsetof(struct ethtool_rx_flow_key, ipv4);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if (v4_m_spec->psrc) {
 | 
				
			||||||
 | 
								match->key.tp.src = v4_spec->psrc;
 | 
				
			||||||
 | 
								match->mask.tp.src = v4_m_spec->psrc;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if (v4_m_spec->pdst) {
 | 
				
			||||||
 | 
								match->key.tp.dst = v4_spec->pdst;
 | 
				
			||||||
 | 
								match->mask.tp.dst = v4_m_spec->pdst;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if (v4_m_spec->psrc ||
 | 
				
			||||||
 | 
							    v4_m_spec->pdst) {
 | 
				
			||||||
 | 
								match->dissector.used_keys |=
 | 
				
			||||||
 | 
									BIT(FLOW_DISSECTOR_KEY_PORTS);
 | 
				
			||||||
 | 
								match->dissector.offset[FLOW_DISSECTOR_KEY_PORTS] =
 | 
				
			||||||
 | 
									offsetof(struct ethtool_rx_flow_key, tp);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if (v4_m_spec->tos) {
 | 
				
			||||||
 | 
								match->key.ip.tos = v4_spec->tos;
 | 
				
			||||||
 | 
								match->mask.ip.tos = v4_m_spec->tos;
 | 
				
			||||||
 | 
								match->dissector.used_keys |=
 | 
				
			||||||
 | 
									BIT(FLOW_DISSECTOR_KEY_IP);
 | 
				
			||||||
 | 
								match->dissector.offset[FLOW_DISSECTOR_KEY_IP] =
 | 
				
			||||||
 | 
									offsetof(struct ethtool_rx_flow_key, ip);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							break;
 | 
				
			||||||
 | 
						case TCP_V6_FLOW:
 | 
				
			||||||
 | 
						case UDP_V6_FLOW: {
 | 
				
			||||||
 | 
							const struct ethtool_tcpip6_spec *v6_spec, *v6_m_spec;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							match->key.basic.n_proto = htons(ETH_P_IPV6);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							v6_spec = &fs->h_u.tcp_ip6_spec;
 | 
				
			||||||
 | 
							v6_m_spec = &fs->m_u.tcp_ip6_spec;
 | 
				
			||||||
 | 
							if (memcmp(v6_m_spec->ip6src, &zero_addr, sizeof(zero_addr))) {
 | 
				
			||||||
 | 
								memcpy(&match->key.ipv6.src, v6_spec->ip6src,
 | 
				
			||||||
 | 
								       sizeof(match->key.ipv6.src));
 | 
				
			||||||
 | 
								memcpy(&match->mask.ipv6.src, v6_m_spec->ip6src,
 | 
				
			||||||
 | 
								       sizeof(match->mask.ipv6.src));
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if (memcmp(v6_m_spec->ip6dst, &zero_addr, sizeof(zero_addr))) {
 | 
				
			||||||
 | 
								memcpy(&match->key.ipv6.dst, v6_spec->ip6dst,
 | 
				
			||||||
 | 
								       sizeof(match->key.ipv6.dst));
 | 
				
			||||||
 | 
								memcpy(&match->mask.ipv6.dst, v6_m_spec->ip6dst,
 | 
				
			||||||
 | 
								       sizeof(match->mask.ipv6.dst));
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if (memcmp(v6_m_spec->ip6src, &zero_addr, sizeof(zero_addr)) ||
 | 
				
			||||||
 | 
							    memcmp(v6_m_spec->ip6src, &zero_addr, sizeof(zero_addr))) {
 | 
				
			||||||
 | 
								match->dissector.used_keys |=
 | 
				
			||||||
 | 
									BIT(FLOW_DISSECTOR_KEY_IPV6_ADDRS);
 | 
				
			||||||
 | 
								match->dissector.offset[FLOW_DISSECTOR_KEY_IPV6_ADDRS] =
 | 
				
			||||||
 | 
									offsetof(struct ethtool_rx_flow_key, ipv6);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if (v6_m_spec->psrc) {
 | 
				
			||||||
 | 
								match->key.tp.src = v6_spec->psrc;
 | 
				
			||||||
 | 
								match->mask.tp.src = v6_m_spec->psrc;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if (v6_m_spec->pdst) {
 | 
				
			||||||
 | 
								match->key.tp.dst = v6_spec->pdst;
 | 
				
			||||||
 | 
								match->mask.tp.dst = v6_m_spec->pdst;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if (v6_m_spec->psrc ||
 | 
				
			||||||
 | 
							    v6_m_spec->pdst) {
 | 
				
			||||||
 | 
								match->dissector.used_keys |=
 | 
				
			||||||
 | 
									BIT(FLOW_DISSECTOR_KEY_PORTS);
 | 
				
			||||||
 | 
								match->dissector.offset[FLOW_DISSECTOR_KEY_PORTS] =
 | 
				
			||||||
 | 
									offsetof(struct ethtool_rx_flow_key, tp);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if (v6_m_spec->tclass) {
 | 
				
			||||||
 | 
								match->key.ip.tos = v6_spec->tclass;
 | 
				
			||||||
 | 
								match->mask.ip.tos = v6_m_spec->tclass;
 | 
				
			||||||
 | 
								match->dissector.used_keys |=
 | 
				
			||||||
 | 
									BIT(FLOW_DISSECTOR_KEY_IP);
 | 
				
			||||||
 | 
								match->dissector.offset[FLOW_DISSECTOR_KEY_IP] =
 | 
				
			||||||
 | 
									offsetof(struct ethtool_rx_flow_key, ip);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							break;
 | 
				
			||||||
 | 
						default:
 | 
				
			||||||
 | 
							ethtool_rx_flow_rule_destroy(flow);
 | 
				
			||||||
 | 
							return ERR_PTR(-EINVAL);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						switch (fs->flow_type & ~(FLOW_EXT | FLOW_MAC_EXT | FLOW_RSS)) {
 | 
				
			||||||
 | 
						case TCP_V4_FLOW:
 | 
				
			||||||
 | 
						case TCP_V6_FLOW:
 | 
				
			||||||
 | 
							match->key.basic.ip_proto = IPPROTO_TCP;
 | 
				
			||||||
 | 
							break;
 | 
				
			||||||
 | 
						case UDP_V4_FLOW:
 | 
				
			||||||
 | 
						case UDP_V6_FLOW:
 | 
				
			||||||
 | 
							match->key.basic.ip_proto = IPPROTO_UDP;
 | 
				
			||||||
 | 
							break;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						match->mask.basic.ip_proto = 0xff;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						match->dissector.used_keys |= BIT(FLOW_DISSECTOR_KEY_BASIC);
 | 
				
			||||||
 | 
						match->dissector.offset[FLOW_DISSECTOR_KEY_BASIC] =
 | 
				
			||||||
 | 
							offsetof(struct ethtool_rx_flow_key, basic);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (fs->flow_type & FLOW_EXT) {
 | 
				
			||||||
 | 
							const struct ethtool_flow_ext *ext_h_spec = &fs->h_ext;
 | 
				
			||||||
 | 
							const struct ethtool_flow_ext *ext_m_spec = &fs->m_ext;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (ext_m_spec->vlan_etype &&
 | 
				
			||||||
 | 
							    ext_m_spec->vlan_tci) {
 | 
				
			||||||
 | 
								match->key.vlan.vlan_tpid = ext_h_spec->vlan_etype;
 | 
				
			||||||
 | 
								match->mask.vlan.vlan_tpid = ext_m_spec->vlan_etype;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								match->key.vlan.vlan_id =
 | 
				
			||||||
 | 
									ntohs(ext_h_spec->vlan_tci) & 0x0fff;
 | 
				
			||||||
 | 
								match->mask.vlan.vlan_id =
 | 
				
			||||||
 | 
									ntohs(ext_m_spec->vlan_tci) & 0x0fff;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								match->key.vlan.vlan_priority =
 | 
				
			||||||
 | 
									(ntohs(ext_h_spec->vlan_tci) & 0xe000) >> 13;
 | 
				
			||||||
 | 
								match->mask.vlan.vlan_priority =
 | 
				
			||||||
 | 
									(ntohs(ext_m_spec->vlan_tci) & 0xe000) >> 13;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								match->dissector.used_keys |=
 | 
				
			||||||
 | 
									BIT(FLOW_DISSECTOR_KEY_VLAN);
 | 
				
			||||||
 | 
								match->dissector.offset[FLOW_DISSECTOR_KEY_VLAN] =
 | 
				
			||||||
 | 
									offsetof(struct ethtool_rx_flow_key, vlan);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if (fs->flow_type & FLOW_MAC_EXT) {
 | 
				
			||||||
 | 
							const struct ethtool_flow_ext *ext_h_spec = &fs->h_ext;
 | 
				
			||||||
 | 
							const struct ethtool_flow_ext *ext_m_spec = &fs->m_ext;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (ext_m_spec->h_dest) {
 | 
				
			||||||
 | 
								memcpy(match->key.eth_addrs.dst, ext_h_spec->h_dest,
 | 
				
			||||||
 | 
								       ETH_ALEN);
 | 
				
			||||||
 | 
								memcpy(match->mask.eth_addrs.dst, ext_m_spec->h_dest,
 | 
				
			||||||
 | 
								       ETH_ALEN);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								match->dissector.used_keys |=
 | 
				
			||||||
 | 
									BIT(FLOW_DISSECTOR_KEY_ETH_ADDRS);
 | 
				
			||||||
 | 
								match->dissector.offset[FLOW_DISSECTOR_KEY_ETH_ADDRS] =
 | 
				
			||||||
 | 
									offsetof(struct ethtool_rx_flow_key, eth_addrs);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						act = &flow->rule->action.entries[0];
 | 
				
			||||||
 | 
						switch (fs->ring_cookie) {
 | 
				
			||||||
 | 
						case RX_CLS_FLOW_DISC:
 | 
				
			||||||
 | 
							act->id = FLOW_ACTION_DROP;
 | 
				
			||||||
 | 
							break;
 | 
				
			||||||
 | 
						case RX_CLS_FLOW_WAKE:
 | 
				
			||||||
 | 
							act->id = FLOW_ACTION_WAKE;
 | 
				
			||||||
 | 
							break;
 | 
				
			||||||
 | 
						default:
 | 
				
			||||||
 | 
							act->id = FLOW_ACTION_QUEUE;
 | 
				
			||||||
 | 
							if (fs->flow_type & FLOW_RSS)
 | 
				
			||||||
 | 
								act->queue.ctx = input->rss_ctx;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							act->queue.vf = ethtool_get_flow_spec_ring_vf(fs->ring_cookie);
 | 
				
			||||||
 | 
							act->queue.index = ethtool_get_flow_spec_ring(fs->ring_cookie);
 | 
				
			||||||
 | 
							break;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return flow;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					EXPORT_SYMBOL(ethtool_rx_flow_rule_create);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void ethtool_rx_flow_rule_destroy(struct ethtool_rx_flow_rule *flow)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						kfree(flow->rule);
 | 
				
			||||||
 | 
						kfree(flow);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					EXPORT_SYMBOL(ethtool_rx_flow_rule_destroy);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue