Flex-bytes allows for packet matching based on an offset and value. This is supported via the ethtool user-def option. It is specified by providing an offset followed by a 2 byte match value. Offset is measured from the start of the MAC address. The following restrictions apply to flex-bytes. The specified offset must be an even number and be smaller than 0x1fe. Example usage: ethtool -N eth0 flow-type tcp4 src-ip 192.168.0.55 dst-ip 172.16.0.55 \ src-port 12 dst-port 13 user-def 0x10ffff action 32 Signed-off-by: Henry Tieman <henry.w.tieman@intel.com> Signed-off-by: Tony Nguyen <anthony.l.nguyen@intel.com> Tested-by: Andrew Bowers <andrewx.bowers@intel.com> Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
		
			
				
	
	
		
			841 lines
		
	
	
		
			28 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			841 lines
		
	
	
		
			28 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| /* Copyright (C) 2018-2020, Intel Corporation. */
 | |
| 
 | |
| #include "ice_common.h"
 | |
| 
 | |
| /* These are training packet headers used to program flow director filters. */
 | |
| static const u8 ice_fdir_tcpv4_pkt[] = {
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x45, 0x00,
 | |
| 	0x00, 0x28, 0x00, 0x01, 0x00, 0x00, 0x40, 0x06,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x50, 0x00,
 | |
| 	0x20, 0x00, 0x00, 0x00, 0x00, 0x00
 | |
| };
 | |
| 
 | |
| static const u8 ice_fdir_udpv4_pkt[] = {
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x45, 0x00,
 | |
| 	0x00, 0x1C, 0x00, 0x00, 0x40, 0x00, 0x40, 0x11,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00,
 | |
| };
 | |
| 
 | |
| static const u8 ice_fdir_sctpv4_pkt[] = {
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x45, 0x00,
 | |
| 	0x00, 0x20, 0x00, 0x00, 0x40, 0x00, 0x40, 0x84,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| };
 | |
| 
 | |
| static const u8 ice_fdir_ipv4_pkt[] = {
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x45, 0x00,
 | |
| 	0x00, 0x14, 0x00, 0x00, 0x40, 0x00, 0x40, 0x10,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00
 | |
| };
 | |
| 
 | |
| static const u8 ice_fdir_tcpv6_pkt[] = {
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x86, 0xDD, 0x60, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x14, 0x06, 0x40, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x50, 0x00, 0x20, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00,
 | |
| };
 | |
| 
 | |
| static const u8 ice_fdir_udpv6_pkt[] = {
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x86, 0xDD, 0x60, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x08, 0x11, 0x40, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x08, 0x00, 0x00,
 | |
| };
 | |
| 
 | |
| static const u8 ice_fdir_sctpv6_pkt[] = {
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x86, 0xDD, 0x60, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x0C, 0x84, 0x40, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00,
 | |
| };
 | |
| 
 | |
| static const u8 ice_fdir_ipv6_pkt[] = {
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x86, 0xDD, 0x60, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x3B, 0x40, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| };
 | |
| 
 | |
| static const u8 ice_fdir_tcp4_tun_pkt[] = {
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x45, 0x00,
 | |
| 	0x00, 0x5a, 0x00, 0x00, 0x40, 0x00, 0x40, 0x11,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x04, 0x00, 0x00, 0x03, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x08, 0x00,
 | |
| 	0x45, 0x00, 0x00, 0x28, 0x00, 0x00, 0x40, 0x00,
 | |
| 	0x40, 0x06, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x50, 0x00, 0x20, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| };
 | |
| 
 | |
| static const u8 ice_fdir_udp4_tun_pkt[] = {
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x45, 0x00,
 | |
| 	0x00, 0x4e, 0x00, 0x00, 0x40, 0x00, 0x40, 0x11,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x04, 0x00, 0x00, 0x03, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x08, 0x00,
 | |
| 	0x45, 0x00, 0x00, 0x1c, 0x00, 0x00, 0x40, 0x00,
 | |
| 	0x40, 0x11, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00,
 | |
| };
 | |
| 
 | |
| static const u8 ice_fdir_sctp4_tun_pkt[] = {
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x45, 0x00,
 | |
| 	0x00, 0x52, 0x00, 0x00, 0x40, 0x00, 0x40, 0x11,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x04, 0x00, 0x00, 0x03, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x08, 0x00,
 | |
| 	0x45, 0x00, 0x00, 0x20, 0x00, 0x01, 0x00, 0x00,
 | |
| 	0x40, 0x84, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| };
 | |
| 
 | |
| static const u8 ice_fdir_ip4_tun_pkt[] = {
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x45, 0x00,
 | |
| 	0x00, 0x46, 0x00, 0x00, 0x40, 0x00, 0x40, 0x11,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x04, 0x00, 0x00, 0x03, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x08, 0x00,
 | |
| 	0x45, 0x00, 0x00, 0x14, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x40, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00,
 | |
| };
 | |
| 
 | |
| static const u8 ice_fdir_tcp6_tun_pkt[] = {
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x45, 0x00,
 | |
| 	0x00, 0x6e, 0x00, 0x00, 0x40, 0x00, 0x40, 0x11,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x04, 0x00, 0x00, 0x03, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x86, 0xdd,
 | |
| 	0x60, 0x00, 0x00, 0x00, 0x00, 0x14, 0x06, 0x40,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x50, 0x00, 0x20, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00,
 | |
| };
 | |
| 
 | |
| static const u8 ice_fdir_udp6_tun_pkt[] = {
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x45, 0x00,
 | |
| 	0x00, 0x62, 0x00, 0x00, 0x40, 0x00, 0x40, 0x11,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x04, 0x00, 0x00, 0x03, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x86, 0xdd,
 | |
| 	0x60, 0x00, 0x00, 0x00, 0x00, 0x08, 0x11, 0x40,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| };
 | |
