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; 148d514f41eSJiri Pirko struct dentry *ddir; 149ab1d0cc0SJiri Pirko struct dentry *ports_ddir; 150d514f41eSJiri Pirko struct bpf_offload_dev *bpf_dev; 151d514f41eSJiri Pirko bool bpf_bind_accept; 152d514f41eSJiri Pirko u32 bpf_bind_verifier_delay; 153d514f41eSJiri Pirko struct dentry *ddir_bpf_bound_progs; 154d514f41eSJiri Pirko u32 prog_id_gen; 155d514f41eSJiri Pirko struct list_head bpf_bound_progs; 156d514f41eSJiri Pirko struct list_head bpf_bound_maps; 157514cf64cSJiri Pirko struct netdev_phys_item_id switch_id; 1588320d145SJiri Pirko struct list_head port_list; 159794b2c05SJiri Pirko struct mutex port_list_lock; /* protects port list */ 160*fa4dfc4aSJiri Pirko bool fw_update_status; 161a60f9e48SJiri Pirko }; 162a60f9e48SJiri Pirko 163d514f41eSJiri Pirko int nsim_dev_init(void); 164d514f41eSJiri Pirko void nsim_dev_exit(void); 1658320d145SJiri Pirko int nsim_dev_probe(struct nsim_bus_dev *nsim_bus_dev); 1668320d145SJiri Pirko void nsim_dev_remove(struct nsim_bus_dev *nsim_bus_dev); 167794b2c05SJiri Pirko int nsim_dev_port_add(struct nsim_bus_dev *nsim_bus_dev, 168794b2c05SJiri Pirko unsigned int port_index); 169794b2c05SJiri Pirko int nsim_dev_port_del(struct nsim_bus_dev *nsim_bus_dev, 170794b2c05SJiri Pirko unsigned int port_index); 17137923ed6SDavid Ahern 1725fc49422SJiri Pirko struct nsim_fib_data *nsim_fib_create(void); 1735fc49422SJiri Pirko void nsim_fib_destroy(struct nsim_fib_data *fib_data); 1745fc49422SJiri Pirko u64 nsim_fib_get_val(struct nsim_fib_data *fib_data, 1755fc49422SJiri Pirko enum nsim_resource_id res_id, bool max); 1765fc49422SJiri Pirko int nsim_fib_set_max(struct nsim_fib_data *fib_data, 1775fc49422SJiri Pirko enum nsim_resource_id res_id, u64 val, 1787fa76d77SDavid Ahern struct netlink_ext_ack *extack); 17937923ed6SDavid Ahern 1807699353dSShannon Nelson #if IS_ENABLED(CONFIG_XFRM_OFFLOAD) 1817699353dSShannon Nelson void nsim_ipsec_init(struct netdevsim *ns); 1827699353dSShannon Nelson void nsim_ipsec_teardown(struct netdevsim *ns); 1837699353dSShannon Nelson bool nsim_ipsec_tx(struct netdevsim *ns, struct sk_buff *skb); 1847699353dSShannon Nelson #else 1857699353dSShannon Nelson static inline void nsim_ipsec_init(struct netdevsim *ns) 1867699353dSShannon Nelson { 1877699353dSShannon Nelson } 1887699353dSShannon Nelson 1897699353dSShannon Nelson static inline void nsim_ipsec_teardown(struct netdevsim *ns) 1907699353dSShannon Nelson { 1917699353dSShannon Nelson } 1927699353dSShannon Nelson 1937699353dSShannon Nelson static inline bool nsim_ipsec_tx(struct netdevsim *ns, struct sk_buff *skb) 1947699353dSShannon Nelson { 1957699353dSShannon Nelson return true; 1967699353dSShannon Nelson } 1977699353dSShannon Nelson #endif 1987699353dSShannon Nelson 19940e4fe4cSJiri Pirko struct nsim_vf_config { 20040e4fe4cSJiri Pirko int link_state; 20140e4fe4cSJiri Pirko u16 min_tx_rate; 20240e4fe4cSJiri Pirko u16 max_tx_rate; 20340e4fe4cSJiri Pirko u16 vlan; 20440e4fe4cSJiri Pirko __be16 vlan_proto; 20540e4fe4cSJiri Pirko u16 qos; 20640e4fe4cSJiri Pirko u8 vf_mac[ETH_ALEN]; 20740e4fe4cSJiri Pirko bool spoofchk_enabled; 20840e4fe4cSJiri Pirko bool trusted; 20940e4fe4cSJiri Pirko bool rss_query_enabled; 21040e4fe4cSJiri Pirko }; 211925f5afeSJiri Pirko 21240e4fe4cSJiri Pirko struct nsim_bus_dev { 21340e4fe4cSJiri Pirko struct device dev; 214f9d9db47SJiri Pirko struct list_head list; 215f9d9db47SJiri Pirko unsigned int port_count; 21640e4fe4cSJiri Pirko unsigned int num_vfs; 21740e4fe4cSJiri Pirko struct nsim_vf_config *vfconfigs; 21840e4fe4cSJiri Pirko }; 219925f5afeSJiri Pirko 220925f5afeSJiri Pirko int nsim_bus_init(void); 221925f5afeSJiri Pirko void nsim_bus_exit(void); 222