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 1631d3ad83SJakub Kicinski #include <linux/debugfs.h> 1783c9e13aSJakub Kicinski #include <linux/etherdevice.h> 18f394d07bSTaehee Yoo #include <linux/ethtool_netlink.h> 1983c9e13aSJakub Kicinski #include <linux/kernel.h> 2083c9e13aSJakub Kicinski #include <linux/module.h> 2183c9e13aSJakub Kicinski #include <linux/netdevice.h> 2283c9e13aSJakub Kicinski #include <linux/slab.h> 23f216306bSJakub Kicinski #include <net/netdev_queues.h> 246917d207SJakub Kicinski #include <net/netdev_rx_queue.h> 251580cbcbSJakub Kicinski #include <net/page_pool/helpers.h> 2683c9e13aSJakub Kicinski #include <net/netlink.h> 27b3ea4164SPaolo Abeni #include <net/net_shaper.h> 2831d3ad83SJakub Kicinski #include <net/pkt_cls.h> 2983c9e13aSJakub Kicinski #include <net/rtnetlink.h> 30424be63aSJakub Kicinski #include <net/udp_tunnel.h> 3183c9e13aSJakub Kicinski 3283c9e13aSJakub Kicinski #include "netdevsim.h" 3383c9e13aSJakub Kicinski 346917d207SJakub Kicinski MODULE_IMPORT_NS("NETDEV_INTERNAL"); 356917d207SJakub Kicinski 363762ec05SDavid Wei #define NSIM_RING_SIZE 256 373762ec05SDavid Wei 383762ec05SDavid Wei static int nsim_napi_rx(struct nsim_rq *rq, struct sk_buff *skb) 393762ec05SDavid Wei { 403762ec05SDavid Wei if (skb_queue_len(&rq->skb_queue) > NSIM_RING_SIZE) { 413762ec05SDavid Wei dev_kfree_skb_any(skb); 423762ec05SDavid Wei return NET_RX_DROP; 433762ec05SDavid Wei } 443762ec05SDavid Wei 453762ec05SDavid Wei skb_queue_tail(&rq->skb_queue, skb); 463762ec05SDavid Wei return NET_RX_SUCCESS; 473762ec05SDavid Wei } 483762ec05SDavid Wei 493762ec05SDavid Wei static int nsim_forward_skb(struct net_device *dev, struct sk_buff *skb, 503762ec05SDavid Wei struct nsim_rq *rq) 513762ec05SDavid Wei { 523762ec05SDavid Wei return __dev_forward_skb(dev, skb) ?: nsim_napi_rx(rq, skb); 533762ec05SDavid Wei } 543762ec05SDavid Wei 5583c9e13aSJakub Kicinski static netdev_tx_t nsim_start_xmit(struct sk_buff *skb, struct net_device *dev) 5683c9e13aSJakub Kicinski { 5783c9e13aSJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 583762ec05SDavid Wei struct net_device *peer_dev; 599eb95228SDavid Wei unsigned int len = skb->len; 609eb95228SDavid Wei struct netdevsim *peer_ns; 613c836451SJakub Kicinski struct netdev_config *cfg; 623762ec05SDavid Wei struct nsim_rq *rq; 633762ec05SDavid Wei int rxq; 6483c9e13aSJakub Kicinski 659eb95228SDavid Wei rcu_read_lock(); 667699353dSShannon Nelson if (!nsim_ipsec_tx(ns, skb)) 679eb95228SDavid Wei goto out_drop_free; 687699353dSShannon Nelson 699eb95228SDavid Wei peer_ns = rcu_dereference(ns->peer); 709eb95228SDavid Wei if (!peer_ns) 719eb95228SDavid Wei goto out_drop_free; 729eb95228SDavid Wei 733762ec05SDavid Wei peer_dev = peer_ns->netdev; 743762ec05SDavid Wei rxq = skb_get_queue_mapping(skb); 753762ec05SDavid Wei if (rxq >= peer_dev->num_rx_queues) 763762ec05SDavid Wei rxq = rxq % peer_dev->num_rx_queues; 77915c82f8SJakub Kicinski rq = peer_ns->rq[rxq]; 783762ec05SDavid Wei 793c836451SJakub Kicinski cfg = peer_dev->cfg; 80f394d07bSTaehee Yoo if (skb_is_nonlinear(skb) && 813c836451SJakub Kicinski (cfg->hds_config != ETHTOOL_TCP_DATA_SPLIT_ENABLED || 823c836451SJakub Kicinski (cfg->hds_config == ETHTOOL_TCP_DATA_SPLIT_ENABLED && 833c836451SJakub Kicinski cfg->hds_thresh > len))) 84f394d07bSTaehee Yoo skb_linearize(skb); 85f394d07bSTaehee Yoo 869eb95228SDavid Wei skb_tx_timestamp(skb); 873762ec05SDavid Wei if (unlikely(nsim_forward_skb(peer_dev, skb, rq) == NET_RX_DROP)) 889eb95228SDavid Wei goto out_drop_cnt; 899eb95228SDavid Wei 90bf3624cfSBreno Leitao if (!hrtimer_active(&rq->napi_timer)) 91bf3624cfSBreno Leitao hrtimer_start(&rq->napi_timer, us_to_ktime(5), HRTIMER_MODE_REL); 923762ec05SDavid Wei 939eb95228SDavid Wei rcu_read_unlock(); 9483c9e13aSJakub Kicinski u64_stats_update_begin(&ns->syncp); 9583c9e13aSJakub Kicinski ns->tx_packets++; 969eb95228SDavid Wei ns->tx_bytes += len; 9783c9e13aSJakub Kicinski u64_stats_update_end(&ns->syncp); 989eb95228SDavid Wei return NETDEV_TX_OK; 9983c9e13aSJakub Kicinski 1009eb95228SDavid Wei out_drop_free: 10183c9e13aSJakub Kicinski dev_kfree_skb(skb); 1029eb95228SDavid Wei out_drop_cnt: 1039eb95228SDavid Wei rcu_read_unlock(); 1049eb95228SDavid Wei u64_stats_update_begin(&ns->syncp); 1059eb95228SDavid Wei ns->tx_dropped++; 1069eb95228SDavid Wei u64_stats_update_end(&ns->syncp); 10783c9e13aSJakub Kicinski return NETDEV_TX_OK; 10883c9e13aSJakub Kicinski } 10983c9e13aSJakub Kicinski 11083c9e13aSJakub Kicinski static void nsim_set_rx_mode(struct net_device *dev) 11183c9e13aSJakub Kicinski { 11283c9e13aSJakub Kicinski } 11383c9e13aSJakub Kicinski 11431d3ad83SJakub Kicinski static int nsim_change_mtu(struct net_device *dev, int new_mtu) 11531d3ad83SJakub Kicinski { 11631d3ad83SJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 11731d3ad83SJakub Kicinski 118799e173dSJakub Kicinski if (ns->xdp.prog && new_mtu > NSIM_XDP_MAX_MTU) 11931d3ad83SJakub Kicinski return -EBUSY; 12031d3ad83SJakub Kicinski 1211eb2cdedSEric Dumazet WRITE_ONCE(dev->mtu, new_mtu); 12231d3ad83SJakub Kicinski 12331d3ad83SJakub Kicinski return 0; 12431d3ad83SJakub Kicinski } 12531d3ad83SJakub Kicinski 12683c9e13aSJakub Kicinski static void 12783c9e13aSJakub Kicinski nsim_get_stats64(struct net_device *dev, struct rtnl_link_stats64 *stats) 12883c9e13aSJakub Kicinski { 12983c9e13aSJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 13083c9e13aSJakub Kicinski unsigned int start; 13183c9e13aSJakub Kicinski 13283c9e13aSJakub Kicinski do { 133068c38adSThomas Gleixner start = u64_stats_fetch_begin(&ns->syncp); 13483c9e13aSJakub Kicinski stats->tx_bytes = ns->tx_bytes; 13583c9e13aSJakub Kicinski stats->tx_packets = ns->tx_packets; 1369eb95228SDavid Wei stats->tx_dropped = ns->tx_dropped; 137068c38adSThomas Gleixner } while (u64_stats_fetch_retry(&ns->syncp, start)); 13883c9e13aSJakub Kicinski } 13983c9e13aSJakub Kicinski 14031d3ad83SJakub Kicinski static int 14131d3ad83SJakub Kicinski nsim_setup_tc_block_cb(enum tc_setup_type type, void *type_data, void *cb_priv) 14231d3ad83SJakub Kicinski { 14331d3ad83SJakub Kicinski return nsim_bpf_setup_tc_block_cb(type, type_data, cb_priv); 14431d3ad83SJakub Kicinski } 14531d3ad83SJakub Kicinski 14679579220SJakub Kicinski static int nsim_set_vf_mac(struct net_device *dev, int vf, u8 *mac) 14779579220SJakub Kicinski { 14879579220SJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 1495e388f3dSJakub Kicinski struct nsim_dev *nsim_dev = ns->nsim_dev; 15079579220SJakub Kicinski 15179579220SJakub Kicinski /* Only refuse multicast addresses, zero address can mean unset/any. */ 1525e388f3dSJakub Kicinski if (vf >= nsim_dev_get_vfs(nsim_dev) || is_multicast_ether_addr(mac)) 15379579220SJakub Kicinski return -EINVAL; 1545e388f3dSJakub Kicinski memcpy(nsim_dev->vfconfigs[vf].vf_mac, mac, ETH_ALEN); 15579579220SJakub Kicinski 15679579220SJakub Kicinski return 0; 15779579220SJakub Kicinski } 15879579220SJakub Kicinski 15979579220SJakub Kicinski static int nsim_set_vf_vlan(struct net_device *dev, int vf, 16079579220SJakub Kicinski u16 vlan, u8 qos, __be16 vlan_proto) 16179579220SJakub Kicinski { 16279579220SJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 1635e388f3dSJakub Kicinski struct nsim_dev *nsim_dev = ns->nsim_dev; 16479579220SJakub Kicinski 1655e388f3dSJakub Kicinski if (vf >= nsim_dev_get_vfs(nsim_dev) || vlan > 4095 || qos > 7) 16679579220SJakub Kicinski return -EINVAL; 16779579220SJakub Kicinski 1685e388f3dSJakub Kicinski nsim_dev->vfconfigs[vf].vlan = vlan; 1695e388f3dSJakub Kicinski nsim_dev->vfconfigs[vf].qos = qos; 1705e388f3dSJakub Kicinski nsim_dev->vfconfigs[vf].vlan_proto = vlan_proto; 17179579220SJakub Kicinski 17279579220SJakub Kicinski return 0; 17379579220SJakub Kicinski } 17479579220SJakub Kicinski 17579579220SJakub Kicinski static int nsim_set_vf_rate(struct net_device *dev, int vf, int min, int max) 17679579220SJakub Kicinski { 17779579220SJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 1785e388f3dSJakub Kicinski struct nsim_dev *nsim_dev = ns->nsim_dev; 17979579220SJakub Kicinski 180160dc373SDmytro Linkin if (nsim_esw_mode_is_switchdev(ns->nsim_dev)) { 181160dc373SDmytro Linkin pr_err("Not supported in switchdev mode. Please use devlink API.\n"); 182160dc373SDmytro Linkin return -EOPNOTSUPP; 183160dc373SDmytro Linkin } 184160dc373SDmytro Linkin 1855e388f3dSJakub Kicinski if (vf >= nsim_dev_get_vfs(nsim_dev)) 18679579220SJakub Kicinski return -EINVAL; 18779579220SJakub Kicinski 1885e388f3dSJakub Kicinski nsim_dev->vfconfigs[vf].min_tx_rate = min; 1895e388f3dSJakub Kicinski nsim_dev->vfconfigs[vf].max_tx_rate = max; 19079579220SJakub Kicinski 19179579220SJakub Kicinski return 0; 19279579220SJakub Kicinski } 19379579220SJakub Kicinski 19479579220SJakub Kicinski static int nsim_set_vf_spoofchk(struct net_device *dev, int vf, bool val) 19579579220SJakub Kicinski { 19679579220SJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 1975e388f3dSJakub Kicinski struct nsim_dev *nsim_dev = ns->nsim_dev; 19879579220SJakub Kicinski 1995e388f3dSJakub Kicinski if (vf >= nsim_dev_get_vfs(nsim_dev)) 20079579220SJakub Kicinski return -EINVAL; 2015e388f3dSJakub Kicinski nsim_dev->vfconfigs[vf].spoofchk_enabled = val; 20279579220SJakub Kicinski 20379579220SJakub Kicinski return 0; 20479579220SJakub Kicinski } 20579579220SJakub Kicinski 20679579220SJakub Kicinski static int nsim_set_vf_rss_query_en(struct net_device *dev, int vf, bool val) 20779579220SJakub Kicinski { 20879579220SJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 2095e388f3dSJakub Kicinski struct nsim_dev *nsim_dev = ns->nsim_dev; 21079579220SJakub Kicinski 2115e388f3dSJakub Kicinski if (vf >= nsim_dev_get_vfs(nsim_dev)) 21279579220SJakub Kicinski return -EINVAL; 2135e388f3dSJakub Kicinski nsim_dev->vfconfigs[vf].rss_query_enabled = val; 21479579220SJakub Kicinski 21579579220SJakub Kicinski return 0; 21679579220SJakub Kicinski } 21779579220SJakub Kicinski 21879579220SJakub Kicinski static int nsim_set_vf_trust(struct net_device *dev, int vf, bool val) 21979579220SJakub Kicinski { 22079579220SJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 2215e388f3dSJakub Kicinski struct nsim_dev *nsim_dev = ns->nsim_dev; 22279579220SJakub Kicinski 2235e388f3dSJakub Kicinski if (vf >= nsim_dev_get_vfs(nsim_dev)) 22479579220SJakub Kicinski return -EINVAL; 2255e388f3dSJakub Kicinski nsim_dev->vfconfigs[vf].trusted = val; 22679579220SJakub Kicinski 22779579220SJakub Kicinski return 0; 22879579220SJakub Kicinski } 22979579220SJakub Kicinski 23079579220SJakub Kicinski static int 23179579220SJakub Kicinski nsim_get_vf_config(struct net_device *dev, int vf, struct ifla_vf_info *ivi) 23279579220SJakub Kicinski { 23379579220SJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 2345e388f3dSJakub Kicinski struct nsim_dev *nsim_dev = ns->nsim_dev; 23579579220SJakub Kicinski 2365e388f3dSJakub Kicinski if (vf >= nsim_dev_get_vfs(nsim_dev)) 23779579220SJakub Kicinski return -EINVAL; 23879579220SJakub Kicinski 23979579220SJakub Kicinski ivi->vf = vf; 2405e388f3dSJakub Kicinski ivi->linkstate = nsim_dev->vfconfigs[vf].link_state; 2415e388f3dSJakub Kicinski ivi->min_tx_rate = nsim_dev->vfconfigs[vf].min_tx_rate; 2425e388f3dSJakub Kicinski ivi->max_tx_rate = nsim_dev->vfconfigs[vf].max_tx_rate; 2435e388f3dSJakub Kicinski ivi->vlan = nsim_dev->vfconfigs[vf].vlan; 2445e388f3dSJakub Kicinski ivi->vlan_proto = nsim_dev->vfconfigs[vf].vlan_proto; 2455e388f3dSJakub Kicinski ivi->qos = nsim_dev->vfconfigs[vf].qos; 2465e388f3dSJakub Kicinski memcpy(&ivi->mac, nsim_dev->vfconfigs[vf].vf_mac, ETH_ALEN); 2475e388f3dSJakub Kicinski ivi->spoofchk = nsim_dev->vfconfigs[vf].spoofchk_enabled; 2485e388f3dSJakub Kicinski ivi->trusted = nsim_dev->vfconfigs[vf].trusted; 2495e388f3dSJakub Kicinski ivi->rss_query_en = nsim_dev->vfconfigs[vf].rss_query_enabled; 25079579220SJakub Kicinski 25179579220SJakub Kicinski return 0; 25279579220SJakub Kicinski } 25379579220SJakub Kicinski 25479579220SJakub Kicinski static int nsim_set_vf_link_state(struct net_device *dev, int vf, int state) 25579579220SJakub Kicinski { 25679579220SJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 2575e388f3dSJakub Kicinski struct nsim_dev *nsim_dev = ns->nsim_dev; 25879579220SJakub Kicinski 2595e388f3dSJakub Kicinski if (vf >= nsim_dev_get_vfs(nsim_dev)) 26079579220SJakub Kicinski return -EINVAL; 26179579220SJakub Kicinski 26279579220SJakub Kicinski switch (state) { 26379579220SJakub Kicinski case IFLA_VF_LINK_STATE_AUTO: 26479579220SJakub Kicinski case IFLA_VF_LINK_STATE_ENABLE: 26579579220SJakub Kicinski case IFLA_VF_LINK_STATE_DISABLE: 26679579220SJakub Kicinski break; 26779579220SJakub Kicinski default: 26879579220SJakub Kicinski return -EINVAL; 26979579220SJakub Kicinski } 27079579220SJakub Kicinski 2715e388f3dSJakub Kicinski nsim_dev->vfconfigs[vf].link_state = state; 27279579220SJakub Kicinski 27379579220SJakub Kicinski return 0; 27479579220SJakub Kicinski } 27579579220SJakub Kicinski 27635da47feSVladimir Oltean static void nsim_taprio_stats(struct tc_taprio_qopt_stats *stats) 27735da47feSVladimir Oltean { 27835da47feSVladimir Oltean stats->window_drops = 0; 27935da47feSVladimir Oltean stats->tx_overruns = 0; 28035da47feSVladimir Oltean } 28135da47feSVladimir Oltean 28235da47feSVladimir Oltean static int nsim_setup_tc_taprio(struct net_device *dev, 28335da47feSVladimir Oltean struct tc_taprio_qopt_offload *offload) 28435da47feSVladimir Oltean { 28535da47feSVladimir Oltean int err = 0; 28635da47feSVladimir Oltean 28735da47feSVladimir Oltean switch (offload->cmd) { 28835da47feSVladimir Oltean case TAPRIO_CMD_REPLACE: 28935da47feSVladimir Oltean case TAPRIO_CMD_DESTROY: 29035da47feSVladimir Oltean break; 29135da47feSVladimir Oltean case TAPRIO_CMD_STATS: 29235da47feSVladimir Oltean nsim_taprio_stats(&offload->stats); 29335da47feSVladimir Oltean break; 29435da47feSVladimir Oltean default: 29535da47feSVladimir Oltean err = -EOPNOTSUPP; 29635da47feSVladimir Oltean } 29735da47feSVladimir Oltean 29835da47feSVladimir Oltean return err; 29935da47feSVladimir Oltean } 30035da47feSVladimir Oltean 301955bcb6eSPablo Neira Ayuso static LIST_HEAD(nsim_block_cb_list); 302955bcb6eSPablo Neira Ayuso 30331d3ad83SJakub Kicinski static int 30431d3ad83SJakub Kicinski nsim_setup_tc(struct net_device *dev, enum tc_setup_type type, void *type_data) 30531d3ad83SJakub Kicinski { 3064e95bc26SPablo Neira Ayuso struct netdevsim *ns = netdev_priv(dev); 3074e95bc26SPablo Neira Ayuso 30831d3ad83SJakub Kicinski switch (type) { 30935da47feSVladimir Oltean case TC_SETUP_QDISC_TAPRIO: 31035da47feSVladimir Oltean return nsim_setup_tc_taprio(dev, type_data); 31131d3ad83SJakub Kicinski case TC_SETUP_BLOCK: 312955bcb6eSPablo Neira Ayuso return flow_block_cb_setup_simple(type_data, 313955bcb6eSPablo Neira Ayuso &nsim_block_cb_list, 3144e95bc26SPablo Neira Ayuso nsim_setup_tc_block_cb, 3154e95bc26SPablo Neira Ayuso ns, ns, true); 31631d3ad83SJakub Kicinski default: 31731d3ad83SJakub Kicinski return -EOPNOTSUPP; 31831d3ad83SJakub Kicinski } 31931d3ad83SJakub Kicinski } 32031d3ad83SJakub Kicinski 32131d3ad83SJakub Kicinski static int 32231d3ad83SJakub Kicinski nsim_set_features(struct net_device *dev, netdev_features_t features) 32331d3ad83SJakub Kicinski { 32431d3ad83SJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 32531d3ad83SJakub Kicinski 32631d3ad83SJakub Kicinski if ((dev->features & NETIF_F_HW_TC) > (features & NETIF_F_HW_TC)) 32731d3ad83SJakub Kicinski return nsim_bpf_disable_tc(ns); 32831d3ad83SJakub Kicinski 32931d3ad83SJakub Kicinski return 0; 33031d3ad83SJakub Kicinski } 33131d3ad83SJakub Kicinski 3328debcf58SDavid Wei static int nsim_get_iflink(const struct net_device *dev) 3338debcf58SDavid Wei { 3348debcf58SDavid Wei struct netdevsim *nsim, *peer; 3358debcf58SDavid Wei int iflink; 3368debcf58SDavid Wei 3378debcf58SDavid Wei nsim = netdev_priv(dev); 3388debcf58SDavid Wei 3398debcf58SDavid Wei rcu_read_lock(); 3408debcf58SDavid Wei peer = rcu_dereference(nsim->peer); 3415add2f72SDavid Wei iflink = peer ? READ_ONCE(peer->netdev->ifindex) : 3425add2f72SDavid Wei READ_ONCE(dev->ifindex); 3438debcf58SDavid Wei rcu_read_unlock(); 3448debcf58SDavid Wei 3458debcf58SDavid Wei return iflink; 3468debcf58SDavid Wei } 3478debcf58SDavid Wei 3483762ec05SDavid Wei static int nsim_rcv(struct nsim_rq *rq, int budget) 3493762ec05SDavid Wei { 3503762ec05SDavid Wei struct sk_buff *skb; 3513762ec05SDavid Wei int i; 3523762ec05SDavid Wei 3533762ec05SDavid Wei for (i = 0; i < budget; i++) { 3543762ec05SDavid Wei if (skb_queue_empty(&rq->skb_queue)) 3553762ec05SDavid Wei break; 3563762ec05SDavid Wei 3573762ec05SDavid Wei skb = skb_dequeue(&rq->skb_queue); 3583762ec05SDavid Wei netif_receive_skb(skb); 3593762ec05SDavid Wei } 3603762ec05SDavid Wei 3613762ec05SDavid Wei return i; 3623762ec05SDavid Wei } 3633762ec05SDavid Wei 3643762ec05SDavid Wei static int nsim_poll(struct napi_struct *napi, int budget) 3653762ec05SDavid Wei { 3663762ec05SDavid Wei struct nsim_rq *rq = container_of(napi, struct nsim_rq, napi); 3673762ec05SDavid Wei int done; 3683762ec05SDavid Wei 3693762ec05SDavid Wei done = nsim_rcv(rq, budget); 3703762ec05SDavid Wei napi_complete(napi); 3713762ec05SDavid Wei 3723762ec05SDavid Wei return done; 3733762ec05SDavid Wei } 3743762ec05SDavid Wei 3755bc8e8dbSJakub Kicinski static int nsim_create_page_pool(struct page_pool **p, struct napi_struct *napi) 3763762ec05SDavid Wei { 3775bc8e8dbSJakub Kicinski struct page_pool_params params = { 3783762ec05SDavid Wei .order = 0, 3793762ec05SDavid Wei .pool_size = NSIM_RING_SIZE, 3803762ec05SDavid Wei .nid = NUMA_NO_NODE, 3815bc8e8dbSJakub Kicinski .dev = &napi->dev->dev, 3825bc8e8dbSJakub Kicinski .napi = napi, 3833762ec05SDavid Wei .dma_dir = DMA_BIDIRECTIONAL, 3845bc8e8dbSJakub Kicinski .netdev = napi->dev, 3853762ec05SDavid Wei }; 3865bc8e8dbSJakub Kicinski struct page_pool *pool; 3873762ec05SDavid Wei 3885bc8e8dbSJakub Kicinski pool = page_pool_create(¶ms); 3895bc8e8dbSJakub Kicinski if (IS_ERR(pool)) 3905bc8e8dbSJakub Kicinski return PTR_ERR(pool); 3913762ec05SDavid Wei 3925bc8e8dbSJakub Kicinski *p = pool; 3933762ec05SDavid Wei return 0; 3943762ec05SDavid Wei } 3953762ec05SDavid Wei 3963762ec05SDavid Wei static int nsim_init_napi(struct netdevsim *ns) 3973762ec05SDavid Wei { 3983762ec05SDavid Wei struct net_device *dev = ns->netdev; 3993762ec05SDavid Wei struct nsim_rq *rq; 4003762ec05SDavid Wei int err, i; 4013762ec05SDavid Wei 4023762ec05SDavid Wei for (i = 0; i < dev->num_rx_queues; i++) { 403915c82f8SJakub Kicinski rq = ns->rq[i]; 4043762ec05SDavid Wei 405d4c22ec6SStanislav Fomichev netif_napi_add_config_locked(dev, &rq->napi, nsim_poll, i); 4063762ec05SDavid Wei } 4073762ec05SDavid Wei 4083762ec05SDavid Wei for (i = 0; i < dev->num_rx_queues; i++) { 409915c82f8SJakub Kicinski rq = ns->rq[i]; 4103762ec05SDavid Wei 4115bc8e8dbSJakub Kicinski err = nsim_create_page_pool(&rq->page_pool, &rq->napi); 4123762ec05SDavid Wei if (err) 4133762ec05SDavid Wei goto err_pp_destroy; 4143762ec05SDavid Wei } 4153762ec05SDavid Wei 4163762ec05SDavid Wei return 0; 4173762ec05SDavid Wei 4183762ec05SDavid Wei err_pp_destroy: 4193762ec05SDavid Wei while (i--) { 420915c82f8SJakub Kicinski page_pool_destroy(ns->rq[i]->page_pool); 421915c82f8SJakub Kicinski ns->rq[i]->page_pool = NULL; 4223762ec05SDavid Wei } 4233762ec05SDavid Wei 4243762ec05SDavid Wei for (i = 0; i < dev->num_rx_queues; i++) 425d4c22ec6SStanislav Fomichev __netif_napi_del_locked(&ns->rq[i]->napi); 4263762ec05SDavid Wei 4273762ec05SDavid Wei return err; 4283762ec05SDavid Wei } 4293762ec05SDavid Wei 430bf3624cfSBreno Leitao static enum hrtimer_restart nsim_napi_schedule(struct hrtimer *timer) 431bf3624cfSBreno Leitao { 432bf3624cfSBreno Leitao struct nsim_rq *rq; 433bf3624cfSBreno Leitao 434bf3624cfSBreno Leitao rq = container_of(timer, struct nsim_rq, napi_timer); 435bf3624cfSBreno Leitao napi_schedule(&rq->napi); 436bf3624cfSBreno Leitao 437bf3624cfSBreno Leitao return HRTIMER_NORESTART; 438bf3624cfSBreno Leitao } 439bf3624cfSBreno Leitao 440bf3624cfSBreno Leitao static void nsim_rq_timer_init(struct nsim_rq *rq) 441bf3624cfSBreno Leitao { 442bf3624cfSBreno Leitao hrtimer_init(&rq->napi_timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL); 443bf3624cfSBreno Leitao rq->napi_timer.function = nsim_napi_schedule; 444bf3624cfSBreno Leitao } 445bf3624cfSBreno Leitao 4463762ec05SDavid Wei static void nsim_enable_napi(struct netdevsim *ns) 4473762ec05SDavid Wei { 4483762ec05SDavid Wei struct net_device *dev = ns->netdev; 4493762ec05SDavid Wei int i; 4503762ec05SDavid Wei 4513762ec05SDavid Wei for (i = 0; i < dev->num_rx_queues; i++) { 452915c82f8SJakub Kicinski struct nsim_rq *rq = ns->rq[i]; 4533762ec05SDavid Wei 4543762ec05SDavid Wei netif_queue_set_napi(dev, i, NETDEV_QUEUE_TYPE_RX, &rq->napi); 455d4c22ec6SStanislav Fomichev napi_enable_locked(&rq->napi); 4563762ec05SDavid Wei } 4573762ec05SDavid Wei } 4583762ec05SDavid Wei 4591580cbcbSJakub Kicinski static int nsim_open(struct net_device *dev) 4601580cbcbSJakub Kicinski { 4611580cbcbSJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 4623762ec05SDavid Wei int err; 4631580cbcbSJakub Kicinski 464d4c22ec6SStanislav Fomichev netdev_assert_locked(dev); 465d4c22ec6SStanislav Fomichev 4663762ec05SDavid Wei err = nsim_init_napi(ns); 4673762ec05SDavid Wei if (err) 4683762ec05SDavid Wei return err; 4691580cbcbSJakub Kicinski 4703762ec05SDavid Wei nsim_enable_napi(ns); 4713762ec05SDavid Wei 4723762ec05SDavid Wei return 0; 4733762ec05SDavid Wei } 4743762ec05SDavid Wei 4753762ec05SDavid Wei static void nsim_del_napi(struct netdevsim *ns) 4763762ec05SDavid Wei { 4773762ec05SDavid Wei struct net_device *dev = ns->netdev; 4783762ec05SDavid Wei int i; 4793762ec05SDavid Wei 4803762ec05SDavid Wei for (i = 0; i < dev->num_rx_queues; i++) { 481915c82f8SJakub Kicinski struct nsim_rq *rq = ns->rq[i]; 4823762ec05SDavid Wei 483d4c22ec6SStanislav Fomichev napi_disable_locked(&rq->napi); 484d4c22ec6SStanislav Fomichev __netif_napi_del_locked(&rq->napi); 4853762ec05SDavid Wei } 4863762ec05SDavid Wei synchronize_net(); 4873762ec05SDavid Wei 4883762ec05SDavid Wei for (i = 0; i < dev->num_rx_queues; i++) { 489915c82f8SJakub Kicinski page_pool_destroy(ns->rq[i]->page_pool); 490915c82f8SJakub Kicinski ns->rq[i]->page_pool = NULL; 4913762ec05SDavid Wei } 4921580cbcbSJakub Kicinski } 4931580cbcbSJakub Kicinski 4941580cbcbSJakub Kicinski static int nsim_stop(struct net_device *dev) 4951580cbcbSJakub Kicinski { 4961580cbcbSJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 4973762ec05SDavid Wei struct netdevsim *peer; 4981580cbcbSJakub Kicinski 499d4c22ec6SStanislav Fomichev netdev_assert_locked(dev); 500d4c22ec6SStanislav Fomichev 5013762ec05SDavid Wei netif_carrier_off(dev); 5023762ec05SDavid Wei peer = rtnl_dereference(ns->peer); 5033762ec05SDavid Wei if (peer) 5043762ec05SDavid Wei netif_carrier_off(peer->netdev); 5053762ec05SDavid Wei 5063762ec05SDavid Wei nsim_del_napi(ns); 5071580cbcbSJakub Kicinski 5081580cbcbSJakub Kicinski return 0; 5091580cbcbSJakub Kicinski } 5101580cbcbSJakub Kicinski 511b3ea4164SPaolo Abeni static int nsim_shaper_set(struct net_shaper_binding *binding, 512b3ea4164SPaolo Abeni const struct net_shaper *shaper, 513b3ea4164SPaolo Abeni struct netlink_ext_ack *extack) 514b3ea4164SPaolo Abeni { 515b3ea4164SPaolo Abeni return 0; 516b3ea4164SPaolo Abeni } 517b3ea4164SPaolo Abeni 518b3ea4164SPaolo Abeni static int nsim_shaper_del(struct net_shaper_binding *binding, 519b3ea4164SPaolo Abeni const struct net_shaper_handle *handle, 520b3ea4164SPaolo Abeni struct netlink_ext_ack *extack) 521b3ea4164SPaolo Abeni { 522b3ea4164SPaolo Abeni return 0; 523b3ea4164SPaolo Abeni } 524b3ea4164SPaolo Abeni 525b3ea4164SPaolo Abeni static int nsim_shaper_group(struct net_shaper_binding *binding, 526b3ea4164SPaolo Abeni int leaves_count, 527b3ea4164SPaolo Abeni const struct net_shaper *leaves, 528b3ea4164SPaolo Abeni const struct net_shaper *root, 529b3ea4164SPaolo Abeni struct netlink_ext_ack *extack) 530b3ea4164SPaolo Abeni { 531b3ea4164SPaolo Abeni return 0; 532b3ea4164SPaolo Abeni } 533b3ea4164SPaolo Abeni 534b3ea4164SPaolo Abeni static void nsim_shaper_cap(struct net_shaper_binding *binding, 535b3ea4164SPaolo Abeni enum net_shaper_scope scope, 536b3ea4164SPaolo Abeni unsigned long *flags) 537b3ea4164SPaolo Abeni { 538b3ea4164SPaolo Abeni *flags = ULONG_MAX; 539b3ea4164SPaolo Abeni } 540b3ea4164SPaolo Abeni 541b3ea4164SPaolo Abeni static const struct net_shaper_ops nsim_shaper_ops = { 542b3ea4164SPaolo Abeni .set = nsim_shaper_set, 543b3ea4164SPaolo Abeni .delete = nsim_shaper_del, 544b3ea4164SPaolo Abeni .group = nsim_shaper_group, 545b3ea4164SPaolo Abeni .capabilities = nsim_shaper_cap, 546b3ea4164SPaolo Abeni }; 547b3ea4164SPaolo Abeni 54883c9e13aSJakub Kicinski static const struct net_device_ops nsim_netdev_ops = { 54983c9e13aSJakub Kicinski .ndo_start_xmit = nsim_start_xmit, 55083c9e13aSJakub Kicinski .ndo_set_rx_mode = nsim_set_rx_mode, 55183c9e13aSJakub Kicinski .ndo_set_mac_address = eth_mac_addr, 55283c9e13aSJakub Kicinski .ndo_validate_addr = eth_validate_addr, 55331d3ad83SJakub Kicinski .ndo_change_mtu = nsim_change_mtu, 55483c9e13aSJakub Kicinski .ndo_get_stats64 = nsim_get_stats64, 55579579220SJakub Kicinski .ndo_set_vf_mac = nsim_set_vf_mac, 55679579220SJakub Kicinski .ndo_set_vf_vlan = nsim_set_vf_vlan, 55779579220SJakub Kicinski .ndo_set_vf_rate = nsim_set_vf_rate, 55879579220SJakub Kicinski .ndo_set_vf_spoofchk = nsim_set_vf_spoofchk, 55979579220SJakub Kicinski .ndo_set_vf_trust = nsim_set_vf_trust, 56079579220SJakub Kicinski .ndo_get_vf_config = nsim_get_vf_config, 56179579220SJakub Kicinski .ndo_set_vf_link_state = nsim_set_vf_link_state, 56279579220SJakub Kicinski .ndo_set_vf_rss_query_en = nsim_set_vf_rss_query_en, 56331d3ad83SJakub Kicinski .ndo_setup_tc = nsim_setup_tc, 56431d3ad83SJakub Kicinski .ndo_set_features = nsim_set_features, 5658debcf58SDavid Wei .ndo_get_iflink = nsim_get_iflink, 56631d3ad83SJakub Kicinski .ndo_bpf = nsim_bpf, 5671580cbcbSJakub Kicinski .ndo_open = nsim_open, 5681580cbcbSJakub Kicinski .ndo_stop = nsim_stop, 569b3ea4164SPaolo Abeni .net_shaper_ops = &nsim_shaper_ops, 57083c9e13aSJakub Kicinski }; 57183c9e13aSJakub Kicinski 57292ba1f29SDmytro Linkin static const struct net_device_ops nsim_vf_netdev_ops = { 57392ba1f29SDmytro Linkin .ndo_start_xmit = nsim_start_xmit, 57492ba1f29SDmytro Linkin .ndo_set_rx_mode = nsim_set_rx_mode, 57592ba1f29SDmytro Linkin .ndo_set_mac_address = eth_mac_addr, 57692ba1f29SDmytro Linkin .ndo_validate_addr = eth_validate_addr, 57792ba1f29SDmytro Linkin .ndo_change_mtu = nsim_change_mtu, 57892ba1f29SDmytro Linkin .ndo_get_stats64 = nsim_get_stats64, 57992ba1f29SDmytro Linkin .ndo_setup_tc = nsim_setup_tc, 58092ba1f29SDmytro Linkin .ndo_set_features = nsim_set_features, 58192ba1f29SDmytro Linkin }; 58292ba1f29SDmytro Linkin 583f216306bSJakub Kicinski /* We don't have true per-queue stats, yet, so do some random fakery here. 584f216306bSJakub Kicinski * Only report stuff for queue 0. 585f216306bSJakub Kicinski */ 586f216306bSJakub Kicinski static void nsim_get_queue_stats_rx(struct net_device *dev, int idx, 587f216306bSJakub Kicinski struct netdev_queue_stats_rx *stats) 588f216306bSJakub Kicinski { 589f216306bSJakub Kicinski struct rtnl_link_stats64 rtstats = {}; 590f216306bSJakub Kicinski 591f216306bSJakub Kicinski if (!idx) 592f216306bSJakub Kicinski nsim_get_stats64(dev, &rtstats); 593f216306bSJakub Kicinski 594f216306bSJakub Kicinski stats->packets = rtstats.rx_packets - !!rtstats.rx_packets; 595f216306bSJakub Kicinski stats->bytes = rtstats.rx_bytes; 596f216306bSJakub Kicinski } 597f216306bSJakub Kicinski 598f216306bSJakub Kicinski static void nsim_get_queue_stats_tx(struct net_device *dev, int idx, 599f216306bSJakub Kicinski struct netdev_queue_stats_tx *stats) 600f216306bSJakub Kicinski { 601f216306bSJakub Kicinski struct rtnl_link_stats64 rtstats = {}; 602f216306bSJakub Kicinski 603f216306bSJakub Kicinski if (!idx) 604f216306bSJakub Kicinski nsim_get_stats64(dev, &rtstats); 605f216306bSJakub Kicinski 606f216306bSJakub Kicinski stats->packets = rtstats.tx_packets - !!rtstats.tx_packets; 607f216306bSJakub Kicinski stats->bytes = rtstats.tx_bytes; 608f216306bSJakub Kicinski } 609f216306bSJakub Kicinski 610f216306bSJakub Kicinski static void nsim_get_base_stats(struct net_device *dev, 611f216306bSJakub Kicinski struct netdev_queue_stats_rx *rx, 612f216306bSJakub Kicinski struct netdev_queue_stats_tx *tx) 613f216306bSJakub Kicinski { 614f216306bSJakub Kicinski struct rtnl_link_stats64 rtstats = {}; 615f216306bSJakub Kicinski 616f216306bSJakub Kicinski nsim_get_stats64(dev, &rtstats); 617f216306bSJakub Kicinski 618f216306bSJakub Kicinski rx->packets = !!rtstats.rx_packets; 619f216306bSJakub Kicinski rx->bytes = 0; 620f216306bSJakub Kicinski tx->packets = !!rtstats.tx_packets; 621f216306bSJakub Kicinski tx->bytes = 0; 622f216306bSJakub Kicinski } 623f216306bSJakub Kicinski 624f216306bSJakub Kicinski static const struct netdev_stat_ops nsim_stat_ops = { 625f216306bSJakub Kicinski .get_queue_stats_tx = nsim_get_queue_stats_tx, 626f216306bSJakub Kicinski .get_queue_stats_rx = nsim_get_queue_stats_rx, 627f216306bSJakub Kicinski .get_base_stats = nsim_get_base_stats, 628f216306bSJakub Kicinski }; 629f216306bSJakub Kicinski 630a565dd04SJakub Kicinski static struct nsim_rq *nsim_queue_alloc(void) 631a565dd04SJakub Kicinski { 632a565dd04SJakub Kicinski struct nsim_rq *rq; 633a565dd04SJakub Kicinski 634a565dd04SJakub Kicinski rq = kzalloc(sizeof(*rq), GFP_KERNEL_ACCOUNT); 635a565dd04SJakub Kicinski if (!rq) 636a565dd04SJakub Kicinski return NULL; 637a565dd04SJakub Kicinski 638a565dd04SJakub Kicinski skb_queue_head_init(&rq->skb_queue); 639bf3624cfSBreno Leitao nsim_rq_timer_init(rq); 640a565dd04SJakub Kicinski return rq; 641a565dd04SJakub Kicinski } 642a565dd04SJakub Kicinski 643a565dd04SJakub Kicinski static void nsim_queue_free(struct nsim_rq *rq) 644a565dd04SJakub Kicinski { 645bf3624cfSBreno Leitao hrtimer_cancel(&rq->napi_timer); 646a565dd04SJakub Kicinski skb_queue_purge_reason(&rq->skb_queue, SKB_DROP_REASON_QUEUE_PURGE); 647a565dd04SJakub Kicinski kfree(rq); 648a565dd04SJakub Kicinski } 649a565dd04SJakub Kicinski 6505bc8e8dbSJakub Kicinski /* Queue reset mode is controlled by ns->rq_reset_mode. 6515bc8e8dbSJakub Kicinski * - normal - new NAPI new pool (old NAPI enabled when new added) 6525bc8e8dbSJakub Kicinski * - mode 1 - allocate new pool (NAPI is only disabled / enabled) 6535bc8e8dbSJakub Kicinski * - mode 2 - new NAPI new pool (old NAPI removed before new added) 6545bc8e8dbSJakub Kicinski * - mode 3 - new NAPI new pool (old NAPI disabled when new added) 6555bc8e8dbSJakub Kicinski */ 6565bc8e8dbSJakub Kicinski struct nsim_queue_mem { 6575bc8e8dbSJakub Kicinski struct nsim_rq *rq; 6585bc8e8dbSJakub Kicinski struct page_pool *pp; 6595bc8e8dbSJakub Kicinski }; 6605bc8e8dbSJakub Kicinski 6615bc8e8dbSJakub Kicinski static int 6625bc8e8dbSJakub Kicinski nsim_queue_mem_alloc(struct net_device *dev, void *per_queue_mem, int idx) 6635bc8e8dbSJakub Kicinski { 6645bc8e8dbSJakub Kicinski struct nsim_queue_mem *qmem = per_queue_mem; 6655bc8e8dbSJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 6665bc8e8dbSJakub Kicinski int err; 6675bc8e8dbSJakub Kicinski 6685bc8e8dbSJakub Kicinski if (ns->rq_reset_mode > 3) 6695bc8e8dbSJakub Kicinski return -EINVAL; 6705bc8e8dbSJakub Kicinski 671285b3f78SJakub Kicinski if (ns->rq_reset_mode == 1) { 672285b3f78SJakub Kicinski if (!netif_running(ns->netdev)) 673285b3f78SJakub Kicinski return -ENETDOWN; 6745bc8e8dbSJakub Kicinski return nsim_create_page_pool(&qmem->pp, &ns->rq[idx]->napi); 675285b3f78SJakub Kicinski } 6765bc8e8dbSJakub Kicinski 6775bc8e8dbSJakub Kicinski qmem->rq = nsim_queue_alloc(); 6785bc8e8dbSJakub Kicinski if (!qmem->rq) 6795bc8e8dbSJakub Kicinski return -ENOMEM; 6805bc8e8dbSJakub Kicinski 6815bc8e8dbSJakub Kicinski err = nsim_create_page_pool(&qmem->rq->page_pool, &qmem->rq->napi); 6825bc8e8dbSJakub Kicinski if (err) 6835bc8e8dbSJakub Kicinski goto err_free; 6845bc8e8dbSJakub Kicinski 6855bc8e8dbSJakub Kicinski if (!ns->rq_reset_mode) 686*cae03e5bSStanislav Fomichev netif_napi_add_config_locked(dev, &qmem->rq->napi, nsim_poll, 687*cae03e5bSStanislav Fomichev idx); 6885bc8e8dbSJakub Kicinski 6895bc8e8dbSJakub Kicinski return 0; 6905bc8e8dbSJakub Kicinski 6915bc8e8dbSJakub Kicinski err_free: 6925bc8e8dbSJakub Kicinski nsim_queue_free(qmem->rq); 6935bc8e8dbSJakub Kicinski return err; 6945bc8e8dbSJakub Kicinski } 6955bc8e8dbSJakub Kicinski 6965bc8e8dbSJakub Kicinski static void nsim_queue_mem_free(struct net_device *dev, void *per_queue_mem) 6975bc8e8dbSJakub Kicinski { 6985bc8e8dbSJakub Kicinski struct nsim_queue_mem *qmem = per_queue_mem; 6995bc8e8dbSJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 7005bc8e8dbSJakub Kicinski 7015bc8e8dbSJakub Kicinski page_pool_destroy(qmem->pp); 7025bc8e8dbSJakub Kicinski if (qmem->rq) { 7035bc8e8dbSJakub Kicinski if (!ns->rq_reset_mode) 704*cae03e5bSStanislav Fomichev netif_napi_del_locked(&qmem->rq->napi); 7055bc8e8dbSJakub Kicinski page_pool_destroy(qmem->rq->page_pool); 7065bc8e8dbSJakub Kicinski nsim_queue_free(qmem->rq); 7075bc8e8dbSJakub Kicinski } 7085bc8e8dbSJakub Kicinski } 7095bc8e8dbSJakub Kicinski 7105bc8e8dbSJakub Kicinski static int 7115bc8e8dbSJakub Kicinski nsim_queue_start(struct net_device *dev, void *per_queue_mem, int idx) 7125bc8e8dbSJakub Kicinski { 7135bc8e8dbSJakub Kicinski struct nsim_queue_mem *qmem = per_queue_mem; 7145bc8e8dbSJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 7155bc8e8dbSJakub Kicinski 716*cae03e5bSStanislav Fomichev netdev_assert_locked(dev); 717*cae03e5bSStanislav Fomichev 7185bc8e8dbSJakub Kicinski if (ns->rq_reset_mode == 1) { 7195bc8e8dbSJakub Kicinski ns->rq[idx]->page_pool = qmem->pp; 720*cae03e5bSStanislav Fomichev napi_enable_locked(&ns->rq[idx]->napi); 7215bc8e8dbSJakub Kicinski return 0; 7225bc8e8dbSJakub Kicinski } 7235bc8e8dbSJakub Kicinski 7245bc8e8dbSJakub Kicinski /* netif_napi_add()/_del() should normally be called from alloc/free, 7255bc8e8dbSJakub Kicinski * here we want to test various call orders. 7265bc8e8dbSJakub Kicinski */ 7275bc8e8dbSJakub Kicinski if (ns->rq_reset_mode == 2) { 728*cae03e5bSStanislav Fomichev netif_napi_del_locked(&ns->rq[idx]->napi); 729*cae03e5bSStanislav Fomichev netif_napi_add_config_locked(dev, &qmem->rq->napi, nsim_poll, 730*cae03e5bSStanislav Fomichev idx); 7315bc8e8dbSJakub Kicinski } else if (ns->rq_reset_mode == 3) { 732*cae03e5bSStanislav Fomichev netif_napi_add_config_locked(dev, &qmem->rq->napi, nsim_poll, 733*cae03e5bSStanislav Fomichev idx); 734*cae03e5bSStanislav Fomichev netif_napi_del_locked(&ns->rq[idx]->napi); 7355bc8e8dbSJakub Kicinski } 7365bc8e8dbSJakub Kicinski 7375bc8e8dbSJakub Kicinski ns->rq[idx] = qmem->rq; 738*cae03e5bSStanislav Fomichev napi_enable_locked(&ns->rq[idx]->napi); 7395bc8e8dbSJakub Kicinski 7405bc8e8dbSJakub Kicinski return 0; 7415bc8e8dbSJakub Kicinski } 7425bc8e8dbSJakub Kicinski 7435bc8e8dbSJakub Kicinski static int nsim_queue_stop(struct net_device *dev, void *per_queue_mem, int idx) 7445bc8e8dbSJakub Kicinski { 7455bc8e8dbSJakub Kicinski struct nsim_queue_mem *qmem = per_queue_mem; 7465bc8e8dbSJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 7475bc8e8dbSJakub Kicinski 748*cae03e5bSStanislav Fomichev netdev_assert_locked(dev); 749*cae03e5bSStanislav Fomichev 750*cae03e5bSStanislav Fomichev napi_disable_locked(&ns->rq[idx]->napi); 7515bc8e8dbSJakub Kicinski 7525bc8e8dbSJakub Kicinski if (ns->rq_reset_mode == 1) { 7535bc8e8dbSJakub Kicinski qmem->pp = ns->rq[idx]->page_pool; 7545bc8e8dbSJakub Kicinski page_pool_disable_direct_recycling(qmem->pp); 7555bc8e8dbSJakub Kicinski } else { 7565bc8e8dbSJakub Kicinski qmem->rq = ns->rq[idx]; 7575bc8e8dbSJakub Kicinski } 7585bc8e8dbSJakub Kicinski 7595bc8e8dbSJakub Kicinski return 0; 7605bc8e8dbSJakub Kicinski } 7615bc8e8dbSJakub Kicinski 7625bc8e8dbSJakub Kicinski static const struct netdev_queue_mgmt_ops nsim_queue_mgmt_ops = { 7635bc8e8dbSJakub Kicinski .ndo_queue_mem_size = sizeof(struct nsim_queue_mem), 7645bc8e8dbSJakub Kicinski .ndo_queue_mem_alloc = nsim_queue_mem_alloc, 7655bc8e8dbSJakub Kicinski .ndo_queue_mem_free = nsim_queue_mem_free, 7665bc8e8dbSJakub Kicinski .ndo_queue_start = nsim_queue_start, 7675bc8e8dbSJakub Kicinski .ndo_queue_stop = nsim_queue_stop, 7685bc8e8dbSJakub Kicinski }; 7695bc8e8dbSJakub Kicinski 7701580cbcbSJakub Kicinski static ssize_t 7716917d207SJakub Kicinski nsim_qreset_write(struct file *file, const char __user *data, 7726917d207SJakub Kicinski size_t count, loff_t *ppos) 7736917d207SJakub Kicinski { 7746917d207SJakub Kicinski struct netdevsim *ns = file->private_data; 7756917d207SJakub Kicinski unsigned int queue, mode; 7766917d207SJakub Kicinski char buf[32]; 7776917d207SJakub Kicinski ssize_t ret; 7786917d207SJakub Kicinski 7796917d207SJakub Kicinski if (count >= sizeof(buf)) 7806917d207SJakub Kicinski return -EINVAL; 7816917d207SJakub Kicinski if (copy_from_user(buf, data, count)) 7826917d207SJakub Kicinski return -EFAULT; 7836917d207SJakub Kicinski buf[count] = '\0'; 7846917d207SJakub Kicinski 7856917d207SJakub Kicinski ret = sscanf(buf, "%u %u", &queue, &mode); 7866917d207SJakub Kicinski if (ret != 2) 7876917d207SJakub Kicinski return -EINVAL; 7886917d207SJakub Kicinski 7896917d207SJakub Kicinski rtnl_lock(); 7906917d207SJakub Kicinski if (queue >= ns->netdev->real_num_rx_queues) { 7916917d207SJakub Kicinski ret = -EINVAL; 7926917d207SJakub Kicinski goto exit_unlock; 7936917d207SJakub Kicinski } 7946917d207SJakub Kicinski 7956917d207SJakub Kicinski ns->rq_reset_mode = mode; 7966917d207SJakub Kicinski ret = netdev_rx_queue_restart(ns->netdev, queue); 7976917d207SJakub Kicinski ns->rq_reset_mode = 0; 7986917d207SJakub Kicinski if (ret) 7996917d207SJakub Kicinski goto exit_unlock; 8006917d207SJakub Kicinski 8016917d207SJakub Kicinski ret = count; 8026917d207SJakub Kicinski exit_unlock: 8036917d207SJakub Kicinski rtnl_unlock(); 8046917d207SJakub Kicinski return ret; 8056917d207SJakub Kicinski } 8066917d207SJakub Kicinski 8076917d207SJakub Kicinski static const struct file_operations nsim_qreset_fops = { 8086917d207SJakub Kicinski .open = simple_open, 8096917d207SJakub Kicinski .write = nsim_qreset_write, 8106917d207SJakub Kicinski .owner = THIS_MODULE, 8116917d207SJakub Kicinski }; 8126917d207SJakub Kicinski 8136917d207SJakub Kicinski static ssize_t 8141580cbcbSJakub Kicinski nsim_pp_hold_read(struct file *file, char __user *data, 8151580cbcbSJakub Kicinski size_t count, loff_t *ppos) 8161580cbcbSJakub Kicinski { 8171580cbcbSJakub Kicinski struct netdevsim *ns = file->private_data; 8181580cbcbSJakub Kicinski char buf[3] = "n\n"; 8191580cbcbSJakub Kicinski 8201580cbcbSJakub Kicinski if (ns->page) 8211580cbcbSJakub Kicinski buf[0] = 'y'; 8221580cbcbSJakub Kicinski 8231580cbcbSJakub Kicinski return simple_read_from_buffer(data, count, ppos, buf, 2); 8241580cbcbSJakub Kicinski } 8251580cbcbSJakub Kicinski 8261580cbcbSJakub Kicinski static ssize_t 8271580cbcbSJakub Kicinski nsim_pp_hold_write(struct file *file, const char __user *data, 8281580cbcbSJakub Kicinski size_t count, loff_t *ppos) 8291580cbcbSJakub Kicinski { 8301580cbcbSJakub Kicinski struct netdevsim *ns = file->private_data; 8311580cbcbSJakub Kicinski ssize_t ret; 8321580cbcbSJakub Kicinski bool val; 8331580cbcbSJakub Kicinski 8341580cbcbSJakub Kicinski ret = kstrtobool_from_user(data, count, &val); 8351580cbcbSJakub Kicinski if (ret) 8361580cbcbSJakub Kicinski return ret; 8371580cbcbSJakub Kicinski 8381580cbcbSJakub Kicinski rtnl_lock(); 8391580cbcbSJakub Kicinski ret = count; 8401580cbcbSJakub Kicinski if (val == !!ns->page) 8411580cbcbSJakub Kicinski goto exit; 8421580cbcbSJakub Kicinski 8431580cbcbSJakub Kicinski if (!netif_running(ns->netdev) && val) { 8441580cbcbSJakub Kicinski ret = -ENETDOWN; 8451580cbcbSJakub Kicinski } else if (val) { 846915c82f8SJakub Kicinski ns->page = page_pool_dev_alloc_pages(ns->rq[0]->page_pool); 8471580cbcbSJakub Kicinski if (!ns->page) 8481580cbcbSJakub Kicinski ret = -ENOMEM; 8491580cbcbSJakub Kicinski } else { 8501580cbcbSJakub Kicinski page_pool_put_full_page(ns->page->pp, ns->page, false); 8511580cbcbSJakub Kicinski ns->page = NULL; 8521580cbcbSJakub Kicinski } 8531580cbcbSJakub Kicinski 8541580cbcbSJakub Kicinski exit: 855b9b8301dSEric Dumazet rtnl_unlock(); 856b9b8301dSEric Dumazet return ret; 8571580cbcbSJakub Kicinski } 8581580cbcbSJakub Kicinski 8591580cbcbSJakub Kicinski static const struct file_operations nsim_pp_hold_fops = { 8601580cbcbSJakub Kicinski .open = simple_open, 8611580cbcbSJakub Kicinski .read = nsim_pp_hold_read, 8621580cbcbSJakub Kicinski .write = nsim_pp_hold_write, 8631580cbcbSJakub Kicinski .llseek = generic_file_llseek, 8641580cbcbSJakub Kicinski .owner = THIS_MODULE, 8651580cbcbSJakub Kicinski }; 8661580cbcbSJakub Kicinski 86783c9e13aSJakub Kicinski static void nsim_setup(struct net_device *dev) 86883c9e13aSJakub Kicinski { 86983c9e13aSJakub Kicinski ether_setup(dev); 87083c9e13aSJakub Kicinski eth_hw_addr_random(dev); 87183c9e13aSJakub Kicinski 87283c9e13aSJakub Kicinski dev->tx_queue_len = 0; 87383c9e13aSJakub Kicinski dev->flags &= ~IFF_MULTICAST; 87483c9e13aSJakub Kicinski dev->priv_flags |= IFF_LIVE_ADDR_CHANGE | 87583c9e13aSJakub Kicinski IFF_NO_QUEUE; 87683c9e13aSJakub Kicinski dev->features |= NETIF_F_HIGHDMA | 87783c9e13aSJakub Kicinski NETIF_F_SG | 87883c9e13aSJakub Kicinski NETIF_F_FRAGLIST | 87983c9e13aSJakub Kicinski NETIF_F_HW_CSUM | 88083c9e13aSJakub Kicinski NETIF_F_TSO; 881494bd83bSSabrina Dubroca dev->hw_features |= NETIF_F_HW_TC | 882494bd83bSSabrina Dubroca NETIF_F_SG | 883494bd83bSSabrina Dubroca NETIF_F_FRAGLIST | 884494bd83bSSabrina Dubroca NETIF_F_HW_CSUM | 885494bd83bSSabrina Dubroca NETIF_F_TSO; 88683c9e13aSJakub Kicinski dev->max_mtu = ETH_MAX_MTU; 88766c0e13aSMarek Majtyka dev->xdp_features = NETDEV_XDP_ACT_HW_OFFLOAD; 88883c9e13aSJakub Kicinski } 88983c9e13aSJakub Kicinski 8903762ec05SDavid Wei static int nsim_queue_init(struct netdevsim *ns) 8913762ec05SDavid Wei { 8923762ec05SDavid Wei struct net_device *dev = ns->netdev; 8933762ec05SDavid Wei int i; 8943762ec05SDavid Wei 895915c82f8SJakub Kicinski ns->rq = kcalloc(dev->num_rx_queues, sizeof(*ns->rq), 896915c82f8SJakub Kicinski GFP_KERNEL_ACCOUNT); 8973762ec05SDavid Wei if (!ns->rq) 8983762ec05SDavid Wei return -ENOMEM; 8993762ec05SDavid Wei 900915c82f8SJakub Kicinski for (i = 0; i < dev->num_rx_queues; i++) { 901a565dd04SJakub Kicinski ns->rq[i] = nsim_queue_alloc(); 902915c82f8SJakub Kicinski if (!ns->rq[i]) 903915c82f8SJakub Kicinski goto err_free_prev; 904915c82f8SJakub Kicinski } 9053762ec05SDavid Wei 9063762ec05SDavid Wei return 0; 907915c82f8SJakub Kicinski 908915c82f8SJakub Kicinski err_free_prev: 909915c82f8SJakub Kicinski while (i--) 910915c82f8SJakub Kicinski kfree(ns->rq[i]); 911915c82f8SJakub Kicinski kfree(ns->rq); 912915c82f8SJakub Kicinski return -ENOMEM; 9133762ec05SDavid Wei } 9143762ec05SDavid Wei 915a565dd04SJakub Kicinski static void nsim_queue_uninit(struct netdevsim *ns) 9163762ec05SDavid Wei { 9173762ec05SDavid Wei struct net_device *dev = ns->netdev; 9183762ec05SDavid Wei int i; 9193762ec05SDavid Wei 920a565dd04SJakub Kicinski for (i = 0; i < dev->num_rx_queues; i++) 921a565dd04SJakub Kicinski nsim_queue_free(ns->rq[i]); 9223762ec05SDavid Wei 923915c82f8SJakub Kicinski kfree(ns->rq); 9243762ec05SDavid Wei ns->rq = NULL; 9253762ec05SDavid Wei } 9263762ec05SDavid Wei 92792ba1f29SDmytro Linkin static int nsim_init_netdevsim(struct netdevsim *ns) 92892ba1f29SDmytro Linkin { 929b63e78fcSVladimir Oltean struct mock_phc *phc; 93092ba1f29SDmytro Linkin int err; 93192ba1f29SDmytro Linkin 932b63e78fcSVladimir Oltean phc = mock_phc_create(&ns->nsim_bus_dev->dev); 933b63e78fcSVladimir Oltean if (IS_ERR(phc)) 934b63e78fcSVladimir Oltean return PTR_ERR(phc); 935b63e78fcSVladimir Oltean 936b63e78fcSVladimir Oltean ns->phc = phc; 93792ba1f29SDmytro Linkin ns->netdev->netdev_ops = &nsim_netdev_ops; 938f216306bSJakub Kicinski ns->netdev->stat_ops = &nsim_stat_ops; 9395bc8e8dbSJakub Kicinski ns->netdev->queue_mgmt_ops = &nsim_queue_mgmt_ops; 94092ba1f29SDmytro Linkin 94192ba1f29SDmytro Linkin err = nsim_udp_tunnels_info_create(ns->nsim_dev, ns->netdev); 94292ba1f29SDmytro Linkin if (err) 943b63e78fcSVladimir Oltean goto err_phc_destroy; 94492ba1f29SDmytro Linkin 94592ba1f29SDmytro Linkin rtnl_lock(); 9463762ec05SDavid Wei err = nsim_queue_init(ns); 94792ba1f29SDmytro Linkin if (err) 94892ba1f29SDmytro Linkin goto err_utn_destroy; 94992ba1f29SDmytro Linkin 9503762ec05SDavid Wei err = nsim_bpf_init(ns); 9513762ec05SDavid Wei if (err) 9523762ec05SDavid Wei goto err_rq_destroy; 9533762ec05SDavid Wei 95402b34d03SSabrina Dubroca nsim_macsec_init(ns); 95592ba1f29SDmytro Linkin nsim_ipsec_init(ns); 95692ba1f29SDmytro Linkin 95792ba1f29SDmytro Linkin err = register_netdevice(ns->netdev); 95892ba1f29SDmytro Linkin if (err) 95992ba1f29SDmytro Linkin goto err_ipsec_teardown; 96092ba1f29SDmytro Linkin rtnl_unlock(); 96192ba1f29SDmytro Linkin return 0; 96292ba1f29SDmytro Linkin 96392ba1f29SDmytro Linkin err_ipsec_teardown: 96492ba1f29SDmytro Linkin nsim_ipsec_teardown(ns); 96502b34d03SSabrina Dubroca nsim_macsec_teardown(ns); 96692ba1f29SDmytro Linkin nsim_bpf_uninit(ns); 9673762ec05SDavid Wei err_rq_destroy: 968a565dd04SJakub Kicinski nsim_queue_uninit(ns); 96992ba1f29SDmytro Linkin err_utn_destroy: 97092ba1f29SDmytro Linkin rtnl_unlock(); 97192ba1f29SDmytro Linkin nsim_udp_tunnels_info_destroy(ns->netdev); 972b63e78fcSVladimir Oltean err_phc_destroy: 973b63e78fcSVladimir Oltean mock_phc_destroy(ns->phc); 97492ba1f29SDmytro Linkin return err; 97592ba1f29SDmytro Linkin } 97692ba1f29SDmytro Linkin 97792ba1f29SDmytro Linkin static int nsim_init_netdevsim_vf(struct netdevsim *ns) 97892ba1f29SDmytro Linkin { 97992ba1f29SDmytro Linkin int err; 98092ba1f29SDmytro Linkin 98192ba1f29SDmytro Linkin ns->netdev->netdev_ops = &nsim_vf_netdev_ops; 98292ba1f29SDmytro Linkin rtnl_lock(); 98392ba1f29SDmytro Linkin err = register_netdevice(ns->netdev); 98492ba1f29SDmytro Linkin rtnl_unlock(); 98592ba1f29SDmytro Linkin return err; 98692ba1f29SDmytro Linkin } 98792ba1f29SDmytro Linkin 988ea937f77SJakub Kicinski static void nsim_exit_netdevsim(struct netdevsim *ns) 989ea937f77SJakub Kicinski { 990ea937f77SJakub Kicinski nsim_udp_tunnels_info_destroy(ns->netdev); 991ea937f77SJakub Kicinski mock_phc_destroy(ns->phc); 992ea937f77SJakub Kicinski } 993ea937f77SJakub Kicinski 994e05b2d14SJiri Pirko struct netdevsim * 995e05b2d14SJiri Pirko nsim_create(struct nsim_dev *nsim_dev, struct nsim_dev_port *nsim_dev_port) 99683c9e13aSJakub Kicinski { 997e05b2d14SJiri Pirko struct net_device *dev; 998e05b2d14SJiri Pirko struct netdevsim *ns; 999af9095f0SJiri Pirko int err; 1000eeeaaf18SJakub Kicinski 1001d4861fc6SPeilin Ye dev = alloc_netdev_mq(sizeof(*ns), "eth%d", NET_NAME_UNKNOWN, nsim_setup, 1002d4861fc6SPeilin Ye nsim_dev->nsim_bus_dev->num_queues); 1003e05b2d14SJiri Pirko if (!dev) 1004e05b2d14SJiri Pirko return ERR_PTR(-ENOMEM); 1005e05b2d14SJiri Pirko 100690d29913SJiri Pirko dev_net_set(dev, nsim_dev_net(nsim_dev)); 1007e05b2d14SJiri Pirko ns = netdev_priv(dev); 10088320d145SJiri Pirko ns->netdev = dev; 1009863a42b2SHillf Danton u64_stats_init(&ns->syncp); 1010e05b2d14SJiri Pirko ns->nsim_dev = nsim_dev; 1011e05b2d14SJiri Pirko ns->nsim_dev_port = nsim_dev_port; 1012e05b2d14SJiri Pirko ns->nsim_bus_dev = nsim_dev->nsim_bus_dev; 101340e4fe4cSJiri Pirko SET_NETDEV_DEV(dev, &ns->nsim_bus_dev->dev); 1014ac73d4bfSJiri Pirko SET_NETDEV_DEVLINK_PORT(dev, &nsim_dev_port->devlink_port); 1015ff1f7c17SJakub Kicinski nsim_ethtool_init(ns); 101692ba1f29SDmytro Linkin if (nsim_dev_port_is_pf(nsim_dev_port)) 101792ba1f29SDmytro Linkin err = nsim_init_netdevsim(ns); 101892ba1f29SDmytro Linkin else 101992ba1f29SDmytro Linkin err = nsim_init_netdevsim_vf(ns); 1020424be63aSJakub Kicinski if (err) 1021424be63aSJakub Kicinski goto err_free_netdev; 10221580cbcbSJakub Kicinski 10231580cbcbSJakub Kicinski ns->pp_dfs = debugfs_create_file("pp_hold", 0600, nsim_dev_port->ddir, 10241580cbcbSJakub Kicinski ns, &nsim_pp_hold_fops); 10256917d207SJakub Kicinski ns->qr_dfs = debugfs_create_file("queue_reset", 0200, 10266917d207SJakub Kicinski nsim_dev_port->ddir, ns, 10276917d207SJakub Kicinski &nsim_qreset_fops); 10281580cbcbSJakub Kicinski 1029e05b2d14SJiri Pirko return ns; 1030e05b2d14SJiri Pirko 1031e05b2d14SJiri Pirko err_free_netdev: 1032e05b2d14SJiri Pirko free_netdev(dev); 1033e05b2d14SJiri Pirko return ERR_PTR(err); 1034e05b2d14SJiri Pirko } 1035e05b2d14SJiri Pirko 1036e05b2d14SJiri Pirko void nsim_destroy(struct netdevsim *ns) 1037e05b2d14SJiri Pirko { 1038e05b2d14SJiri Pirko struct net_device *dev = ns->netdev; 1039f532957dSDavid Wei struct netdevsim *peer; 1040e05b2d14SJiri Pirko 10416917d207SJakub Kicinski debugfs_remove(ns->qr_dfs); 10421580cbcbSJakub Kicinski debugfs_remove(ns->pp_dfs); 10431580cbcbSJakub Kicinski 1044e05b2d14SJiri Pirko rtnl_lock(); 1045f532957dSDavid Wei peer = rtnl_dereference(ns->peer); 1046f532957dSDavid Wei if (peer) 1047f532957dSDavid Wei RCU_INIT_POINTER(peer->peer, NULL); 1048f532957dSDavid Wei RCU_INIT_POINTER(ns->peer, NULL); 1049e05b2d14SJiri Pirko unregister_netdevice(dev); 105092ba1f29SDmytro Linkin if (nsim_dev_port_is_pf(ns->nsim_dev_port)) { 105102b34d03SSabrina Dubroca nsim_macsec_teardown(ns); 1052e05b2d14SJiri Pirko nsim_ipsec_teardown(ns); 1053e05b2d14SJiri Pirko nsim_bpf_uninit(ns); 1054a565dd04SJakub Kicinski nsim_queue_uninit(ns); 105592ba1f29SDmytro Linkin } 1056e05b2d14SJiri Pirko rtnl_unlock(); 105792ba1f29SDmytro Linkin if (nsim_dev_port_is_pf(ns->nsim_dev_port)) 1058ea937f77SJakub Kicinski nsim_exit_netdevsim(ns); 10591580cbcbSJakub Kicinski 10601580cbcbSJakub Kicinski /* Put this intentionally late to exercise the orphaning path */ 10611580cbcbSJakub Kicinski if (ns->page) { 10621580cbcbSJakub Kicinski page_pool_put_full_page(ns->page->pp, ns->page, false); 10631580cbcbSJakub Kicinski ns->page = NULL; 10641580cbcbSJakub Kicinski } 10651580cbcbSJakub Kicinski 1066e05b2d14SJiri Pirko free_netdev(dev); 1067e05b2d14SJiri Pirko } 1068e05b2d14SJiri Pirko 1069f532957dSDavid Wei bool netdev_is_nsim(struct net_device *dev) 1070f532957dSDavid Wei { 1071f532957dSDavid Wei return dev->netdev_ops == &nsim_netdev_ops; 1072f532957dSDavid Wei } 1073f532957dSDavid Wei 1074e05b2d14SJiri Pirko static int nsim_validate(struct nlattr *tb[], struct nlattr *data[], 1075e05b2d14SJiri Pirko struct netlink_ext_ack *extack) 1076e05b2d14SJiri Pirko { 1077d4861fc6SPeilin Ye NL_SET_ERR_MSG_MOD(extack, 1078d4861fc6SPeilin Ye "Please use: echo \"[ID] [PORT_COUNT] [NUM_QUEUES]\" > /sys/bus/netdevsim/new_device"); 1079e05b2d14SJiri Pirko return -EOPNOTSUPP; 1080eeeaaf18SJakub Kicinski } 1081eeeaaf18SJakub Kicinski 108283c9e13aSJakub Kicinski static struct rtnl_link_ops nsim_link_ops __read_mostly = { 108383c9e13aSJakub Kicinski .kind = DRV_NAME, 108483c9e13aSJakub Kicinski .validate = nsim_validate, 108583c9e13aSJakub Kicinski }; 108683c9e13aSJakub Kicinski 108783c9e13aSJakub Kicinski static int __init nsim_module_init(void) 108883c9e13aSJakub Kicinski { 108931d3ad83SJakub Kicinski int err; 109031d3ad83SJakub Kicinski 1091d514f41eSJiri Pirko err = nsim_dev_init(); 1092af9095f0SJiri Pirko if (err) 1093ab1d0cc0SJiri Pirko return err; 1094eeeaaf18SJakub Kicinski 1095925f5afeSJiri Pirko err = nsim_bus_init(); 109631d3ad83SJakub Kicinski if (err) 1097d514f41eSJiri Pirko goto err_dev_exit; 109831d3ad83SJakub Kicinski 1099a5facc4cSJiri Pirko err = rtnl_link_register(&nsim_link_ops); 110079579220SJakub Kicinski if (err) 1101925f5afeSJiri Pirko goto err_bus_exit; 110279579220SJakub Kicinski 110331d3ad83SJakub Kicinski return 0; 110431d3ad83SJakub Kicinski 1105925f5afeSJiri Pirko err_bus_exit: 1106925f5afeSJiri Pirko nsim_bus_exit(); 1107d514f41eSJiri Pirko err_dev_exit: 1108d514f41eSJiri Pirko nsim_dev_exit(); 110931d3ad83SJakub Kicinski return err; 111083c9e13aSJakub Kicinski } 111183c9e13aSJakub Kicinski 111283c9e13aSJakub Kicinski static void __exit nsim_module_exit(void) 111383c9e13aSJakub Kicinski { 111483c9e13aSJakub Kicinski rtnl_link_unregister(&nsim_link_ops); 1115925f5afeSJiri Pirko nsim_bus_exit(); 1116d514f41eSJiri Pirko nsim_dev_exit(); 111783c9e13aSJakub Kicinski } 111883c9e13aSJakub Kicinski 111983c9e13aSJakub Kicinski module_init(nsim_module_init); 112083c9e13aSJakub Kicinski module_exit(nsim_module_exit); 112183c9e13aSJakub Kicinski MODULE_LICENSE("GPL"); 11221fff1f79SJakub Kicinski MODULE_DESCRIPTION("Simulated networking device for testing"); 112383c9e13aSJakub Kicinski MODULE_ALIAS_RTNL_LINK(DRV_NAME); 1124