| 
 | |
| static const u8 ice_fdir_sctp6_tun_pkt[] = {
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x45, 0x00,
 | |
| 	0x00, 0x66, 0x00, 0x00, 0x40, 0x00, 0x40, 0x11,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x04, 0x00, 0x00, 0x03, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x86, 0xdd,
 | |
| 	0x60, 0x00, 0x00, 0x00, 0x00, 0x0c, 0x84, 0x40,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00,
 | |
| };
 | |
| 
 | |
| static const u8 ice_fdir_ip6_tun_pkt[] = {
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x45, 0x00,
 | |
| 	0x00, 0x5a, 0x00, 0x00, 0x40, 0x00, 0x40, 0x11,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x04, 0x00, 0x00, 0x03, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x86, 0xdd,
 | |
| 	0x60, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3b, 0x40,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| 	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 | |
| };
 | |
| 
 | |
| /* Flow Director no-op training packet table */
 | |
| static const struct ice_fdir_base_pkt ice_fdir_pkt[] = {
 | |
| 	{
 | |
| 		ICE_FLTR_PTYPE_NONF_IPV4_TCP,
 | |
| 		sizeof(ice_fdir_tcpv4_pkt), ice_fdir_tcpv4_pkt,
 | |
| 		sizeof(ice_fdir_tcp4_tun_pkt), ice_fdir_tcp4_tun_pkt,
 | |
| 	},
 | |
| 	{
 | |
| 		ICE_FLTR_PTYPE_NONF_IPV4_UDP,
 | |
| 		sizeof(ice_fdir_udpv4_pkt), ice_fdir_udpv4_pkt,
 | |
| 		sizeof(ice_fdir_udp4_tun_pkt), ice_fdir_udp4_tun_pkt,
 | |
| 	},
 | |
| 	{
 | |
| 		ICE_FLTR_PTYPE_NONF_IPV4_SCTP,
 | |
| 		sizeof(ice_fdir_sctpv4_pkt), ice_fdir_sctpv4_pkt,
 | |
| 		sizeof(ice_fdir_sctp4_tun_pkt), ice_fdir_sctp4_tun_pkt,
 | |
| 	},
 | |
| 	{
 | |
| 		ICE_FLTR_PTYPE_NONF_IPV4_OTHER,
 | |
| 		sizeof(ice_fdir_ipv4_pkt), ice_fdir_ipv4_pkt,
 | |
| 		sizeof(ice_fdir_ip4_tun_pkt), ice_fdir_ip4_tun_pkt,
 | |
| 	},
 | |
| 	{
 | |
| 		ICE_FLTR_PTYPE_NONF_IPV6_TCP,
 | |
| 		sizeof(ice_fdir_tcpv6_pkt), ice_fdir_tcpv6_pkt,
 | |
| 		sizeof(ice_fdir_tcp6_tun_pkt), ice_fdir_tcp6_tun_pkt,
 | |
| 	},
 | |
| 	{
 | |
| 		ICE_FLTR_PTYPE_NONF_IPV6_UDP,
 | |
| 		sizeof(ice_fdir_udpv6_pkt), ice_fdir_udpv6_pkt,
 | |
| 		sizeof(ice_fdir_udp6_tun_pkt), ice_fdir_udp6_tun_pkt,
 | |
| 	},
 | |
| 	{
 | |
| 		ICE_FLTR_PTYPE_NONF_IPV6_SCTP,
 | |
| 		sizeof(ice_fdir_sctpv6_pkt), ice_fdir_sctpv6_pkt,
 | |
| 		sizeof(ice_fdir_sctp6_tun_pkt), ice_fdir_sctp6_tun_pkt,
 | |
| 	},
 | |
| 	{
 | |
| 		ICE_FLTR_PTYPE_NONF_IPV6_OTHER,
 | |
| 		sizeof(ice_fdir_ipv6_pkt), ice_fdir_ipv6_pkt,
 | |
| 		sizeof(ice_fdir_ip6_tun_pkt), ice_fdir_ip6_tun_pkt,
 | |
| 	},
 | |
| };
 | |
| 
 | |
| #define ICE_FDIR_NUM_PKT ARRAY_SIZE(ice_fdir_pkt)
 | |
| 
 | |
| /**
 | |
|  * ice_set_dflt_val_fd_desc
 | |
|  * @fd_fltr_ctx: pointer to fd filter descriptor
 | |
|  */
 | |
| static void ice_set_dflt_val_fd_desc(struct ice_fd_fltr_desc_ctx *fd_fltr_ctx)
 | |
