| /* SPDX-License-Identifier: (GPL-2.0 OR MIT) */ |
| /* Microsemi Ocelot Switch driver |
| * Copyright (c) 2019 Microsemi Corporation |
| */ |
| |
| #ifndef _MSCC_OCELOT_ACE_H_ |
| #define _MSCC_OCELOT_ACE_H_ |
| |
| #include "ocelot.h" |
| #include "ocelot_police.h" |
| #include <net/sch_generic.h> |
| #include <net/pkt_cls.h> |
| |
| struct ocelot_ipv4 { |
| u8 addr[4]; |
| }; |
| |
| enum ocelot_vcap_bit { |
| OCELOT_VCAP_BIT_ANY, |
| OCELOT_VCAP_BIT_0, |
| OCELOT_VCAP_BIT_1 |
| }; |
| |
| struct ocelot_vcap_u8 { |
| u8 value[1]; |
| u8 mask[1]; |
| }; |
| |
| struct ocelot_vcap_u16 { |
| u8 value[2]; |
| u8 mask[2]; |
| }; |
| |
| struct ocelot_vcap_u24 { |
| u8 value[3]; |
| u8 mask[3]; |
| }; |
| |
| struct ocelot_vcap_u32 { |
| u8 value[4]; |
| u8 mask[4]; |
| }; |
| |
| struct ocelot_vcap_u40 { |
| u8 value[5]; |
| u8 mask[5]; |
| }; |
| |
| struct ocelot_vcap_u48 { |
| u8 value[6]; |
| u8 mask[6]; |
| }; |
| |
| struct ocelot_vcap_u64 { |
| u8 value[8]; |
| u8 mask[8]; |
| }; |
| |
| struct ocelot_vcap_u128 { |
| u8 value[16]; |
| u8 mask[16]; |
| }; |
| |
| struct ocelot_vcap_vid { |
| u16 value; |
| u16 mask; |
| }; |
| |
| struct ocelot_vcap_ipv4 { |
| struct ocelot_ipv4 value; |
| struct ocelot_ipv4 mask; |
| }; |
| |
| struct ocelot_vcap_udp_tcp { |
| u16 value; |
| u16 mask; |
| }; |
| |
| enum ocelot_ace_type { |
| OCELOT_ACE_TYPE_ANY, |
| OCELOT_ACE_TYPE_ETYPE, |
| OCELOT_ACE_TYPE_LLC, |
| OCELOT_ACE_TYPE_SNAP, |
| OCELOT_ACE_TYPE_ARP, |
| OCELOT_ACE_TYPE_IPV4, |
| OCELOT_ACE_TYPE_IPV6 |
| }; |
| |
| struct ocelot_ace_vlan { |
| struct ocelot_vcap_vid vid; /* VLAN ID (12 bit) */ |
| struct ocelot_vcap_u8 pcp; /* PCP (3 bit) */ |
| enum ocelot_vcap_bit dei; /* DEI */ |
| enum ocelot_vcap_bit tagged; /* Tagged/untagged frame */ |
| }; |
| |
| struct ocelot_ace_frame_etype { |
| struct ocelot_vcap_u48 dmac; |
| struct ocelot_vcap_u48 smac; |
| struct ocelot_vcap_u16 etype; |
| struct ocelot_vcap_u16 data; /* MAC data */ |
| }; |
| |
| struct ocelot_ace_frame_llc { |
| struct ocelot_vcap_u48 dmac; |
| struct ocelot_vcap_u48 smac; |
| |
| /* LLC header: DSAP at byte 0, SSAP at byte 1, Control at byte 2 */ |
| struct ocelot_vcap_u32 llc; |
| }; |
| |
| struct ocelot_ace_frame_snap { |
| struct ocelot_vcap_u48 dmac; |
| struct ocelot_vcap_u48 smac; |
| |
| /* SNAP header: Organization Code at byte 0, Type at byte 3 */ |
| struct ocelot_vcap_u40 snap; |
| }; |
| |
| struct ocelot_ace_frame_arp { |
| struct ocelot_vcap_u48 smac; |
| enum ocelot_vcap_bit arp; /* Opcode ARP/RARP */ |
| enum ocelot_vcap_bit req; /* Opcode request/reply */ |
| enum ocelot_vcap_bit unknown; /* Opcode unknown */ |
| enum ocelot_vcap_bit smac_match; /* Sender MAC matches SMAC */ |
| enum ocelot_vcap_bit dmac_match; /* Target MAC matches DMAC */ |
| |
| /**< Protocol addr. length 4, hardware length 6 */ |
| enum ocelot_vcap_bit length; |
| |
| enum ocelot_vcap_bit ip; /* Protocol address type IP */ |
| enum ocelot_vcap_bit ethernet; /* Hardware address type Ethernet */ |
| struct ocelot_vcap_ipv4 sip; /* Sender IP address */ |
| struct ocelot_vcap_ipv4 dip; /* Target IP address */ |
| }; |
| |
| struct ocelot_ace_frame_ipv4 { |
| enum ocelot_vcap_bit ttl; /* TTL zero */ |
| enum ocelot_vcap_bit fragment; /* Fragment */ |
| enum ocelot_vcap_bit options; /* Header options */ |
| struct ocelot_vcap_u8 ds; |
| struct ocelot_vcap_u8 proto; /* Protocol */ |
| struct ocelot_vcap_ipv4 sip; /* Source IP address */ |
| struct ocelot_vcap_ipv4 dip; /* Destination IP address */ |
| struct ocelot_vcap_u48 data; /* Not UDP/TCP: IP data */ |
| struct ocelot_vcap_udp_tcp sport; /* UDP/TCP: Source port */ |
| struct ocelot_vcap_udp_tcp dport; /* UDP/TCP: Destination port */ |
| enum ocelot_vcap_bit tcp_fin; |
| enum ocelot_vcap_bit tcp_syn; |
| enum ocelot_vcap_bit tcp_rst; |
| enum ocelot_vcap_bit tcp_psh; |
| enum ocelot_vcap_bit tcp_ack; |
| enum ocelot_vcap_bit tcp_urg; |
| enum ocelot_vcap_bit sip_eq_dip; /* SIP equals DIP */ |
| enum ocelot_vcap_bit sport_eq_dport; /* SPORT equals DPORT */ |
| enum ocelot_vcap_bit seq_zero; /* TCP sequence number is zero */ |
| }; |
| |
| struct ocelot_ace_frame_ipv6 { |
| struct ocelot_vcap_u8 proto; /* IPv6 protocol */ |
| struct ocelot_vcap_u128 sip; /* IPv6 source (byte 0-7 ignored) */ |
| enum ocelot_vcap_bit ttl; /* TTL zero */ |
| struct ocelot_vcap_u8 ds; |
| struct ocelot_vcap_u48 data; /* Not UDP/TCP: IP data */ |
| struct ocelot_vcap_udp_tcp sport; |
| struct ocelot_vcap_udp_tcp dport; |
| enum ocelot_vcap_bit tcp_fin; |
| enum ocelot_vcap_bit tcp_syn; |
| enum ocelot_vcap_bit tcp_rst; |
| enum ocelot_vcap_bit tcp_psh; |
| enum ocelot_vcap_bit tcp_ack; |
| enum ocelot_vcap_bit tcp_urg; |
| enum ocelot_vcap_bit sip_eq_dip; /* SIP equals DIP */ |
| enum ocelot_vcap_bit sport_eq_dport; /* SPORT equals DPORT */ |
| enum ocelot_vcap_bit seq_zero; /* TCP sequence number is zero */ |
| }; |
| |
| enum ocelot_ace_action { |
| OCELOT_ACL_ACTION_DROP, |
| OCELOT_ACL_ACTION_TRAP, |
| OCELOT_ACL_ACTION_POLICE, |
| }; |
| |
| struct ocelot_ace_stats { |
| u64 bytes; |
| u64 pkts; |
| u64 used; |
| }; |
| |
| struct ocelot_ace_rule { |
| struct list_head list; |
| |
| u16 prio; |
| u32 id; |
| |
| enum ocelot_ace_action action; |
| struct ocelot_ace_stats stats; |
| unsigned long ingress_port_mask; |
| |
| enum ocelot_vcap_bit dmac_mc; |
| enum ocelot_vcap_bit dmac_bc; |
| struct ocelot_ace_vlan vlan; |
| |
| enum ocelot_ace_type type; |
| union { |
| /* ocelot_ACE_TYPE_ANY: No specific fields */ |
| struct ocelot_ace_frame_etype etype; |
| struct ocelot_ace_frame_llc llc; |
| struct ocelot_ace_frame_snap snap; |
| struct ocelot_ace_frame_arp arp; |
| struct ocelot_ace_frame_ipv4 ipv4; |
| struct ocelot_ace_frame_ipv6 ipv6; |
| } frame; |
| struct ocelot_policer pol; |
| u32 pol_ix; |
| }; |
| |
| int ocelot_ace_rule_offload_add(struct ocelot *ocelot, |
| struct ocelot_ace_rule *rule, |
| struct netlink_ext_ack *extack); |
| int ocelot_ace_rule_offload_del(struct ocelot *ocelot, |
| struct ocelot_ace_rule *rule); |
| int ocelot_ace_rule_stats_update(struct ocelot *ocelot, |
| struct ocelot_ace_rule *rule); |
| |
| int ocelot_ace_init(struct ocelot *ocelot); |
| |
| int ocelot_setup_tc_cls_flower(struct ocelot_port_private *priv, |
| struct flow_cls_offload *f, |
| bool ingress); |
| |
| #endif /* _MSCC_OCELOT_ACE_H_ */ |