183c9e13aSJakub Kicinski /* 283c9e13aSJakub Kicinski * Copyright (C) 2017 Netronome Systems, Inc. 383c9e13aSJakub Kicinski * 483c9e13aSJakub Kicinski * This software is licensed under the GNU General License Version 2, 583c9e13aSJakub Kicinski * June 1991 as shown in the file COPYING in the top-level directory of this 683c9e13aSJakub Kicinski * source tree. 783c9e13aSJakub Kicinski * 883c9e13aSJakub Kicinski * THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" 983c9e13aSJakub Kicinski * WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, 1083c9e13aSJakub Kicinski * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 1183c9e13aSJakub Kicinski * FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE 1283c9e13aSJakub Kicinski * OF THE PROGRAM IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME 1383c9e13aSJakub Kicinski * THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION. 1483c9e13aSJakub Kicinski */ 1583c9e13aSJakub Kicinski 1679579220SJakub Kicinski #include <linux/device.h> 1783c9e13aSJakub Kicinski #include <linux/kernel.h> 1831d3ad83SJakub Kicinski #include <linux/list.h> 1983c9e13aSJakub Kicinski #include <linux/netdevice.h> 2083c9e13aSJakub Kicinski #include <linux/u64_stats_sync.h> 218320d145SJiri Pirko #include <net/devlink.h> 2205296620SJakub Kicinski #include <net/xdp.h> 2383c9e13aSJakub Kicinski 2483c9e13aSJakub Kicinski #define DRV_NAME "netdevsim" 2583c9e13aSJakub Kicinski 2631d3ad83SJakub Kicinski #define NSIM_XDP_MAX_MTU 4000 2731d3ad83SJakub Kicinski 2831d3ad83SJakub Kicinski #define NSIM_EA(extack, msg) NL_SET_ERR_MSG_MOD((extack), msg) 2931d3ad83SJakub Kicinski 307699353dSShannon Nelson #define NSIM_IPSEC_MAX_SA_COUNT 33 317699353dSShannon Nelson #define NSIM_IPSEC_VALID BIT(31) 327699353dSShannon Nelson 337699353dSShannon Nelson struct nsim_sa { 347699353dSShannon Nelson struct xfrm_state *xs; 357699353dSShannon Nelson __be32 ipaddr[4]; 367699353dSShannon Nelson u32 key[4]; 377699353dSShannon Nelson u32 salt; 387699353dSShannon Nelson bool used; 397699353dSShannon Nelson bool crypt; 407699353dSShannon Nelson bool rx; 417699353dSShannon Nelson }; 427699353dSShannon Nelson 437699353dSShannon Nelson struct nsim_ipsec { 447699353dSShannon Nelson struct nsim_sa sa[NSIM_IPSEC_MAX_SA_COUNT]; 457699353dSShannon Nelson struct dentry *pfile; 467699353dSShannon Nelson u32 count; 477699353dSShannon Nelson u32 tx; 487699353dSShannon Nelson u32 ok; 497699353dSShannon Nelson }; 507699353dSShannon Nelson 5183c9e13aSJakub Kicinski struct netdevsim { 5231d3ad83SJakub Kicinski struct net_device *netdev; 53a60f9e48SJiri Pirko struct nsim_dev *nsim_dev; 54e05b2d14SJiri Pirko struct nsim_dev_port *nsim_dev_port; 5531d3ad83SJakub Kicinski 5683c9e13aSJakub Kicinski u64 tx_packets; 5783c9e13aSJakub Kicinski u64 tx_bytes; 5883c9e13aSJakub Kicinski struct u64_stats_sync syncp; 5931d3ad83SJakub Kicinski 6040e4fe4cSJiri Pirko struct nsim_bus_dev *nsim_bus_dev; 6179579220SJakub Kicinski 6231d3ad83SJakub Kicinski struct bpf_prog *bpf_offloaded; 6331d3ad83SJakub Kicinski u32 bpf_offloaded_id; 6431d3ad83SJakub Kicinski 6505296620SJakub Kicinski struct xdp_attachment_info xdp; 66799e173dSJakub Kicinski struct xdp_attachment_info xdp_hw; 6731d3ad83SJakub Kicinski 6831d3ad83SJakub Kicinski bool bpf_tc_accept; 6931d3ad83SJakub Kicinski bool bpf_tc_non_bound_accept; 7031d3ad83SJakub Kicinski bool bpf_xdpdrv_accept; 7131d3ad83SJakub Kicinski bool bpf_xdpoffload_accept; 72395cacb5SJakub Kicinski 73395cacb5SJakub Kicinski bool bpf_map_accept; 747699353dSShannon Nelson struct nsim_ipsec ipsec; 7583c9e13aSJakub Kicinski }; 7631d3ad83SJakub Kicinski 77e05b2d14SJiri Pirko struct netdevsim * 78e05b2d14SJiri Pirko nsim_create(struct nsim_dev *nsim_dev, struct nsim_dev_port *nsim_dev_port); 79e05b2d14SJiri Pirko void nsim_destroy(struct netdevsim *ns); 80e05b2d14SJiri Pirko 817c5db7e7SJakub Kicinski #ifdef CONFIG_BPF_SYSCALL 82d514f41eSJiri Pirko int nsim_bpf_dev_init(struct nsim_dev *nsim_dev); 83d514f41eSJiri Pirko void nsim_bpf_dev_exit(struct nsim_dev *nsim_dev); 8431d3ad83SJakub Kicinski int nsim_bpf_init(struct netdevsim *ns); 8531d3ad83SJakub Kicinski void nsim_bpf_uninit(struct netdevsim *ns); 8631d3ad83SJakub Kicinski int nsim_bpf(struct net_device *dev, struct netdev_bpf *bpf); 8731d3ad83SJakub Kicinski int nsim_bpf_disable_tc(struct netdevsim *ns); 8831d3ad83SJakub Kicinski int nsim_bpf_setup_tc_block_cb(enum tc_setup_type type, 8931d3ad83SJakub Kicinski void *type_data, void *cb_priv); 907c5db7e7SJakub Kicinski #else 91d514f41eSJiri Pirko 92d514f41eSJiri Pirko static inline int nsim_bpf_dev_init(struct nsim_dev *nsim_dev) 93d514f41eSJiri Pirko { 94d514f41eSJiri Pirko return 0; 95d514f41eSJiri Pirko } 96d514f41eSJiri Pirko 97d514f41eSJiri Pirko static inline void nsim_bpf_dev_exit(struct nsim_dev *nsim_dev) 98d514f41eSJiri Pirko { 99d514f41eSJiri Pirko } 1007c5db7e7SJakub Kicinski static inline int nsim_bpf_init(struct netdevsim *ns) 1017c5db7e7SJakub Kicinski { 1027c5db7e7SJakub Kicinski return 0; 1037c5db7e7SJakub Kicinski } 1047c5db7e7SJakub Kicinski 1057c5db7e7SJakub Kicinski static inline void nsim_bpf_uninit(struct netdevsim *ns) 1067c5db7e7SJakub Kicinski { 1077c5db7e7SJakub Kicinski } 1087c5db7e7SJakub Kicinski 1097c5db7e7SJakub Kicinski static inline int nsim_bpf(struct net_device *dev, struct netdev_bpf *bpf) 1107c5db7e7SJakub Kicinski { 1117c5db7e7SJakub Kicinski return bpf->command == XDP_QUERY_PROG ? 0 : -EOPNOTSUPP; 1127c5db7e7SJakub Kicinski } 1137c5db7e7SJakub Kicinski 1147c5db7e7SJakub Kicinski static inline int nsim_bpf_disable_tc(struct netdevsim *ns) 1157c5db7e7SJakub Kicinski { 1167c5db7e7SJakub Kicinski return 0; 1177c5db7e7SJakub Kicinski } 1187c5db7e7SJakub Kicinski 1197c5db7e7SJakub Kicinski static inline int 1207c5db7e7SJakub Kicinski nsim_bpf_setup_tc_block_cb(enum tc_setup_type type, void *type_data, 1217c5db7e7SJakub Kicinski void *cb_priv) 1227c5db7e7SJakub Kicinski { 1237c5db7e7SJakub Kicinski return -EOPNOTSUPP; 1247c5db7e7SJakub Kicinski } 1257c5db7e7SJakub Kicinski #endif 12679579220SJakub Kicinski 12737923ed6SDavid Ahern enum nsim_resource_id { 12837923ed6SDavid Ahern NSIM_RESOURCE_NONE, /* DEVLINK_RESOURCE_ID_PARENT_TOP */ 12937923ed6SDavid Ahern NSIM_RESOURCE_IPV4, 13037923ed6SDavid Ahern NSIM_RESOURCE_IPV4_FIB, 13137923ed6SDavid Ahern NSIM_RESOURCE_IPV4_FIB_RULES, 13237923ed6SDavid Ahern NSIM_RESOURCE_IPV6, 13337923ed6SDavid Ahern NSIM_RESOURCE_IPV6_FIB, 13437923ed6SDavid Ahern NSIM_RESOURCE_IPV6_FIB_RULES, 13537923ed6SDavid Ahern }; 13637923ed6SDavid Ahern 1378320d145SJiri Pirko struct nsim_dev_port { 1388320d145SJiri Pirko struct list_head list; 1398320d145SJiri Pirko struct devlink_port devlink_port; 1408320d145SJiri Pirko unsigned int port_index; 1418320d145SJiri Pirko struct dentry *ddir; 142e05b2d14SJiri Pirko struct netdevsim *ns; 1438320d145SJiri Pirko }; 1448320d145SJiri Pirko 145a60f9e48SJiri Pirko struct nsim_dev { 146d514f41eSJiri Pirko struct nsim_bus_dev *nsim_bus_dev; 147a60f9e48SJiri Pirko struct nsim_fib_data *fib_data; 148da58f90fSIdo Schimmel struct nsim_trap_data *trap_data; 149d514f41eSJiri Pirko struct dentry *ddir; 150ab1d0cc0SJiri Pirko struct dentry *ports_ddir; 151d514f41eSJiri Pirko struct bpf_offload_dev *bpf_dev; 152d514f41eSJiri Pirko bool bpf_bind_accept; 153d514f41eSJiri Pirko u32 bpf_bind_verifier_delay; 154d514f41eSJiri Pirko struct dentry *ddir_bpf_bound_progs; 155d514f41eSJiri Pirko u32 prog_id_gen; 156d514f41eSJiri Pirko struct list_head bpf_bound_progs; 157d514f41eSJiri Pirko struct list_head bpf_bound_maps; 158514cf64cSJiri Pirko struct netdev_phys_item_id switch_id; 1598320d145SJiri Pirko struct list_head port_list; 160794b2c05SJiri Pirko struct mutex port_list_lock; /* protects port list */ 161fa4dfc4aSJiri Pirko bool fw_update_status; 162150e8f8aSJiri Pirko u32 max_macs; 163150e8f8aSJiri Pirko bool test1; 164*155ddfc5SJiri Pirko bool dont_allow_reload; 165*155ddfc5SJiri Pirko bool fail_reload; 1664418f862SJiri Pirko struct devlink_region *dummy_region; 167a60f9e48SJiri Pirko }; 168a60f9e48SJiri Pirko 16990d29913SJiri Pirko static inline struct net *nsim_dev_net(struct nsim_dev *nsim_dev) 17090d29913SJiri Pirko { 17190d29913SJiri Pirko return devlink_net(priv_to_devlink(nsim_dev)); 17290d29913SJiri Pirko } 17390d29913SJiri Pirko 174d514f41eSJiri Pirko int nsim_dev_init(void); 175d514f41eSJiri Pirko void nsim_dev_exit(void); 1768320d145SJiri Pirko int nsim_dev_probe(struct nsim_bus_dev *nsim_bus_dev); 1778320d145SJiri Pirko void nsim_dev_remove(struct nsim_bus_dev *nsim_bus_dev); 178794b2c05SJiri Pirko int nsim_dev_port_add(struct nsim_bus_dev *nsim_bus_dev, 179794b2c05SJiri Pirko unsigned int port_index); 180794b2c05SJiri Pirko int nsim_dev_port_del(struct nsim_bus_dev *nsim_bus_dev, 181794b2c05SJiri Pirko unsigned int port_index); 18237923ed6SDavid Ahern 18375ba029fSJiri Pirko struct nsim_fib_data *nsim_fib_create(struct devlink *devlink, 18475ba029fSJiri Pirko struct netlink_ext_ack *extack); 18575ba029fSJiri Pirko void nsim_fib_destroy(struct devlink *devlink, struct nsim_fib_data *fib_data); 186a5facc4cSJiri Pirko u64 nsim_fib_get_val(struct nsim_fib_data *fib_data, 187a5facc4cSJiri Pirko enum nsim_resource_id res_id, bool max); 18837923ed6SDavid Ahern 1897699353dSShannon Nelson #if IS_ENABLED(CONFIG_XFRM_OFFLOAD) 1907699353dSShannon Nelson void nsim_ipsec_init(struct netdevsim *ns); 1917699353dSShannon Nelson void nsim_ipsec_teardown(struct netdevsim *ns); 1927699353dSShannon Nelson bool nsim_ipsec_tx(struct netdevsim *ns, struct sk_buff *skb); 1937699353dSShannon Nelson #else 1947699353dSShannon Nelson static inline void nsim_ipsec_init(struct netdevsim *ns) 1957699353dSShannon Nelson { 1967699353dSShannon Nelson } 1977699353dSShannon Nelson 1987699353dSShannon Nelson static inline void nsim_ipsec_teardown(struct netdevsim *ns) 1997699353dSShannon Nelson { 2007699353dSShannon Nelson } 2017699353dSShannon Nelson 2027699353dSShannon Nelson static inline bool nsim_ipsec_tx(struct netdevsim *ns, struct sk_buff *skb) 2037699353dSShannon Nelson { 2047699353dSShannon Nelson return true; 2057699353dSShannon Nelson } 2067699353dSShannon Nelson #endif 2077699353dSShannon Nelson 20840e4fe4cSJiri Pirko struct nsim_vf_config { 20940e4fe4cSJiri Pirko int link_state; 21040e4fe4cSJiri Pirko u16 min_tx_rate; 21140e4fe4cSJiri Pirko u16 max_tx_rate; 21240e4fe4cSJiri Pirko u16 vlan; 21340e4fe4cSJiri Pirko __be16 vlan_proto; 21440e4fe4cSJiri Pirko u16 qos; 21540e4fe4cSJiri Pirko u8 vf_mac[ETH_ALEN]; 21640e4fe4cSJiri Pirko bool spoofchk_enabled; 21740e4fe4cSJiri Pirko bool trusted; 21840e4fe4cSJiri Pirko bool rss_query_enabled; 21940e4fe4cSJiri Pirko }; 220925f5afeSJiri Pirko 22140e4fe4cSJiri Pirko struct nsim_bus_dev { 22240e4fe4cSJiri Pirko struct device dev; 223f9d9db47SJiri Pirko struct list_head list; 224f9d9db47SJiri Pirko unsigned int port_count; 2257b60027bSJiri Pirko struct net *initial_net; /* Purpose of this is to carry net pointer 2267b60027bSJiri Pirko * during the probe time only. 2277b60027bSJiri Pirko */ 22840e4fe4cSJiri Pirko unsigned int num_vfs; 22940e4fe4cSJiri Pirko struct nsim_vf_config *vfconfigs; 23040e4fe4cSJiri Pirko }; 231925f5afeSJiri Pirko 232925f5afeSJiri Pirko int nsim_bus_init(void); 233925f5afeSJiri Pirko void nsim_bus_exit(void); 234