| {
 | |
| 	fd_fltr_ctx->comp_q = ICE_FXD_FLTR_QW0_COMP_Q_ZERO;
 | |
| 	fd_fltr_ctx->comp_report = ICE_FXD_FLTR_QW0_COMP_REPORT_SW_FAIL;
 | |
| 	fd_fltr_ctx->fd_space = ICE_FXD_FLTR_QW0_FD_SPACE_GUAR_BEST;
 | |
| 	fd_fltr_ctx->cnt_ena = ICE_FXD_FLTR_QW0_STAT_ENA_PKTS;
 | |
| 	fd_fltr_ctx->evict_ena = ICE_FXD_FLTR_QW0_EVICT_ENA_TRUE;
 | |
| 	fd_fltr_ctx->toq = ICE_FXD_FLTR_QW0_TO_Q_EQUALS_QINDEX;
 | |
| 	fd_fltr_ctx->toq_prio = ICE_FXD_FLTR_QW0_TO_Q_PRIO1;
 | |
| 	fd_fltr_ctx->dpu_recipe = ICE_FXD_FLTR_QW0_DPU_RECIPE_DFLT;
 | |
| 	fd_fltr_ctx->drop = ICE_FXD_FLTR_QW0_DROP_NO;
 | |
| 	fd_fltr_ctx->flex_prio = ICE_FXD_FLTR_QW0_FLEX_PRI_NONE;
 | |
| 	fd_fltr_ctx->flex_mdid = ICE_FXD_FLTR_QW0_FLEX_MDID0;
 | |
| 	fd_fltr_ctx->flex_val = ICE_FXD_FLTR_QW0_FLEX_VAL0;
 | |
| 	fd_fltr_ctx->dtype = ICE_TX_DESC_DTYPE_FLTR_PROG;
 | |
| 	fd_fltr_ctx->desc_prof_prio = ICE_FXD_FLTR_QW1_PROF_PRIO_ZERO;
 | |
| 	fd_fltr_ctx->desc_prof = ICE_FXD_FLTR_QW1_PROF_ZERO;
 | |
| 	fd_fltr_ctx->swap = ICE_FXD_FLTR_QW1_SWAP_SET;
 | |
| 	fd_fltr_ctx->fdid_prio = ICE_FXD_FLTR_QW1_FDID_PRI_ONE;
 | |
| 	fd_fltr_ctx->fdid_mdid = ICE_FXD_FLTR_QW1_FDID_MDID_FD;
 | |
| 	fd_fltr_ctx->fdid = ICE_FXD_FLTR_QW1_FDID_ZERO;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_set_fd_desc_val
 | |
|  * @ctx: pointer to fd filter descriptor context
 | |
|  * @fdir_desc: populated with fd filter descriptor values
 | |
|  */
 | |
| static void
 | |
| ice_set_fd_desc_val(struct ice_fd_fltr_desc_ctx *ctx,
 | |
| 		    struct ice_fltr_desc *fdir_desc)
 | |
| {
 | |
| 	u64 qword;
 | |
| 
 | |
| 	/* prep QW0 of FD filter programming desc */
 | |
| 	qword = ((u64)ctx->qindex << ICE_FXD_FLTR_QW0_QINDEX_S) &
 | |
| 		ICE_FXD_FLTR_QW0_QINDEX_M;
 | |
| 	qword |= ((u64)ctx->comp_q << ICE_FXD_FLTR_QW0_COMP_Q_S) &
 | |
| 		 ICE_FXD_FLTR_QW0_COMP_Q_M;
 | |
| 	qword |= ((u64)ctx->comp_report << ICE_FXD_FLTR_QW0_COMP_REPORT_S) &
 | |
| 		 ICE_FXD_FLTR_QW0_COMP_REPORT_M;
 | |
| 	qword |= ((u64)ctx->fd_space << ICE_FXD_FLTR_QW0_FD_SPACE_S) &
 | |
| 		 ICE_FXD_FLTR_QW0_FD_SPACE_M;
 | |
| 	qword |= ((u64)ctx->cnt_index << ICE_FXD_FLTR_QW0_STAT_CNT_S) &
 | |
| 		 ICE_FXD_FLTR_QW0_STAT_CNT_M;
 | |
| 	qword |= ((u64)ctx->cnt_ena << ICE_FXD_FLTR_QW0_STAT_ENA_S) &
 | |
| 		 ICE_FXD_FLTR_QW0_STAT_ENA_M;
 | |
| 	qword |= ((u64)ctx->evict_ena << ICE_FXD_FLTR_QW0_EVICT_ENA_S) &
 | |
| 		 ICE_FXD_FLTR_QW0_EVICT_ENA_M;
 | |
| 	qword |= ((u64)ctx->toq << ICE_FXD_FLTR_QW0_TO_Q_S) &
 | |
| 		 ICE_FXD_FLTR_QW0_TO_Q_M;
 | |
| 	qword |= ((u64)ctx->toq_prio << ICE_FXD_FLTR_QW0_TO_Q_PRI_S) &
 | |
| 		 ICE_FXD_FLTR_QW0_TO_Q_PRI_M;
 | |
| 	qword |= ((u64)ctx->dpu_recipe << ICE_FXD_FLTR_QW0_DPU_RECIPE_S) &
 | |
| 		 ICE_FXD_FLTR_QW0_DPU_RECIPE_M;
 | |
| 	qword |= ((u64)ctx->drop << ICE_FXD_FLTR_QW0_DROP_S) &
 | |
| 		 ICE_FXD_FLTR_QW0_DROP_M;
 | |
| 	qword |= ((u64)ctx->flex_prio << ICE_FXD_FLTR_QW0_FLEX_PRI_S) &
 | |
| 		 ICE_FXD_FLTR_QW0_FLEX_PRI_M;
 | |
| 	qword |= ((u64)ctx->flex_mdid << ICE_FXD_FLTR_QW0_FLEX_MDID_S) &
 | |
| 		 ICE_FXD_FLTR_QW0_FLEX_MDID_M;
 | |
| 	qword |= ((u64)ctx->flex_val << ICE_FXD_FLTR_QW0_FLEX_VAL_S) &
 | |
| 		 ICE_FXD_FLTR_QW0_FLEX_VAL_M;
 | |
| 	fdir_desc->qidx_compq_space_stat = cpu_to_le64(qword);
 | |
| 
 | |
| 	/* prep QW1 of FD filter programming desc */
 | |
| 	qword = ((u64)ctx->dtype << ICE_FXD_FLTR_QW1_DTYPE_S) &
 | |
| 		ICE_FXD_FLTR_QW1_DTYPE_M;
 | |
| 	qword |= ((u64)ctx->pcmd << ICE_FXD_FLTR_QW1_PCMD_S) &
 | |
| 		 ICE_FXD_FLTR_QW1_PCMD_M;
 | |
| 	qword |= ((u64)ctx->desc_prof_prio << ICE_FXD_FLTR_QW1_PROF_PRI_S) &
 | |
| 		 ICE_FXD_FLTR_QW1_PROF_PRI_M;
 | |
| 	qword |= ((u64)ctx->desc_prof << ICE_FXD_FLTR_QW1_PROF_S) &
 | |
| 		 ICE_FXD_FLTR_QW1_PROF_M;
 | |
| 	qword |= ((u64)ctx->fd_vsi << ICE_FXD_FLTR_QW1_FD_VSI_S) &
 | |
| 		 ICE_FXD_FLTR_QW1_FD_VSI_M;
 | |
| 	qword |= ((u64)ctx->swap << ICE_FXD_FLTR_QW1_SWAP_S) &
 | |
| 		 ICE_FXD_FLTR_QW1_SWAP_M;
 | |
| 	qword |= ((u64)ctx->fdid_prio << ICE_FXD_FLTR_QW1_FDID_PRI_S) &
 | |
| 		 ICE_FXD_FLTR_QW1_FDID_PRI_M;
 | |
| 	qword |= ((u64)ctx->fdid_mdid << ICE_FXD_FLTR_QW1_FDID_MDID_S) &
 | |
| 		 ICE_FXD_FLTR_QW1_FDID_MDID_M;
 | |
| 	qword |= ((u64)ctx->fdid << ICE_FXD_FLTR_QW1_FDID_S) &
 | |
| 		 ICE_FXD_FLTR_QW1_FDID_M;
 | |
| 	fdir_desc->dtype_cmd_vsi_fdid = cpu_to_le64(qword);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_get_prgm_desc - set a fdir descriptor from a fdir filter struct
 | |
|  * @hw: pointer to the hardware structure
 | |
|  * @input: filter
 | |
|  * @fdesc: filter descriptor
 | |
|  * @add: if add is true, this is an add operation, false implies delete
 | |
|  */
 | |
| void
 | |
| ice_fdir_get_prgm_desc(struct ice_hw *hw, struct ice_fdir_fltr *input,
 | |
| 		       struct ice_fltr_desc *fdesc, bool add)
 | |
| {
 | |
| 	struct ice_fd_fltr_desc_ctx fdir_fltr_ctx = { 0 };
 | |
| 
 | |
| 	/* set default context info */
 | |
| 	ice_set_dflt_val_fd_desc(&fdir_fltr_ctx);
 | |
| 
 | |
| 	/* change sideband filtering values */
 | |
| 	fdir_fltr_ctx.fdid = input->fltr_id;
 | |
| 	if (input->dest_ctl == ICE_FLTR_PRGM_DESC_DEST_DROP_PKT) {
 | |
| 		fdir_fltr_ctx.drop = ICE_FXD_FLTR_QW0_DROP_YES;
 | |
| 		fdir_fltr_ctx.qindex = 0;
 | |
| 	} else {
 | |
| 		fdir_fltr_ctx.drop = ICE_FXD_FLTR_QW0_DROP_NO;
 | |
| 		fdir_fltr_ctx.qindex = input->q_index;
 | |
| 	}
 | |
| 	fdir_fltr_ctx.cnt_ena = ICE_FXD_FLTR_QW0_STAT_ENA_PKTS;
 | |
| 	fdir_fltr_ctx.cnt_index = input->cnt_index;
 | |
| 	fdir_fltr_ctx.fd_vsi = ice_get_hw_vsi_num(hw, input->dest_vsi);
 | |
| 	fdir_fltr_ctx.evict_ena = ICE_FXD_FLTR_QW0_EVICT_ENA_FALSE;
 | |
| 	fdir_fltr_ctx.toq_prio = 3;
 | |
| 	fdir_fltr_ctx.pcmd = add ? ICE_FXD_FLTR_QW1_PCMD_ADD :
 | |
| 		ICE_FXD_FLTR_QW1_PCMD_REMOVE;
 | |
| 	fdir_fltr_ctx.swap = ICE_FXD_FLTR_QW1_SWAP_NOT_SET;
 | |
| 	fdir_fltr_ctx.comp_q = ICE_FXD_FLTR_QW0_COMP_Q_ZERO;
 | |
| 	fdir_fltr_ctx.comp_report = ICE_FXD_FLTR_QW0_COMP_REPORT_SW_FAIL;
 | |
| 	fdir_fltr_ctx.fdid_prio = 3;
 | |
| 	fdir_fltr_ctx.desc_prof = 1;
 | |
| 	fdir_fltr_ctx.desc_prof_prio = 3;
 | |
| 	ice_set_fd_desc_val(&fdir_fltr_ctx, fdesc);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_alloc_fd_res_cntr - obtain counter resource for FD type
 | |
|  * @hw: pointer to the hardware structure
 | |
|  * @cntr_id: returns counter index
 | |
|  */
 | |
| enum ice_status ice_alloc_fd_res_cntr(struct ice_hw *hw, u16 *cntr_id)
 | |
| {
 | |
| 	return ice_alloc_res_cntr(hw, ICE_AQC_RES_TYPE_FDIR_COUNTER_BLOCK,
 | |
| 				  ICE_AQC_RES_TYPE_FLAG_DEDICATED, 1, cntr_id);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_free_fd_res_cntr - Free counter resource for FD type
 | |
|  * @hw: pointer to the hardware structure
 | |
|  * @cntr_id: counter index to be freed
 | |
|  */
 | |
| enum ice_status ice_free_fd_res_cntr(struct ice_hw *hw, u16 cntr_id)
 | |
| {
 | |
| 	return ice_free_res_cntr(hw, ICE_AQC_RES_TYPE_FDIR_COUNTER_BLOCK,
 | |
| 				 ICE_AQC_RES_TYPE_FLAG_DEDICATED, 1, cntr_id);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_alloc_fd_guar_item - allocate resource for FD guaranteed entries
 | |
|  * @hw: pointer to the hardware structure
 | |
|  * @cntr_id: returns counter index
 | |
|  * @num_fltr: number of filter entries to be allocated
 | |
|  */
 | |
| enum ice_status
 | |
| ice_alloc_fd_guar_item(struct ice_hw *hw, u16 *cntr_id, u16 num_fltr)
 | |
| {
 | |
| 	return ice_alloc_res_cntr(hw, ICE_AQC_RES_TYPE_FDIR_GUARANTEED_ENTRIES,
 | |
| 				  ICE_AQC_RES_TYPE_FLAG_DEDICATED, num_fltr,
 | |
| 				  cntr_id);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_alloc_fd_shrd_item - allocate resource for flow director shared entries
 | |
|  * @hw: pointer to the hardware structure
 | |
|  * @cntr_id: returns counter index
 | |
|  * @num_fltr: number of filter entries to be allocated
 | |
|  */
 | |
| enum ice_status
 | |
| ice_alloc_fd_shrd_item(struct ice_hw *hw, u16 *cntr_id, u16 num_fltr)
 | |
| {
 | |
| 	return ice_alloc_res_cntr(hw, ICE_AQC_RES_TYPE_FDIR_SHARED_ENTRIES,
 | |
| 				  ICE_AQC_RES_TYPE_FLAG_DEDICATED, num_fltr,
 | |
| 				  cntr_id);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_get_fdir_cnt_all - get the number of Flow Director filters
 | |
|  * @hw: hardware data structure
 | |
|  *
 | |
|  * Returns the number of filters available on device
 | |
|  */
 | |
| int ice_get_fdir_cnt_all(struct ice_hw *hw)
 | |
| {
 | |
| 	return hw->func_caps.fd_fltr_guar + hw->func_caps.fd_fltr_best_effort;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_pkt_insert_ipv6_addr - insert a be32 IPv6 address into a memory buffer
 | |
|  * @pkt: packet buffer
 | |
|  * @offset: offset into buffer
 | |
|  * @addr: IPv6 address to convert and insert into pkt at offset
 | |
|  */
 | |
| static void ice_pkt_insert_ipv6_addr(u8 *pkt, int offset, __be32 *addr)
 | |
| {
 | |
| 	int idx;
 | |
| 
 | |
| 	for (idx = 0; idx < ICE_IPV6_ADDR_LEN_AS_U32; idx++)
 | |
| 		memcpy(pkt + offset + idx * sizeof(*addr), &addr[idx],
 | |
| 		       sizeof(*addr));
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_pkt_insert_u16 - insert a be16 value into a memory buffer
 | |
|  * @pkt: packet buffer
 | |
|  * @offset: offset into buffer
 | |
|  * @data: 16 bit value to convert and insert into pkt at offset
 | |
|  */
 | |
| static void ice_pkt_insert_u16(u8 *pkt, int offset, __be16 data)
 | |
| {
 | |
| 	memcpy(pkt + offset, &data, sizeof(data));
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_pkt_insert_u32 - insert a be32 value into a memory buffer
 | |
|  * @pkt: packet buffer
 | |
|  * @offset: offset into buffer
 | |
|  * @data: 32 bit value to convert and insert into pkt at offset
 | |
|  */
 | |
| static void ice_pkt_insert_u32(u8 *pkt, int offset, __be32 data)
 | |
| {
 | |
| 	memcpy(pkt + offset, &data, sizeof(data));
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_get_gen_prgm_pkt - generate a training packet
 | |
|  * @hw: pointer to the hardware structure
 | |
|  * @input: flow director filter data structure
 | |
|  * @pkt: pointer to return filter packet
 | |
|  * @frag: generate a fragment packet
 | |
|  * @tun: true implies generate a tunnel packet
 | |
|  */
 | |
| enum ice_status
 | |
| ice_fdir_get_gen_prgm_pkt(struct ice_hw *hw, struct ice_fdir_fltr *input,
 | |
| 			  u8 *pkt, bool frag, bool tun)
 | |
| {
 | |
| 	enum ice_fltr_ptype flow;
 | |
| 	u16 tnl_port;
 | |
| 	u8 *loc;
 | |
| 	u16 idx;
 | |
| 
 | |
| 	if (input->flow_type == ICE_FLTR_PTYPE_NONF_IPV4_OTHER) {
 | |
| 		switch (input->ip.v4.proto) {
 | |
| 		case IPPROTO_TCP:
 | |
| 			flow = ICE_FLTR_PTYPE_NONF_IPV4_TCP;
 | |
| 			break;
 | |
| 		case IPPROTO_UDP:
 | |
| 			flow = ICE_FLTR_PTYPE_NONF_IPV4_UDP;
 | |
| 			break;
 | |
| 		case IPPROTO_SCTP:
 | |
| 			flow = ICE_FLTR_PTYPE_NONF_IPV4_SCTP;
 | |
| 			break;
 | |
| 		case IPPROTO_IP:
 | |
| 			flow = ICE_FLTR_PTYPE_NONF_IPV4_OTHER;
 | |
| 			break;
 | |
| 		default:
 | |
| 			return ICE_ERR_PARAM;
 | |
| 		}
 | |
| 	} else if (input->flow_type == ICE_FLTR_PTYPE_NONF_IPV6_OTHER) {
 | |
| 		switch (input->ip.v6.proto) {
 | |
| 		case IPPROTO_TCP:
 | |
| 			flow = ICE_FLTR_PTYPE_NONF_IPV6_TCP;
 | |
| 			break;
 | |
| 		case IPPROTO_UDP:
 | |
| 			flow = ICE_FLTR_PTYPE_NONF_IPV6_UDP;
 | |
| 			break;
 | |
| 		case IPPROTO_SCTP:
 | |
| 			flow = ICE_FLTR_PTYPE_NONF_IPV6_SCTP;
 | |
| 			break;
 | |
| 		case IPPROTO_IP:
 | |
| 			flow = ICE_FLTR_PTYPE_NONF_IPV6_OTHER;
 | |
| 			break;
 | |
| 		default:
 | |
| 			return ICE_ERR_PARAM;
 | |
| 		}
 | |
| 	} else {
 | |
| 		flow = input->flow_type;
 | |
| 	}
 | |
| 
 | |
| 	for (idx = 0; idx < ICE_FDIR_NUM_PKT; idx++)
 | |
| 		if (ice_fdir_pkt[idx].flow == flow)
 | |
| 			break;
 | |
| 	if (idx == ICE_FDIR_NUM_PKT)
 | |
| 		return ICE_ERR_PARAM;
 | |
| 	if (!tun) {
 | |
| 		memcpy(pkt, ice_fdir_pkt[idx].pkt, ice_fdir_pkt[idx].pkt_len);
 | |
| 		loc = pkt;
 | |
| 	} else {
 | |
| 		if (!ice_get_open_tunnel_port(hw, TNL_ALL, &tnl_port))
 | |
| 			return ICE_ERR_DOES_NOT_EXIST;
 | |
| 		if (!ice_fdir_pkt[idx].tun_pkt)
 | |
| 			return ICE_ERR_PARAM;
 | |
| 		memcpy(pkt, ice_fdir_pkt[idx].tun_pkt,
 | |
| 		       ice_fdir_pkt[idx].tun_pkt_len);
 | |
| 		ice_pkt_insert_u16(pkt, ICE_IPV4_UDP_DST_PORT_OFFSET,
 | |
| 				   htons(tnl_port));
 | |
| 		loc = &pkt[ICE_FDIR_TUN_PKT_OFF];
 | |
| 	}
 | |
| 
 | |
| 	/* Reverse the src and dst, since the HW expects them to be from Tx
 | |
| 	 * perspective. The input from user is from Rx filter perspective.
 | |
| 	 */
 | |
| 	switch (flow) {
 | |
| 	case ICE_FLTR_PTYPE_NONF_IPV4_TCP:
 | |
| 		ice_pkt_insert_u32(loc, ICE_IPV4_DST_ADDR_OFFSET,
 | |
| 				   input->ip.v4.src_ip);
 | |
| 		ice_pkt_insert_u16(loc, ICE_IPV4_TCP_DST_PORT_OFFSET,
 | |
| 				   input->ip.v4.src_port);
 | |
| 		ice_pkt_insert_u32(loc, ICE_IPV4_SRC_ADDR_OFFSET,
 | |
| 				   input->ip.v4.dst_ip);
 | |
| 		ice_pkt_insert_u16(loc, ICE_IPV4_TCP_SRC_PORT_OFFSET,
 | |
| 				   input->ip.v4.dst_port);
 | |
| 		if (frag)
 | |
| 			loc[20] = ICE_FDIR_IPV4_PKT_FLAG_DF;
 | |
| 		break;
 | |
| 	case ICE_FLTR_PTYPE_NONF_IPV4_UDP:
 | |
| 		ice_pkt_insert_u32(loc, ICE_IPV4_DST_ADDR_OFFSET,
 | |
| 				   input->ip.v4.src_ip);
 | |
| 		ice_pkt_insert_u16(loc, ICE_IPV4_UDP_DST_PORT_OFFSET,
 | |
| 				   input->ip.v4.src_port);
 | |
| 		ice_pkt_insert_u32(loc, ICE_IPV4_SRC_ADDR_OFFSET,
 | |
| 				   input->ip.v4.dst_ip);
 | |
| 		ice_pkt_insert_u16(loc, ICE_IPV4_UDP_SRC_PORT_OFFSET,
 | |
| 				   input->ip.v4.dst_port);
 | |
| 		break;
 | |
| 	case ICE_FLTR_PTYPE_NONF_IPV4_SCTP:
 | |
| 		ice_pkt_insert_u32(loc, ICE_IPV4_DST_ADDR_OFFSET,
 | |
| 				   input->ip.v4.src_ip);
 | |
| 		ice_pkt_insert_u16(loc, ICE_IPV4_SCTP_DST_PORT_OFFSET,
 | |
| 				   input->ip.v4.src_port);
 | |
| 		ice_pkt_insert_u32(loc, ICE_IPV4_SRC_ADDR_OFFSET,
 | |
| 				   input->ip.v4.dst_ip);
 | |
| 		ice_pkt_insert_u16(loc, ICE_IPV4_SCTP_SRC_PORT_OFFSET,
 | |
| 				   input->ip.v4.dst_port);
 | |
| 		break;
 | |
| 	case ICE_FLTR_PTYPE_NONF_IPV4_OTHER:
 | |
| 		ice_pkt_insert_u32(loc, ICE_IPV4_DST_ADDR_OFFSET,
 | |
| 				   input->ip.v4.src_ip);
 | |
| 		ice_pkt_insert_u32(loc, ICE_IPV4_SRC_ADDR_OFFSET,
 | |
| 				   input->ip.v4.dst_ip);
 | |
| 		ice_pkt_insert_u16(loc, ICE_IPV4_PROTO_OFFSET, 0);
 | |
| 		break;
 | |
| 	case ICE_FLTR_PTYPE_NONF_IPV6_TCP:
 | |
| 		ice_pkt_insert_ipv6_addr(loc, ICE_IPV6_DST_ADDR_OFFSET,
 | |
| 					 input->ip.v6.src_ip);
 | |
| 		ice_pkt_insert_ipv6_addr(loc, ICE_IPV6_SRC_ADDR_OFFSET,
 | |
| 					 input->ip.v6.dst_ip);
 | |
| 		ice_pkt_insert_u16(loc, ICE_IPV6_TCP_DST_PORT_OFFSET,
 | |
| 				   input->ip.v6.src_port);
 | |
| 		ice_pkt_insert_u16(loc, ICE_IPV6_TCP_SRC_PORT_OFFSET,
 | |
| 				   input->ip.v6.dst_port);
 | |
| 		break;
 | |
| 	case ICE_FLTR_PTYPE_NONF_IPV6_UDP:
 | |
| 		ice_pkt_insert_ipv6_addr(loc, ICE_IPV6_DST_ADDR_OFFSET,
 | |
| 					 input->ip.v6.src_ip);
 | |
| 		ice_pkt_insert_ipv6_addr(loc, ICE_IPV6_SRC_ADDR_OFFSET,
 | |
| 					 input->ip.v6.dst_ip);
 | |
| 		ice_pkt_insert_u16(loc, ICE_IPV6_UDP_DST_PORT_OFFSET,
 | |
| 				   input->ip.v6.src_port);
 | |
| 		ice_pkt_insert_u16(loc, ICE_IPV6_UDP_SRC_PORT_OFFSET,
 | |
| 				   input->ip.v6.dst_port);
 | |
| 		break;
 | |
| 	case ICE_FLTR_PTYPE_NONF_IPV6_SCTP:
 | |
| 		ice_pkt_insert_ipv6_addr(loc, ICE_IPV6_DST_ADDR_OFFSET,
 | |
| 					 input->ip.v6.src_ip);
 | |
| 		ice_pkt_insert_ipv6_addr(loc, ICE_IPV6_SRC_ADDR_OFFSET,
 | |
| 					 input->ip.v6.dst_ip);
 | |
| 		ice_pkt_insert_u16(loc, ICE_IPV6_SCTP_DST_PORT_OFFSET,
 | |
| 				   input->ip.v6.src_port);
 | |
| 		ice_pkt_insert_u16(loc, ICE_IPV6_SCTP_SRC_PORT_OFFSET,
 | |
| 				   input->ip.v6.dst_port);
 | |
| 		break;
 | |
| 	case ICE_FLTR_PTYPE_NONF_IPV6_OTHER:
 | |
| 		ice_pkt_insert_ipv6_addr(loc, ICE_IPV6_DST_ADDR_OFFSET,
 | |
| 					 input->ip.v6.src_ip);
 | |
| 		ice_pkt_insert_ipv6_addr(loc, ICE_IPV6_SRC_ADDR_OFFSET,
 | |
| 					 input->ip.v6.dst_ip);
 | |
| 		break;
 | |
| 	default:
 | |
| 		return ICE_ERR_PARAM;
 | |
| 	}
 | |
| 
 | |
| 	if (input->flex_fltr)
 | |
| 		ice_pkt_insert_u16(loc, input->flex_offset, input->flex_word);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_has_frag - does flow type have 2 ptypes
 | |
|  * @flow: flow ptype
 | |
|  *
 | |
|  * returns true is there is a fragment packet for this ptype
 | |
|  */
 | |
| bool ice_fdir_has_frag(enum ice_fltr_ptype flow)
 | |
| {
 | |
| 	if (flow == ICE_FLTR_PTYPE_NONF_IPV4_OTHER)
 | |
| 		return true;
 | |
| 	else
 | |
| 		return false;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_find_by_idx - find filter with idx
 | |
|  * @hw: pointer to hardware structure
 | |
|  * @fltr_idx: index to find.
 | |
|  *
 | |
|  * Returns pointer to filter if found or null
 | |
|  */
 | |
| struct ice_fdir_fltr *
 | |
| ice_fdir_find_fltr_by_idx(struct ice_hw *hw, u32 fltr_idx)
 | |
| {
 | |
| 	struct ice_fdir_fltr *rule;
 | |
| 
 | |
| 	list_for_each_entry(rule, &hw->fdir_list_head, fltr_node) {
 | |
| 		/* rule ID found in the list */
 | |
| 		if (fltr_idx == rule->fltr_id)
 | |
| 			return rule;
 | |
| 		if (fltr_idx < rule->fltr_id)
 | |
| 			break;
 | |
| 	}
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_list_add_fltr - add a new node to the flow director filter list
 | |
|  * @hw: hardware structure
 | |
|  * @fltr: filter node to add to structure
 | |
|  */
 | |
| void ice_fdir_list_add_fltr(struct ice_hw *hw, struct ice_fdir_fltr *fltr)
 | |
| {
 | |
| 	struct ice_fdir_fltr *rule, *parent = NULL;
 | |
| 
 | |
| 	list_for_each_entry(rule, &hw->fdir_list_head, fltr_node) {
 | |
| 		/* rule ID found or pass its spot in the list */
 | |
| 		if (rule->fltr_id >= fltr->fltr_id)
 | |
| 			break;
 | |
| 		parent = rule;
 | |
| 	}
 | |
| 
 | |
| 	if (parent)
 | |
| 		list_add(&fltr->fltr_node, &parent->fltr_node);
 | |
| 	else
 | |
| 		list_add(&fltr->fltr_node, &hw->fdir_list_head);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_update_cntrs - increment / decrement filter counter
 | |
|  * @hw: pointer to hardware structure
 | |
|  * @flow: filter flow type
 | |
|  * @add: true implies filters added
 | |
|  */
 | |
| void
 | |
| ice_fdir_update_cntrs(struct ice_hw *hw, enum ice_fltr_ptype flow, bool add)
 | |
| {
 | |
| 	int incr;
 | |
| 
 | |
| 	incr = add ? 1 : -1;
 | |
| 	hw->fdir_active_fltr += incr;
 | |
| 
 | |
| 	if (flow == ICE_FLTR_PTYPE_NONF_NONE || flow >= ICE_FLTR_PTYPE_MAX)
 | |
| 		ice_debug(hw, ICE_DBG_SW, "Unknown filter type %d\n", flow);
 | |
| 	else
 | |
| 		hw->fdir_fltr_cnt[flow] += incr;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_cmp_ipv6_addr - compare 2 IP v6 addresses
 | |
|  * @a: IP v6 address
 | |
|  * @b: IP v6 address
 | |
|  *
 | |
|  * Returns 0 on equal, returns non-0 if different
 | |
|  */
 | |
| static int ice_cmp_ipv6_addr(__be32 *a, __be32 *b)
 | |
| {
 | |
| 	return memcmp(a, b, 4 * sizeof(__be32));
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_comp_rules - compare 2 filters
 | |
|  * @a: a Flow Director filter data structure
 | |
|  * @b: a Flow Director filter data structure
 | |
|  * @v6: bool true if v6 filter
 | |
|  *
 | |
|  * Returns true if the filters match
 | |
|  */
 | |
| static bool
 | |
| ice_fdir_comp_rules(struct ice_fdir_fltr *a,  struct ice_fdir_fltr *b, bool v6)
 | |
| {
 | |
| 	enum ice_fltr_ptype flow_type = a->flow_type;
 | |
| 
 | |
| 	/* The calling function already checks that the two filters have the
 | |
| 	 * same flow_type.
 | |
| 	 */
 | |
| 	if (!v6) {
 | |
| 		if (flow_type == ICE_FLTR_PTYPE_NONF_IPV4_TCP ||
 | |
| 		    flow_type == ICE_FLTR_PTYPE_NONF_IPV4_UDP ||
 | |
| 		    flow_type == ICE_FLTR_PTYPE_NONF_IPV4_SCTP) {
 | |
| 			if (a->ip.v4.dst_ip == b->ip.v4.dst_ip &&
 | |
| 			    a->ip.v4.src_ip == b->ip.v4.src_ip &&
 | |
| 			    a->ip.v4.dst_port == b->ip.v4.dst_port &&
 | |
| 			    a->ip.v4.src_port == b->ip.v4.src_port)
 | |
| 				return true;
 | |
| 		} else if (flow_type == ICE_FLTR_PTYPE_NONF_IPV4_OTHER) {
 | |
| 			if (a->ip.v4.dst_ip == b->ip.v4.dst_ip &&
 | |
| 			    a->ip.v4.src_ip == b->ip.v4.src_ip &&
 | |
| 			    a->ip.v4.l4_header == b->ip.v4.l4_header &&
 | |
| 			    a->ip.v4.proto == b->ip.v4.proto &&
 | |
| 			    a->ip.v4.ip_ver == b->ip.v4.ip_ver &&
 | |
| 			    a->ip.v4.tos == b->ip.v4.tos)
 | |
| 				return true;
 | |
| 		}
 | |
| 	} else {
 | |
| 		if (flow_type == ICE_FLTR_PTYPE_NONF_IPV6_UDP ||
 | |
| 		    flow_type == ICE_FLTR_PTYPE_NONF_IPV6_TCP ||
 | |
| 		    flow_type == ICE_FLTR_PTYPE_NONF_IPV6_SCTP) {
 | |
| 			if (a->ip.v6.dst_port == b->ip.v6.dst_port &&
 | |
| 			    a->ip.v6.src_port == b->ip.v6.src_port &&
 | |
| 			    !ice_cmp_ipv6_addr(a->ip.v6.dst_ip,
 | |
| 					       b->ip.v6.dst_ip) &&
 | |
| 			    !ice_cmp_ipv6_addr(a->ip.v6.src_ip,
 | |
| 					       b->ip.v6.src_ip))
 | |
| 				return true;
 | |
| 		} else if (flow_type == ICE_FLTR_PTYPE_NONF_IPV6_OTHER) {
 | |
| 			if (a->ip.v6.dst_port == b->ip.v6.dst_port &&
 | |
| 			    a->ip.v6.src_port == b->ip.v6.src_port)
 | |
| 				return true;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ice_fdir_is_dup_fltr - test if filter is already in list for PF
 | |
|  * @hw: hardware data structure
 | |
|  * @input: Flow Director filter data structure
 | |
|  *
 | |
|  * Returns true if the filter is found in the list
 | |
|  */
 | |
| bool ice_fdir_is_dup_fltr(struct ice_hw *hw, struct ice_fdir_fltr *input)
 | |
| {
 | |
| 	struct ice_fdir_fltr *rule;
 | |
| 	bool ret = false;
 | |
| 
 | |
| 	list_for_each_entry(rule, &hw->fdir_list_head, fltr_node) {
 | |
| 		enum ice_fltr_ptype flow_type;
 | |
| 
 | |
| 		if (rule->flow_type != input->flow_type)
 | |
| 			continue;
 | |
| 
 | |
| 		flow_type = input->flow_type;
 | |
| 		if (flow_type == ICE_FLTR_PTYPE_NONF_IPV4_TCP ||
 | |
| 		    flow_type == ICE_FLTR_PTYPE_NONF_IPV4_UDP ||
 | |
| 		    flow_type == ICE_FLTR_PTYPE_NONF_IPV4_SCTP ||
 | |
| 		    flow_type == ICE_FLTR_PTYPE_NONF_IPV4_OTHER)
 | |
| 			ret = ice_fdir_comp_rules(rule, input, false);
 | |
| 		else
 | |
| 			ret = ice_fdir_comp_rules(rule, input, true);
 | |
| 		if (ret) {
 | |
| 			if (rule->fltr_id == input->fltr_id &&
 | |
| 			    rule->q_index != input->q_index)
 | |
| 				ret = false;
 | |
| 			else
 | |
| 				break;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return ret;
 | |
| }
 |