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 16424be63aSJakub Kicinski #include <linux/debugfs.h> 1779579220SJakub Kicinski #include <linux/device.h> 18*a7fc6db0SAntonio Cardace #include <linux/ethtool.h> 1983c9e13aSJakub Kicinski #include <linux/kernel.h> 2031d3ad83SJakub Kicinski #include <linux/list.h> 2183c9e13aSJakub Kicinski #include <linux/netdevice.h> 2283c9e13aSJakub Kicinski #include <linux/u64_stats_sync.h> 238320d145SJiri Pirko #include <net/devlink.h> 24dc9c0753SJakub Kicinski #include <net/udp_tunnel.h> 2505296620SJakub Kicinski #include <net/xdp.h> 2683c9e13aSJakub Kicinski 2783c9e13aSJakub Kicinski #define DRV_NAME "netdevsim" 2883c9e13aSJakub Kicinski 2931d3ad83SJakub Kicinski #define NSIM_XDP_MAX_MTU 4000 3031d3ad83SJakub Kicinski 3131d3ad83SJakub Kicinski #define NSIM_EA(extack, msg) NL_SET_ERR_MSG_MOD((extack), msg) 3231d3ad83SJakub Kicinski 337699353dSShannon Nelson #define NSIM_IPSEC_MAX_SA_COUNT 33 347699353dSShannon Nelson #define NSIM_IPSEC_VALID BIT(31) 35424be63aSJakub Kicinski #define NSIM_UDP_TUNNEL_N_PORTS 4 367699353dSShannon Nelson 377699353dSShannon Nelson struct nsim_sa { 387699353dSShannon Nelson struct xfrm_state *xs; 397699353dSShannon Nelson __be32 ipaddr[4]; 407699353dSShannon Nelson u32 key[4]; 417699353dSShannon Nelson u32 salt; 427699353dSShannon Nelson bool used; 437699353dSShannon Nelson bool crypt; 447699353dSShannon Nelson bool rx; 457699353dSShannon Nelson }; 467699353dSShannon Nelson 477699353dSShannon Nelson struct nsim_ipsec { 487699353dSShannon Nelson struct nsim_sa sa[NSIM_IPSEC_MAX_SA_COUNT]; 497699353dSShannon Nelson struct dentry *pfile; 507699353dSShannon Nelson u32 count; 517699353dSShannon Nelson u32 tx; 527699353dSShannon Nelson u32 ok; 537699353dSShannon Nelson }; 547699353dSShannon Nelson 5577f9591bSAntonio Cardace struct nsim_ethtool_pauseparam { 56ff1f7c17SJakub Kicinski bool rx; 57ff1f7c17SJakub Kicinski bool tx; 58ff1f7c17SJakub Kicinski bool report_stats_rx; 59ff1f7c17SJakub Kicinski bool report_stats_tx; 60ff1f7c17SJakub Kicinski }; 61ff1f7c17SJakub Kicinski 6277f9591bSAntonio Cardace struct nsim_ethtool { 6377f9591bSAntonio Cardace struct nsim_ethtool_pauseparam pauseparam; 64*a7fc6db0SAntonio Cardace struct ethtool_coalesce coalesce; 65*a7fc6db0SAntonio Cardace struct ethtool_ringparam ring; 6677f9591bSAntonio Cardace }; 6777f9591bSAntonio Cardace 6883c9e13aSJakub Kicinski struct netdevsim { 6931d3ad83SJakub Kicinski struct net_device *netdev; 70a60f9e48SJiri Pirko struct nsim_dev *nsim_dev; 71e05b2d14SJiri Pirko struct nsim_dev_port *nsim_dev_port; 7231d3ad83SJakub Kicinski 7383c9e13aSJakub Kicinski u64 tx_packets; 7483c9e13aSJakub Kicinski u64 tx_bytes; 7583c9e13aSJakub Kicinski struct u64_stats_sync syncp; 7631d3ad83SJakub Kicinski 7740e4fe4cSJiri Pirko struct nsim_bus_dev *nsim_bus_dev; 7879579220SJakub Kicinski 7931d3ad83SJakub Kicinski struct bpf_prog *bpf_offloaded; 8031d3ad83SJakub Kicinski u32 bpf_offloaded_id; 8131d3ad83SJakub Kicinski 8205296620SJakub Kicinski struct xdp_attachment_info xdp; 83799e173dSJakub Kicinski struct xdp_attachment_info xdp_hw; 8431d3ad83SJakub Kicinski 8531d3ad83SJakub Kicinski bool bpf_tc_accept; 8631d3ad83SJakub Kicinski bool bpf_tc_non_bound_accept; 8731d3ad83SJakub Kicinski bool bpf_xdpdrv_accept; 8831d3ad83SJakub Kicinski bool bpf_xdpoffload_accept; 89395cacb5SJakub Kicinski 90395cacb5SJakub Kicinski bool bpf_map_accept; 917699353dSShannon Nelson struct nsim_ipsec ipsec; 92424be63aSJakub Kicinski struct { 93424be63aSJakub Kicinski u32 inject_error; 94424be63aSJakub Kicinski u32 sleep; 95dc9c0753SJakub Kicinski u32 __ports[2][NSIM_UDP_TUNNEL_N_PORTS]; 96dc9c0753SJakub Kicinski u32 (*ports)[NSIM_UDP_TUNNEL_N_PORTS]; 97424be63aSJakub Kicinski struct debugfs_u32_array dfs_ports[2]; 98424be63aSJakub Kicinski } udp_ports; 99ff1f7c17SJakub Kicinski 100ff1f7c17SJakub Kicinski struct nsim_ethtool ethtool; 10183c9e13aSJakub Kicinski }; 10231d3ad83SJakub Kicinski 103e05b2d14SJiri Pirko struct netdevsim * 104e05b2d14SJiri Pirko nsim_create(struct nsim_dev *nsim_dev, struct nsim_dev_port *nsim_dev_port); 105e05b2d14SJiri Pirko void nsim_destroy(struct netdevsim *ns); 106e05b2d14SJiri Pirko 107ff1f7c17SJakub Kicinski void nsim_ethtool_init(struct netdevsim *ns); 108ff1f7c17SJakub Kicinski 109424be63aSJakub Kicinski void nsim_udp_tunnels_debugfs_create(struct nsim_dev *nsim_dev); 110424be63aSJakub Kicinski int nsim_udp_tunnels_info_create(struct nsim_dev *nsim_dev, 111424be63aSJakub Kicinski struct net_device *dev); 112424be63aSJakub Kicinski void nsim_udp_tunnels_info_destroy(struct net_device *dev); 113424be63aSJakub Kicinski 1147c5db7e7SJakub Kicinski #ifdef CONFIG_BPF_SYSCALL 115d514f41eSJiri Pirko int nsim_bpf_dev_init(struct nsim_dev *nsim_dev); 116d514f41eSJiri Pirko void nsim_bpf_dev_exit(struct nsim_dev *nsim_dev); 11731d3ad83SJakub Kicinski int nsim_bpf_init(struct netdevsim *ns); 11831d3ad83SJakub Kicinski void nsim_bpf_uninit(struct netdevsim *ns); 11931d3ad83SJakub Kicinski int nsim_bpf(struct net_device *dev, struct netdev_bpf *bpf); 12031d3ad83SJakub Kicinski int nsim_bpf_disable_tc(struct netdevsim *ns); 12131d3ad83SJakub Kicinski int nsim_bpf_setup_tc_block_cb(enum tc_setup_type type, 12231d3ad83SJakub Kicinski void *type_data, void *cb_priv); 1237c5db7e7SJakub Kicinski #else 124d514f41eSJiri Pirko 125d514f41eSJiri Pirko static inline int nsim_bpf_dev_init(struct nsim_dev *nsim_dev) 126d514f41eSJiri Pirko { 127d514f41eSJiri Pirko return 0; 128d514f41eSJiri Pirko } 129d514f41eSJiri Pirko 130d514f41eSJiri Pirko static inline void nsim_bpf_dev_exit(struct nsim_dev *nsim_dev) 131d514f41eSJiri Pirko { 132d514f41eSJiri Pirko } 1337c5db7e7SJakub Kicinski static inline int nsim_bpf_init(struct netdevsim *ns) 1347c5db7e7SJakub Kicinski { 1357c5db7e7SJakub Kicinski return 0; 1367c5db7e7SJakub Kicinski } 1377c5db7e7SJakub Kicinski 1387c5db7e7SJakub Kicinski static inline void nsim_bpf_uninit(struct netdevsim *ns) 1397c5db7e7SJakub Kicinski { 1407c5db7e7SJakub Kicinski } 1417c5db7e7SJakub Kicinski 1427c5db7e7SJakub Kicinski static inline int nsim_bpf(struct net_device *dev, struct netdev_bpf *bpf) 1437c5db7e7SJakub Kicinski { 144e8407fdeSAndrii Nakryiko return -EOPNOTSUPP; 1457c5db7e7SJakub Kicinski } 1467c5db7e7SJakub Kicinski 1477c5db7e7SJakub Kicinski static inline int nsim_bpf_disable_tc(struct netdevsim *ns) 1487c5db7e7SJakub Kicinski { 1497c5db7e7SJakub Kicinski return 0; 1507c5db7e7SJakub Kicinski } 1517c5db7e7SJakub Kicinski 1527c5db7e7SJakub Kicinski static inline int 1537c5db7e7SJakub Kicinski nsim_bpf_setup_tc_block_cb(enum tc_setup_type type, void *type_data, 1547c5db7e7SJakub Kicinski void *cb_priv) 1557c5db7e7SJakub Kicinski { 1567c5db7e7SJakub Kicinski return -EOPNOTSUPP; 1577c5db7e7SJakub Kicinski } 1587c5db7e7SJakub Kicinski #endif 15979579220SJakub Kicinski 16037923ed6SDavid Ahern enum nsim_resource_id { 16137923ed6SDavid Ahern NSIM_RESOURCE_NONE, /* DEVLINK_RESOURCE_ID_PARENT_TOP */ 16237923ed6SDavid Ahern NSIM_RESOURCE_IPV4, 16337923ed6SDavid Ahern NSIM_RESOURCE_IPV4_FIB, 16437923ed6SDavid Ahern NSIM_RESOURCE_IPV4_FIB_RULES, 16537923ed6SDavid Ahern NSIM_RESOURCE_IPV6, 16637923ed6SDavid Ahern NSIM_RESOURCE_IPV6_FIB, 16737923ed6SDavid Ahern NSIM_RESOURCE_IPV6_FIB_RULES, 16835266255SIdo Schimmel NSIM_RESOURCE_NEXTHOPS, 16937923ed6SDavid Ahern }; 17037923ed6SDavid Ahern 17182c93a87SJiri Pirko struct nsim_dev_health { 17282c93a87SJiri Pirko struct devlink_health_reporter *empty_reporter; 17382c93a87SJiri Pirko struct devlink_health_reporter *dummy_reporter; 17482c93a87SJiri Pirko struct dentry *ddir; 17582c93a87SJiri Pirko char *recovered_break_msg; 17682c93a87SJiri Pirko u32 binary_len; 17782c93a87SJiri Pirko bool fail_recover; 17882c93a87SJiri Pirko }; 17982c93a87SJiri Pirko 18082c93a87SJiri Pirko int nsim_dev_health_init(struct nsim_dev *nsim_dev, struct devlink *devlink); 18182c93a87SJiri Pirko void nsim_dev_health_exit(struct nsim_dev *nsim_dev); 18282c93a87SJiri Pirko 1838320d145SJiri Pirko struct nsim_dev_port { 1848320d145SJiri Pirko struct list_head list; 1858320d145SJiri Pirko struct devlink_port devlink_port; 1868320d145SJiri Pirko unsigned int port_index; 1878320d145SJiri Pirko struct dentry *ddir; 188e05b2d14SJiri Pirko struct netdevsim *ns; 1898320d145SJiri Pirko }; 1908320d145SJiri Pirko 191a60f9e48SJiri Pirko struct nsim_dev { 192d514f41eSJiri Pirko struct nsim_bus_dev *nsim_bus_dev; 193a60f9e48SJiri Pirko struct nsim_fib_data *fib_data; 194da58f90fSIdo Schimmel struct nsim_trap_data *trap_data; 195d514f41eSJiri Pirko struct dentry *ddir; 196ab1d0cc0SJiri Pirko struct dentry *ports_ddir; 1978526ad96STaehee Yoo struct dentry *take_snapshot; 198d514f41eSJiri Pirko struct bpf_offload_dev *bpf_dev; 199d514f41eSJiri Pirko bool bpf_bind_accept; 200d514f41eSJiri Pirko u32 bpf_bind_verifier_delay; 201d514f41eSJiri Pirko struct dentry *ddir_bpf_bound_progs; 202d514f41eSJiri Pirko u32 prog_id_gen; 203d514f41eSJiri Pirko struct list_head bpf_bound_progs; 204d514f41eSJiri Pirko struct list_head bpf_bound_maps; 205514cf64cSJiri Pirko struct netdev_phys_item_id switch_id; 2068320d145SJiri Pirko struct list_head port_list; 207794b2c05SJiri Pirko struct mutex port_list_lock; /* protects port list */ 208fa4dfc4aSJiri Pirko bool fw_update_status; 209cbb58368SJacob Keller u32 fw_update_overwrite_mask; 210150e8f8aSJiri Pirko u32 max_macs; 211150e8f8aSJiri Pirko bool test1; 212155ddfc5SJiri Pirko bool dont_allow_reload; 213155ddfc5SJiri Pirko bool fail_reload; 2144418f862SJiri Pirko struct devlink_region *dummy_region; 21582c93a87SJiri Pirko struct nsim_dev_health health; 216d3cbb907SJiri Pirko struct flow_action_cookie *fa_cookie; 217d3cbb907SJiri Pirko spinlock_t fa_cookie_lock; /* protects fa_cookie */ 2180dc8249aSIdo Schimmel bool fail_trap_group_set; 219ad188458SIdo Schimmel bool fail_trap_policer_set; 220ad188458SIdo Schimmel bool fail_trap_policer_counter_get; 221424be63aSJakub Kicinski struct { 222dc9c0753SJakub Kicinski struct udp_tunnel_nic_shared utn_shared; 223dc9c0753SJakub Kicinski u32 __ports[2][NSIM_UDP_TUNNEL_N_PORTS]; 224424be63aSJakub Kicinski bool sync_all; 225424be63aSJakub Kicinski bool open_only; 226424be63aSJakub Kicinski bool ipv4_only; 227dc9c0753SJakub Kicinski bool shared; 228dda75173SJakub Kicinski bool static_iana_vxlan; 229424be63aSJakub Kicinski u32 sleep; 230424be63aSJakub Kicinski } udp_ports; 231a60f9e48SJiri Pirko }; 232a60f9e48SJiri Pirko 23390d29913SJiri Pirko static inline struct net *nsim_dev_net(struct nsim_dev *nsim_dev) 23490d29913SJiri Pirko { 23590d29913SJiri Pirko return devlink_net(priv_to_devlink(nsim_dev)); 23690d29913SJiri Pirko } 23790d29913SJiri Pirko 238d514f41eSJiri Pirko int nsim_dev_init(void); 239d514f41eSJiri Pirko void nsim_dev_exit(void); 2408320d145SJiri Pirko int nsim_dev_probe(struct nsim_bus_dev *nsim_bus_dev); 2418320d145SJiri Pirko void nsim_dev_remove(struct nsim_bus_dev *nsim_bus_dev); 242794b2c05SJiri Pirko int nsim_dev_port_add(struct nsim_bus_dev *nsim_bus_dev, 243794b2c05SJiri Pirko unsigned int port_index); 244794b2c05SJiri Pirko int nsim_dev_port_del(struct nsim_bus_dev *nsim_bus_dev, 245794b2c05SJiri Pirko unsigned int port_index); 24637923ed6SDavid Ahern 24775ba029fSJiri Pirko struct nsim_fib_data *nsim_fib_create(struct devlink *devlink, 24875ba029fSJiri Pirko struct netlink_ext_ack *extack); 24975ba029fSJiri Pirko void nsim_fib_destroy(struct devlink *devlink, struct nsim_fib_data *fib_data); 250a5facc4cSJiri Pirko u64 nsim_fib_get_val(struct nsim_fib_data *fib_data, 251a5facc4cSJiri Pirko enum nsim_resource_id res_id, bool max); 25237923ed6SDavid Ahern 2537699353dSShannon Nelson #if IS_ENABLED(CONFIG_XFRM_OFFLOAD) 2547699353dSShannon Nelson void nsim_ipsec_init(struct netdevsim *ns); 2557699353dSShannon Nelson void nsim_ipsec_teardown(struct netdevsim *ns); 2567699353dSShannon Nelson bool nsim_ipsec_tx(struct netdevsim *ns, struct sk_buff *skb); 2577699353dSShannon Nelson #else 2587699353dSShannon Nelson static inline void nsim_ipsec_init(struct netdevsim *ns) 2597699353dSShannon Nelson { 2607699353dSShannon Nelson } 2617699353dSShannon Nelson 2627699353dSShannon Nelson static inline void nsim_ipsec_teardown(struct netdevsim *ns) 2637699353dSShannon Nelson { 2647699353dSShannon Nelson } 2657699353dSShannon Nelson 2667699353dSShannon Nelson static inline bool nsim_ipsec_tx(struct netdevsim *ns, struct sk_buff *skb) 2677699353dSShannon Nelson { 2687699353dSShannon Nelson return true; 2697699353dSShannon Nelson } 2707699353dSShannon Nelson #endif 2717699353dSShannon Nelson 27240e4fe4cSJiri Pirko struct nsim_vf_config { 27340e4fe4cSJiri Pirko int link_state; 27440e4fe4cSJiri Pirko u16 min_tx_rate; 27540e4fe4cSJiri Pirko u16 max_tx_rate; 27640e4fe4cSJiri Pirko u16 vlan; 27740e4fe4cSJiri Pirko __be16 vlan_proto; 27840e4fe4cSJiri Pirko u16 qos; 27940e4fe4cSJiri Pirko u8 vf_mac[ETH_ALEN]; 28040e4fe4cSJiri Pirko bool spoofchk_enabled; 28140e4fe4cSJiri Pirko bool trusted; 28240e4fe4cSJiri Pirko bool rss_query_enabled; 28340e4fe4cSJiri Pirko }; 284925f5afeSJiri Pirko 28540e4fe4cSJiri Pirko struct nsim_bus_dev { 28640e4fe4cSJiri Pirko struct device dev; 287f9d9db47SJiri Pirko struct list_head list; 288f9d9db47SJiri Pirko unsigned int port_count; 2897b60027bSJiri Pirko struct net *initial_net; /* Purpose of this is to carry net pointer 2907b60027bSJiri Pirko * during the probe time only. 2917b60027bSJiri Pirko */ 29240e4fe4cSJiri Pirko unsigned int num_vfs; 29340e4fe4cSJiri Pirko struct nsim_vf_config *vfconfigs; 2946ab63366STaehee Yoo /* Lock for devlink->reload_enabled in netdevsim module */ 2956ab63366STaehee Yoo struct mutex nsim_bus_reload_lock; 296f5cd2160STaehee Yoo bool init; 29740e4fe4cSJiri Pirko }; 298925f5afeSJiri Pirko 299925f5afeSJiri Pirko int nsim_bus_init(void); 300925f5afeSJiri Pirko void nsim_bus_exit(void); 301