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 90*bf3624cfSBreno Leitao if (!hrtimer_active(&rq->napi_timer)) 91*bf3624cfSBreno 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 40500adf88bSJakub Kicinski netif_napi_add_config(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++) 425915c82f8SJakub Kicinski __netif_napi_del(&ns->rq[i]->napi); 4263762ec05SDavid Wei 4273762ec05SDavid Wei return err; 4283762ec05SDavid Wei } 4293762ec05SDavid Wei 430*bf3624cfSBreno Leitao static enum hrtimer_restart nsim_napi_schedule(struct hrtimer *timer) 431*bf3624cfSBreno Leitao { 432*bf3624cfSBreno Leitao struct nsim_rq *rq; 433*bf3624cfSBreno Leitao 434*bf3624cfSBreno Leitao rq = container_of(timer, struct nsim_rq, napi_timer); 435*bf3624cfSBreno Leitao napi_schedule(&rq->napi); 436*bf3624cfSBreno Leitao 437*bf3624cfSBreno Leitao return HRTIMER_NORESTART; 438*bf3624cfSBreno Leitao } 439*bf3624cfSBreno Leitao 440*bf3624cfSBreno Leitao static void nsim_rq_timer_init(struct nsim_rq *rq) 441*bf3624cfSBreno Leitao { 442*bf3624cfSBreno Leitao hrtimer_init(&rq->napi_timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL); 443*bf3624cfSBreno Leitao rq->napi_timer.function = nsim_napi_schedule; 444*bf3624cfSBreno Leitao } 445*bf3624cfSBreno 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); 4553762ec05SDavid Wei napi_enable(&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 4643762ec05SDavid Wei err = nsim_init_napi(ns); 4653762ec05SDavid Wei if (err) 4663762ec05SDavid Wei return err; 4671580cbcbSJakub Kicinski 4683762ec05SDavid Wei nsim_enable_napi(ns); 4693762ec05SDavid Wei 4703762ec05SDavid Wei return 0; 4713762ec05SDavid Wei } 4723762ec05SDavid Wei 4733762ec05SDavid Wei static void nsim_del_napi(struct netdevsim *ns) 4743762ec05SDavid Wei { 4753762ec05SDavid Wei struct net_device *dev = ns->netdev; 4763762ec05SDavid Wei int i; 4773762ec05SDavid Wei 4783762ec05SDavid Wei for (i = 0; i < dev->num_rx_queues; i++) { 479915c82f8SJakub Kicinski struct nsim_rq *rq = ns->rq[i]; 4803762ec05SDavid Wei 4813762ec05SDavid Wei napi_disable(&rq->napi); 4823762ec05SDavid Wei __netif_napi_del(&rq->napi); 4833762ec05SDavid Wei } 4843762ec05SDavid Wei synchronize_net(); 4853762ec05SDavid Wei 4863762ec05SDavid Wei for (i = 0; i < dev->num_rx_queues; i++) { 487915c82f8SJakub Kicinski page_pool_destroy(ns->rq[i]->page_pool); 488915c82f8SJakub Kicinski ns->rq[i]->page_pool = NULL; 4893762ec05SDavid Wei } 4901580cbcbSJakub Kicinski } 4911580cbcbSJakub Kicinski 4921580cbcbSJakub Kicinski static int nsim_stop(struct net_device *dev) 4931580cbcbSJakub Kicinski { 4941580cbcbSJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 4953762ec05SDavid Wei struct netdevsim *peer; 4961580cbcbSJakub Kicinski 4973762ec05SDavid Wei netif_carrier_off(dev); 4983762ec05SDavid Wei peer = rtnl_dereference(ns->peer); 4993762ec05SDavid Wei if (peer) 5003762ec05SDavid Wei netif_carrier_off(peer->netdev); 5013762ec05SDavid Wei 5023762ec05SDavid Wei nsim_del_napi(ns); 5031580cbcbSJakub Kicinski 5041580cbcbSJakub Kicinski return 0; 5051580cbcbSJakub Kicinski } 5061580cbcbSJakub Kicinski 507b3ea4164SPaolo Abeni static int nsim_shaper_set(struct net_shaper_binding *binding, 508b3ea4164SPaolo Abeni const struct net_shaper *shaper, 509b3ea4164SPaolo Abeni struct netlink_ext_ack *extack) 510b3ea4164SPaolo Abeni { 511b3ea4164SPaolo Abeni return 0; 512b3ea4164SPaolo Abeni } 513b3ea4164SPaolo Abeni 514b3ea4164SPaolo Abeni static int nsim_shaper_del(struct net_shaper_binding *binding, 515b3ea4164SPaolo Abeni const struct net_shaper_handle *handle, 516b3ea4164SPaolo Abeni struct netlink_ext_ack *extack) 517b3ea4164SPaolo Abeni { 518b3ea4164SPaolo Abeni return 0; 519b3ea4164SPaolo Abeni } 520b3ea4164SPaolo Abeni 521b3ea4164SPaolo Abeni static int nsim_shaper_group(struct net_shaper_binding *binding, 522b3ea4164SPaolo Abeni int leaves_count, 523b3ea4164SPaolo Abeni const struct net_shaper *leaves, 524b3ea4164SPaolo Abeni const struct net_shaper *root, 525b3ea4164SPaolo Abeni struct netlink_ext_ack *extack) 526b3ea4164SPaolo Abeni { 527b3ea4164SPaolo Abeni return 0; 528b3ea4164SPaolo Abeni } 529b3ea4164SPaolo Abeni 530b3ea4164SPaolo Abeni static void nsim_shaper_cap(struct net_shaper_binding *binding, 531b3ea4164SPaolo Abeni enum net_shaper_scope scope, 532b3ea4164SPaolo Abeni unsigned long *flags) 533b3ea4164SPaolo Abeni { 534b3ea4164SPaolo Abeni *flags = ULONG_MAX; 535b3ea4164SPaolo Abeni } 536b3ea4164SPaolo Abeni 537b3ea4164SPaolo Abeni static const struct net_shaper_ops nsim_shaper_ops = { 538b3ea4164SPaolo Abeni .set = nsim_shaper_set, 539b3ea4164SPaolo Abeni .delete = nsim_shaper_del, 540b3ea4164SPaolo Abeni .group = nsim_shaper_group, 541b3ea4164SPaolo Abeni .capabilities = nsim_shaper_cap, 542b3ea4164SPaolo Abeni }; 543b3ea4164SPaolo Abeni 54483c9e13aSJakub Kicinski static const struct net_device_ops nsim_netdev_ops = { 54583c9e13aSJakub Kicinski .ndo_start_xmit = nsim_start_xmit, 54683c9e13aSJakub Kicinski .ndo_set_rx_mode = nsim_set_rx_mode, 54783c9e13aSJakub Kicinski .ndo_set_mac_address = eth_mac_addr, 54883c9e13aSJakub Kicinski .ndo_validate_addr = eth_validate_addr, 54931d3ad83SJakub Kicinski .ndo_change_mtu = nsim_change_mtu, 55083c9e13aSJakub Kicinski .ndo_get_stats64 = nsim_get_stats64, 55179579220SJakub Kicinski .ndo_set_vf_mac = nsim_set_vf_mac, 55279579220SJakub Kicinski .ndo_set_vf_vlan = nsim_set_vf_vlan, 55379579220SJakub Kicinski .ndo_set_vf_rate = nsim_set_vf_rate, 55479579220SJakub Kicinski .ndo_set_vf_spoofchk = nsim_set_vf_spoofchk, 55579579220SJakub Kicinski .ndo_set_vf_trust = nsim_set_vf_trust, 55679579220SJakub Kicinski .ndo_get_vf_config = nsim_get_vf_config, 55779579220SJakub Kicinski .ndo_set_vf_link_state = nsim_set_vf_link_state, 55879579220SJakub Kicinski .ndo_set_vf_rss_query_en = nsim_set_vf_rss_query_en, 55931d3ad83SJakub Kicinski .ndo_setup_tc = nsim_setup_tc, 56031d3ad83SJakub Kicinski .ndo_set_features = nsim_set_features, 5618debcf58SDavid Wei .ndo_get_iflink = nsim_get_iflink, 56231d3ad83SJakub Kicinski .ndo_bpf = nsim_bpf, 5631580cbcbSJakub Kicinski .ndo_open = nsim_open, 5641580cbcbSJakub Kicinski .ndo_stop = nsim_stop, 565b3ea4164SPaolo Abeni .net_shaper_ops = &nsim_shaper_ops, 56683c9e13aSJakub Kicinski }; 56783c9e13aSJakub Kicinski 56892ba1f29SDmytro Linkin static const struct net_device_ops nsim_vf_netdev_ops = { 56992ba1f29SDmytro Linkin .ndo_start_xmit = nsim_start_xmit, 57092ba1f29SDmytro Linkin .ndo_set_rx_mode = nsim_set_rx_mode, 57192ba1f29SDmytro Linkin .ndo_set_mac_address = eth_mac_addr, 57292ba1f29SDmytro Linkin .ndo_validate_addr = eth_validate_addr, 57392ba1f29SDmytro Linkin .ndo_change_mtu = nsim_change_mtu, 57492ba1f29SDmytro Linkin .ndo_get_stats64 = nsim_get_stats64, 57592ba1f29SDmytro Linkin .ndo_setup_tc = nsim_setup_tc, 57692ba1f29SDmytro Linkin .ndo_set_features = nsim_set_features, 57792ba1f29SDmytro Linkin }; 57892ba1f29SDmytro Linkin 579f216306bSJakub Kicinski /* We don't have true per-queue stats, yet, so do some random fakery here. 580f216306bSJakub Kicinski * Only report stuff for queue 0. 581f216306bSJakub Kicinski */ 582f216306bSJakub Kicinski static void nsim_get_queue_stats_rx(struct net_device *dev, int idx, 583f216306bSJakub Kicinski struct netdev_queue_stats_rx *stats) 584f216306bSJakub Kicinski { 585f216306bSJakub Kicinski struct rtnl_link_stats64 rtstats = {}; 586f216306bSJakub Kicinski 587f216306bSJakub Kicinski if (!idx) 588f216306bSJakub Kicinski nsim_get_stats64(dev, &rtstats); 589f216306bSJakub Kicinski 590f216306bSJakub Kicinski stats->packets = rtstats.rx_packets - !!rtstats.rx_packets; 591f216306bSJakub Kicinski stats->bytes = rtstats.rx_bytes; 592f216306bSJakub Kicinski } 593f216306bSJakub Kicinski 594f216306bSJakub Kicinski static void nsim_get_queue_stats_tx(struct net_device *dev, int idx, 595f216306bSJakub Kicinski struct netdev_queue_stats_tx *stats) 596f216306bSJakub Kicinski { 597f216306bSJakub Kicinski struct rtnl_link_stats64 rtstats = {}; 598f216306bSJakub Kicinski 599f216306bSJakub Kicinski if (!idx) 600f216306bSJakub Kicinski nsim_get_stats64(dev, &rtstats); 601f216306bSJakub Kicinski 602f216306bSJakub Kicinski stats->packets = rtstats.tx_packets - !!rtstats.tx_packets; 603f216306bSJakub Kicinski stats->bytes = rtstats.tx_bytes; 604f216306bSJakub Kicinski } 605f216306bSJakub Kicinski 606f216306bSJakub Kicinski static void nsim_get_base_stats(struct net_device *dev, 607f216306bSJakub Kicinski struct netdev_queue_stats_rx *rx, 608f216306bSJakub Kicinski struct netdev_queue_stats_tx *tx) 609f216306bSJakub Kicinski { 610f216306bSJakub Kicinski struct rtnl_link_stats64 rtstats = {}; 611f216306bSJakub Kicinski 612f216306bSJakub Kicinski nsim_get_stats64(dev, &rtstats); 613f216306bSJakub Kicinski 614f216306bSJakub Kicinski rx->packets = !!rtstats.rx_packets; 615f216306bSJakub Kicinski rx->bytes = 0; 616f216306bSJakub Kicinski tx->packets = !!rtstats.tx_packets; 617f216306bSJakub Kicinski tx->bytes = 0; 618f216306bSJakub Kicinski } 619f216306bSJakub Kicinski 620f216306bSJakub Kicinski static const struct netdev_stat_ops nsim_stat_ops = { 621f216306bSJakub Kicinski .get_queue_stats_tx = nsim_get_queue_stats_tx, 622f216306bSJakub Kicinski .get_queue_stats_rx = nsim_get_queue_stats_rx, 623f216306bSJakub Kicinski .get_base_stats = nsim_get_base_stats, 624f216306bSJakub Kicinski }; 625f216306bSJakub Kicinski 626a565dd04SJakub Kicinski static struct nsim_rq *nsim_queue_alloc(void) 627a565dd04SJakub Kicinski { 628a565dd04SJakub Kicinski struct nsim_rq *rq; 629a565dd04SJakub Kicinski 630a565dd04SJakub Kicinski rq = kzalloc(sizeof(*rq), GFP_KERNEL_ACCOUNT); 631a565dd04SJakub Kicinski if (!rq) 632a565dd04SJakub Kicinski return NULL; 633a565dd04SJakub Kicinski 634a565dd04SJakub Kicinski skb_queue_head_init(&rq->skb_queue); 635*bf3624cfSBreno Leitao nsim_rq_timer_init(rq); 636a565dd04SJakub Kicinski return rq; 637a565dd04SJakub Kicinski } 638a565dd04SJakub Kicinski 639a565dd04SJakub Kicinski static void nsim_queue_free(struct nsim_rq *rq) 640a565dd04SJakub Kicinski { 641*bf3624cfSBreno Leitao hrtimer_cancel(&rq->napi_timer); 642a565dd04SJakub Kicinski skb_queue_purge_reason(&rq->skb_queue, SKB_DROP_REASON_QUEUE_PURGE); 643a565dd04SJakub Kicinski kfree(rq); 644a565dd04SJakub Kicinski } 645a565dd04SJakub Kicinski 6465bc8e8dbSJakub Kicinski /* Queue reset mode is controlled by ns->rq_reset_mode. 6475bc8e8dbSJakub Kicinski * - normal - new NAPI new pool (old NAPI enabled when new added) 6485bc8e8dbSJakub Kicinski * - mode 1 - allocate new pool (NAPI is only disabled / enabled) 6495bc8e8dbSJakub Kicinski * - mode 2 - new NAPI new pool (old NAPI removed before new added) 6505bc8e8dbSJakub Kicinski * - mode 3 - new NAPI new pool (old NAPI disabled when new added) 6515bc8e8dbSJakub Kicinski */ 6525bc8e8dbSJakub Kicinski struct nsim_queue_mem { 6535bc8e8dbSJakub Kicinski struct nsim_rq *rq; 6545bc8e8dbSJakub Kicinski struct page_pool *pp; 6555bc8e8dbSJakub Kicinski }; 6565bc8e8dbSJakub Kicinski 6575bc8e8dbSJakub Kicinski static int 6585bc8e8dbSJakub Kicinski nsim_queue_mem_alloc(struct net_device *dev, void *per_queue_mem, int idx) 6595bc8e8dbSJakub Kicinski { 6605bc8e8dbSJakub Kicinski struct nsim_queue_mem *qmem = per_queue_mem; 6615bc8e8dbSJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 6625bc8e8dbSJakub Kicinski int err; 6635bc8e8dbSJakub Kicinski 6645bc8e8dbSJakub Kicinski if (ns->rq_reset_mode > 3) 6655bc8e8dbSJakub Kicinski return -EINVAL; 6665bc8e8dbSJakub Kicinski 667285b3f78SJakub Kicinski if (ns->rq_reset_mode == 1) { 668285b3f78SJakub Kicinski if (!netif_running(ns->netdev)) 669285b3f78SJakub Kicinski return -ENETDOWN; 6705bc8e8dbSJakub Kicinski return nsim_create_page_pool(&qmem->pp, &ns->rq[idx]->napi); 671285b3f78SJakub Kicinski } 6725bc8e8dbSJakub Kicinski 6735bc8e8dbSJakub Kicinski qmem->rq = nsim_queue_alloc(); 6745bc8e8dbSJakub Kicinski if (!qmem->rq) 6755bc8e8dbSJakub Kicinski return -ENOMEM; 6765bc8e8dbSJakub Kicinski 6775bc8e8dbSJakub Kicinski err = nsim_create_page_pool(&qmem->rq->page_pool, &qmem->rq->napi); 6785bc8e8dbSJakub Kicinski if (err) 6795bc8e8dbSJakub Kicinski goto err_free; 6805bc8e8dbSJakub Kicinski 6815bc8e8dbSJakub Kicinski if (!ns->rq_reset_mode) 6825bc8e8dbSJakub Kicinski netif_napi_add_config(dev, &qmem->rq->napi, nsim_poll, idx); 6835bc8e8dbSJakub Kicinski 6845bc8e8dbSJakub Kicinski return 0; 6855bc8e8dbSJakub Kicinski 6865bc8e8dbSJakub Kicinski err_free: 6875bc8e8dbSJakub Kicinski nsim_queue_free(qmem->rq); 6885bc8e8dbSJakub Kicinski return err; 6895bc8e8dbSJakub Kicinski } 6905bc8e8dbSJakub Kicinski 6915bc8e8dbSJakub Kicinski static void nsim_queue_mem_free(struct net_device *dev, void *per_queue_mem) 6925bc8e8dbSJakub Kicinski { 6935bc8e8dbSJakub Kicinski struct nsim_queue_mem *qmem = per_queue_mem; 6945bc8e8dbSJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 6955bc8e8dbSJakub Kicinski 6965bc8e8dbSJakub Kicinski page_pool_destroy(qmem->pp); 6975bc8e8dbSJakub Kicinski if (qmem->rq) { 6985bc8e8dbSJakub Kicinski if (!ns->rq_reset_mode) 6995bc8e8dbSJakub Kicinski netif_napi_del(&qmem->rq->napi); 7005bc8e8dbSJakub Kicinski page_pool_destroy(qmem->rq->page_pool); 7015bc8e8dbSJakub Kicinski nsim_queue_free(qmem->rq); 7025bc8e8dbSJakub Kicinski } 7035bc8e8dbSJakub Kicinski } 7045bc8e8dbSJakub Kicinski 7055bc8e8dbSJakub Kicinski static int 7065bc8e8dbSJakub Kicinski nsim_queue_start(struct net_device *dev, void *per_queue_mem, int idx) 7075bc8e8dbSJakub Kicinski { 7085bc8e8dbSJakub Kicinski struct nsim_queue_mem *qmem = per_queue_mem; 7095bc8e8dbSJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 7105bc8e8dbSJakub Kicinski 7115bc8e8dbSJakub Kicinski if (ns->rq_reset_mode == 1) { 7125bc8e8dbSJakub Kicinski ns->rq[idx]->page_pool = qmem->pp; 7135bc8e8dbSJakub Kicinski napi_enable(&ns->rq[idx]->napi); 7145bc8e8dbSJakub Kicinski return 0; 7155bc8e8dbSJakub Kicinski } 7165bc8e8dbSJakub Kicinski 7175bc8e8dbSJakub Kicinski /* netif_napi_add()/_del() should normally be called from alloc/free, 7185bc8e8dbSJakub Kicinski * here we want to test various call orders. 7195bc8e8dbSJakub Kicinski */ 7205bc8e8dbSJakub Kicinski if (ns->rq_reset_mode == 2) { 7215bc8e8dbSJakub Kicinski netif_napi_del(&ns->rq[idx]->napi); 7225bc8e8dbSJakub Kicinski netif_napi_add_config(dev, &qmem->rq->napi, nsim_poll, idx); 7235bc8e8dbSJakub Kicinski } else if (ns->rq_reset_mode == 3) { 7245bc8e8dbSJakub Kicinski netif_napi_add_config(dev, &qmem->rq->napi, nsim_poll, idx); 7255bc8e8dbSJakub Kicinski netif_napi_del(&ns->rq[idx]->napi); 7265bc8e8dbSJakub Kicinski } 7275bc8e8dbSJakub Kicinski 7285bc8e8dbSJakub Kicinski ns->rq[idx] = qmem->rq; 7295bc8e8dbSJakub Kicinski napi_enable(&ns->rq[idx]->napi); 7305bc8e8dbSJakub Kicinski 7315bc8e8dbSJakub Kicinski return 0; 7325bc8e8dbSJakub Kicinski } 7335bc8e8dbSJakub Kicinski 7345bc8e8dbSJakub Kicinski static int nsim_queue_stop(struct net_device *dev, void *per_queue_mem, int idx) 7355bc8e8dbSJakub Kicinski { 7365bc8e8dbSJakub Kicinski struct nsim_queue_mem *qmem = per_queue_mem; 7375bc8e8dbSJakub Kicinski struct netdevsim *ns = netdev_priv(dev); 7385bc8e8dbSJakub Kicinski 7395bc8e8dbSJakub Kicinski napi_disable(&ns->rq[idx]->napi); 7405bc8e8dbSJakub Kicinski 7415bc8e8dbSJakub Kicinski if (ns->rq_reset_mode == 1) { 7425bc8e8dbSJakub Kicinski qmem->pp = ns->rq[idx]->page_pool; 7435bc8e8dbSJakub Kicinski page_pool_disable_direct_recycling(qmem->pp); 7445bc8e8dbSJakub Kicinski } else { 7455bc8e8dbSJakub Kicinski qmem->rq = ns->rq[idx]; 7465bc8e8dbSJakub Kicinski } 7475bc8e8dbSJakub Kicinski 7485bc8e8dbSJakub Kicinski return 0; 7495bc8e8dbSJakub Kicinski } 7505bc8e8dbSJakub Kicinski 7515bc8e8dbSJakub Kicinski static const struct netdev_queue_mgmt_ops nsim_queue_mgmt_ops = { 7525bc8e8dbSJakub Kicinski .ndo_queue_mem_size = sizeof(struct nsim_queue_mem), 7535bc8e8dbSJakub Kicinski .ndo_queue_mem_alloc = nsim_queue_mem_alloc, 7545bc8e8dbSJakub Kicinski .ndo_queue_mem_free = nsim_queue_mem_free, 7555bc8e8dbSJakub Kicinski .ndo_queue_start = nsim_queue_start, 7565bc8e8dbSJakub Kicinski .ndo_queue_stop = nsim_queue_stop, 7575bc8e8dbSJakub Kicinski }; 7585bc8e8dbSJakub Kicinski 7591580cbcbSJakub Kicinski static ssize_t 7606917d207SJakub Kicinski nsim_qreset_write(struct file *file, const char __user *data, 7616917d207SJakub Kicinski size_t count, loff_t *ppos) 7626917d207SJakub Kicinski { 7636917d207SJakub Kicinski struct netdevsim *ns = file->private_data; 7646917d207SJakub Kicinski unsigned int queue, mode; 7656917d207SJakub Kicinski char buf[32]; 7666917d207SJakub Kicinski ssize_t ret; 7676917d207SJakub Kicinski 7686917d207SJakub Kicinski if (count >= sizeof(buf)) 7696917d207SJakub Kicinski return -EINVAL; 7706917d207SJakub Kicinski if (copy_from_user(buf, data, count)) 7716917d207SJakub Kicinski return -EFAULT; 7726917d207SJakub Kicinski buf[count] = '\0'; 7736917d207SJakub Kicinski 7746917d207SJakub Kicinski ret = sscanf(buf, "%u %u", &queue, &mode); 7756917d207SJakub Kicinski if (ret != 2) 7766917d207SJakub Kicinski return -EINVAL; 7776917d207SJakub Kicinski 7786917d207SJakub Kicinski rtnl_lock(); 7796917d207SJakub Kicinski if (queue >= ns->netdev->real_num_rx_queues) { 7806917d207SJakub Kicinski ret = -EINVAL; 7816917d207SJakub Kicinski goto exit_unlock; 7826917d207SJakub Kicinski } 7836917d207SJakub Kicinski 7846917d207SJakub Kicinski ns->rq_reset_mode = mode; 7856917d207SJakub Kicinski ret = netdev_rx_queue_restart(ns->netdev, queue); 7866917d207SJakub Kicinski ns->rq_reset_mode = 0; 7876917d207SJakub Kicinski if (ret) 7886917d207SJakub Kicinski goto exit_unlock; 7896917d207SJakub Kicinski 7906917d207SJakub Kicinski ret = count; 7916917d207SJakub Kicinski exit_unlock: 7926917d207SJakub Kicinski rtnl_unlock(); 7936917d207SJakub Kicinski return ret; 7946917d207SJakub Kicinski } 7956917d207SJakub Kicinski 7966917d207SJakub Kicinski static const struct file_operations nsim_qreset_fops = { 7976917d207SJakub Kicinski .open = simple_open, 7986917d207SJakub Kicinski .write = nsim_qreset_write, 7996917d207SJakub Kicinski .owner = THIS_MODULE, 8006917d207SJakub Kicinski }; 8016917d207SJakub Kicinski 8026917d207SJakub Kicinski static ssize_t 8031580cbcbSJakub Kicinski nsim_pp_hold_read(struct file *file, char __user *data, 8041580cbcbSJakub Kicinski size_t count, loff_t *ppos) 8051580cbcbSJakub Kicinski { 8061580cbcbSJakub Kicinski struct netdevsim *ns = file->private_data; 8071580cbcbSJakub Kicinski char buf[3] = "n\n"; 8081580cbcbSJakub Kicinski 8091580cbcbSJakub Kicinski if (ns->page) 8101580cbcbSJakub Kicinski buf[0] = 'y'; 8111580cbcbSJakub Kicinski 8121580cbcbSJakub Kicinski return simple_read_from_buffer(data, count, ppos, buf, 2); 8131580cbcbSJakub Kicinski } 8141580cbcbSJakub Kicinski 8151580cbcbSJakub Kicinski static ssize_t 8161580cbcbSJakub Kicinski nsim_pp_hold_write(struct file *file, const char __user *data, 8171580cbcbSJakub Kicinski size_t count, loff_t *ppos) 8181580cbcbSJakub Kicinski { 8191580cbcbSJakub Kicinski struct netdevsim *ns = file->private_data; 8201580cbcbSJakub Kicinski ssize_t ret; 8211580cbcbSJakub Kicinski bool val; 8221580cbcbSJakub Kicinski 8231580cbcbSJakub Kicinski ret = kstrtobool_from_user(data, count, &val); 8241580cbcbSJakub Kicinski if (ret) 8251580cbcbSJakub Kicinski return ret; 8261580cbcbSJakub Kicinski 8271580cbcbSJakub Kicinski rtnl_lock(); 8281580cbcbSJakub Kicinski ret = count; 8291580cbcbSJakub Kicinski if (val == !!ns->page) 8301580cbcbSJakub Kicinski goto exit; 8311580cbcbSJakub Kicinski 8321580cbcbSJakub Kicinski if (!netif_running(ns->netdev) && val) { 8331580cbcbSJakub Kicinski ret = -ENETDOWN; 8341580cbcbSJakub Kicinski } else if (val) { 835915c82f8SJakub Kicinski ns->page = page_pool_dev_alloc_pages(ns->rq[0]->page_pool); 8361580cbcbSJakub Kicinski if (!ns->page) 8371580cbcbSJakub Kicinski ret = -ENOMEM; 8381580cbcbSJakub Kicinski } else { 8391580cbcbSJakub Kicinski page_pool_put_full_page(ns->page->pp, ns->page, false); 8401580cbcbSJakub Kicinski ns->page = NULL; 8411580cbcbSJakub Kicinski } 8421580cbcbSJakub Kicinski 8431580cbcbSJakub Kicinski exit: 844b9b8301dSEric Dumazet rtnl_unlock(); 845b9b8301dSEric Dumazet return ret; 8461580cbcbSJakub Kicinski } 8471580cbcbSJakub Kicinski 8481580cbcbSJakub Kicinski static const struct file_operations nsim_pp_hold_fops = { 8491580cbcbSJakub Kicinski .open = simple_open, 8501580cbcbSJakub Kicinski .read = nsim_pp_hold_read, 8511580cbcbSJakub Kicinski .write = nsim_pp_hold_write, 8521580cbcbSJakub Kicinski .llseek = generic_file_llseek, 8531580cbcbSJakub Kicinski .owner = THIS_MODULE, 8541580cbcbSJakub Kicinski }; 8551580cbcbSJakub Kicinski 85683c9e13aSJakub Kicinski static void nsim_setup(struct net_device *dev) 85783c9e13aSJakub Kicinski { 85883c9e13aSJakub Kicinski ether_setup(dev); 85983c9e13aSJakub Kicinski eth_hw_addr_random(dev); 86083c9e13aSJakub Kicinski 86183c9e13aSJakub Kicinski dev->tx_queue_len = 0; 86283c9e13aSJakub Kicinski dev->flags &= ~IFF_MULTICAST; 86383c9e13aSJakub Kicinski dev->priv_flags |= IFF_LIVE_ADDR_CHANGE | 86483c9e13aSJakub Kicinski IFF_NO_QUEUE; 86583c9e13aSJakub Kicinski dev->features |= NETIF_F_HIGHDMA | 86683c9e13aSJakub Kicinski NETIF_F_SG | 86783c9e13aSJakub Kicinski NETIF_F_FRAGLIST | 86883c9e13aSJakub Kicinski NETIF_F_HW_CSUM | 86983c9e13aSJakub Kicinski NETIF_F_TSO; 870494bd83bSSabrina Dubroca dev->hw_features |= NETIF_F_HW_TC | 871494bd83bSSabrina Dubroca NETIF_F_SG | 872494bd83bSSabrina Dubroca NETIF_F_FRAGLIST | 873494bd83bSSabrina Dubroca NETIF_F_HW_CSUM | 874494bd83bSSabrina Dubroca NETIF_F_TSO; 87583c9e13aSJakub Kicinski dev->max_mtu = ETH_MAX_MTU; 87666c0e13aSMarek Majtyka dev->xdp_features = NETDEV_XDP_ACT_HW_OFFLOAD; 87783c9e13aSJakub Kicinski } 87883c9e13aSJakub Kicinski 8793762ec05SDavid Wei static int nsim_queue_init(struct netdevsim *ns) 8803762ec05SDavid Wei { 8813762ec05SDavid Wei struct net_device *dev = ns->netdev; 8823762ec05SDavid Wei int i; 8833762ec05SDavid Wei 884915c82f8SJakub Kicinski ns->rq = kcalloc(dev->num_rx_queues, sizeof(*ns->rq), 885915c82f8SJakub Kicinski GFP_KERNEL_ACCOUNT); 8863762ec05SDavid Wei if (!ns->rq) 8873762ec05SDavid Wei return -ENOMEM; 8883762ec05SDavid Wei 889915c82f8SJakub Kicinski for (i = 0; i < dev->num_rx_queues; i++) { 890a565dd04SJakub Kicinski ns->rq[i] = nsim_queue_alloc(); 891915c82f8SJakub Kicinski if (!ns->rq[i]) 892915c82f8SJakub Kicinski goto err_free_prev; 893915c82f8SJakub Kicinski } 8943762ec05SDavid Wei 8953762ec05SDavid Wei return 0; 896915c82f8SJakub Kicinski 897915c82f8SJakub Kicinski err_free_prev: 898915c82f8SJakub Kicinski while (i--) 899915c82f8SJakub Kicinski kfree(ns->rq[i]); 900915c82f8SJakub Kicinski kfree(ns->rq); 901915c82f8SJakub Kicinski return -ENOMEM; 9023762ec05SDavid Wei } 9033762ec05SDavid Wei 904a565dd04SJakub Kicinski static void nsim_queue_uninit(struct netdevsim *ns) 9053762ec05SDavid Wei { 9063762ec05SDavid Wei struct net_device *dev = ns->netdev; 9073762ec05SDavid Wei int i; 9083762ec05SDavid Wei 909a565dd04SJakub Kicinski for (i = 0; i < dev->num_rx_queues; i++) 910a565dd04SJakub Kicinski nsim_queue_free(ns->rq[i]); 9113762ec05SDavid Wei 912915c82f8SJakub Kicinski kfree(ns->rq); 9133762ec05SDavid Wei ns->rq = NULL; 9143762ec05SDavid Wei } 9153762ec05SDavid Wei 91692ba1f29SDmytro Linkin static int nsim_init_netdevsim(struct netdevsim *ns) 91792ba1f29SDmytro Linkin { 918b63e78fcSVladimir Oltean struct mock_phc *phc; 91992ba1f29SDmytro Linkin int err; 92092ba1f29SDmytro Linkin 921b63e78fcSVladimir Oltean phc = mock_phc_create(&ns->nsim_bus_dev->dev); 922b63e78fcSVladimir Oltean if (IS_ERR(phc)) 923b63e78fcSVladimir Oltean return PTR_ERR(phc); 924b63e78fcSVladimir Oltean 925b63e78fcSVladimir Oltean ns->phc = phc; 92692ba1f29SDmytro Linkin ns->netdev->netdev_ops = &nsim_netdev_ops; 927f216306bSJakub Kicinski ns->netdev->stat_ops = &nsim_stat_ops; 9285bc8e8dbSJakub Kicinski ns->netdev->queue_mgmt_ops = &nsim_queue_mgmt_ops; 92992ba1f29SDmytro Linkin 93092ba1f29SDmytro Linkin err = nsim_udp_tunnels_info_create(ns->nsim_dev, ns->netdev); 93192ba1f29SDmytro Linkin if (err) 932b63e78fcSVladimir Oltean goto err_phc_destroy; 93392ba1f29SDmytro Linkin 93492ba1f29SDmytro Linkin rtnl_lock(); 9353762ec05SDavid Wei err = nsim_queue_init(ns); 93692ba1f29SDmytro Linkin if (err) 93792ba1f29SDmytro Linkin goto err_utn_destroy; 93892ba1f29SDmytro Linkin 9393762ec05SDavid Wei err = nsim_bpf_init(ns); 9403762ec05SDavid Wei if (err) 9413762ec05SDavid Wei goto err_rq_destroy; 9423762ec05SDavid Wei 94302b34d03SSabrina Dubroca nsim_macsec_init(ns); 94492ba1f29SDmytro Linkin nsim_ipsec_init(ns); 94592ba1f29SDmytro Linkin 94692ba1f29SDmytro Linkin err = register_netdevice(ns->netdev); 94792ba1f29SDmytro Linkin if (err) 94892ba1f29SDmytro Linkin goto err_ipsec_teardown; 94992ba1f29SDmytro Linkin rtnl_unlock(); 95092ba1f29SDmytro Linkin return 0; 95192ba1f29SDmytro Linkin 95292ba1f29SDmytro Linkin err_ipsec_teardown: 95392ba1f29SDmytro Linkin nsim_ipsec_teardown(ns); 95402b34d03SSabrina Dubroca nsim_macsec_teardown(ns); 95592ba1f29SDmytro Linkin nsim_bpf_uninit(ns); 9563762ec05SDavid Wei err_rq_destroy: 957a565dd04SJakub Kicinski nsim_queue_uninit(ns); 95892ba1f29SDmytro Linkin err_utn_destroy: 95992ba1f29SDmytro Linkin rtnl_unlock(); 96092ba1f29SDmytro Linkin nsim_udp_tunnels_info_destroy(ns->netdev); 961b63e78fcSVladimir Oltean err_phc_destroy: 962b63e78fcSVladimir Oltean mock_phc_destroy(ns->phc); 96392ba1f29SDmytro Linkin return err; 96492ba1f29SDmytro Linkin } 96592ba1f29SDmytro Linkin 96692ba1f29SDmytro Linkin static int nsim_init_netdevsim_vf(struct netdevsim *ns) 96792ba1f29SDmytro Linkin { 96892ba1f29SDmytro Linkin int err; 96992ba1f29SDmytro Linkin 97092ba1f29SDmytro Linkin ns->netdev->netdev_ops = &nsim_vf_netdev_ops; 97192ba1f29SDmytro Linkin rtnl_lock(); 97292ba1f29SDmytro Linkin err = register_netdevice(ns->netdev); 97392ba1f29SDmytro Linkin rtnl_unlock(); 97492ba1f29SDmytro Linkin return err; 97592ba1f29SDmytro Linkin } 97692ba1f29SDmytro Linkin 977ea937f77SJakub Kicinski static void nsim_exit_netdevsim(struct netdevsim *ns) 978ea937f77SJakub Kicinski { 979ea937f77SJakub Kicinski nsim_udp_tunnels_info_destroy(ns->netdev); 980ea937f77SJakub Kicinski mock_phc_destroy(ns->phc); 981ea937f77SJakub Kicinski } 982ea937f77SJakub Kicinski 983e05b2d14SJiri Pirko struct netdevsim * 984e05b2d14SJiri Pirko nsim_create(struct nsim_dev *nsim_dev, struct nsim_dev_port *nsim_dev_port) 98583c9e13aSJakub Kicinski { 986e05b2d14SJiri Pirko struct net_device *dev; 987e05b2d14SJiri Pirko struct netdevsim *ns; 988af9095f0SJiri Pirko int err; 989eeeaaf18SJakub Kicinski 990d4861fc6SPeilin Ye dev = alloc_netdev_mq(sizeof(*ns), "eth%d", NET_NAME_UNKNOWN, nsim_setup, 991d4861fc6SPeilin Ye nsim_dev->nsim_bus_dev->num_queues); 992e05b2d14SJiri Pirko if (!dev) 993e05b2d14SJiri Pirko return ERR_PTR(-ENOMEM); 994e05b2d14SJiri Pirko 99590d29913SJiri Pirko dev_net_set(dev, nsim_dev_net(nsim_dev)); 996e05b2d14SJiri Pirko ns = netdev_priv(dev); 9978320d145SJiri Pirko ns->netdev = dev; 998863a42b2SHillf Danton u64_stats_init(&ns->syncp); 999e05b2d14SJiri Pirko ns->nsim_dev = nsim_dev; 1000e05b2d14SJiri Pirko ns->nsim_dev_port = nsim_dev_port; 1001e05b2d14SJiri Pirko ns->nsim_bus_dev = nsim_dev->nsim_bus_dev; 100240e4fe4cSJiri Pirko SET_NETDEV_DEV(dev, &ns->nsim_bus_dev->dev); 1003ac73d4bfSJiri Pirko SET_NETDEV_DEVLINK_PORT(dev, &nsim_dev_port->devlink_port); 1004ff1f7c17SJakub Kicinski nsim_ethtool_init(ns); 100592ba1f29SDmytro Linkin if (nsim_dev_port_is_pf(nsim_dev_port)) 100692ba1f29SDmytro Linkin err = nsim_init_netdevsim(ns); 100792ba1f29SDmytro Linkin else 100892ba1f29SDmytro Linkin err = nsim_init_netdevsim_vf(ns); 1009424be63aSJakub Kicinski if (err) 1010424be63aSJakub Kicinski goto err_free_netdev; 10111580cbcbSJakub Kicinski 10121580cbcbSJakub Kicinski ns->pp_dfs = debugfs_create_file("pp_hold", 0600, nsim_dev_port->ddir, 10131580cbcbSJakub Kicinski ns, &nsim_pp_hold_fops); 10146917d207SJakub Kicinski ns->qr_dfs = debugfs_create_file("queue_reset", 0200, 10156917d207SJakub Kicinski nsim_dev_port->ddir, ns, 10166917d207SJakub Kicinski &nsim_qreset_fops); 10171580cbcbSJakub Kicinski 1018e05b2d14SJiri Pirko return ns; 1019e05b2d14SJiri Pirko 1020e05b2d14SJiri Pirko err_free_netdev: 1021e05b2d14SJiri Pirko free_netdev(dev); 1022e05b2d14SJiri Pirko return ERR_PTR(err); 1023e05b2d14SJiri Pirko } 1024e05b2d14SJiri Pirko 1025e05b2d14SJiri Pirko void nsim_destroy(struct netdevsim *ns) 1026e05b2d14SJiri Pirko { 1027e05b2d14SJiri Pirko struct net_device *dev = ns->netdev; 1028f532957dSDavid Wei struct netdevsim *peer; 1029e05b2d14SJiri Pirko 10306917d207SJakub Kicinski debugfs_remove(ns->qr_dfs); 10311580cbcbSJakub Kicinski debugfs_remove(ns->pp_dfs); 10321580cbcbSJakub Kicinski 1033e05b2d14SJiri Pirko rtnl_lock(); 1034f532957dSDavid Wei peer = rtnl_dereference(ns->peer); 1035f532957dSDavid Wei if (peer) 1036f532957dSDavid Wei RCU_INIT_POINTER(peer->peer, NULL); 1037f532957dSDavid Wei RCU_INIT_POINTER(ns->peer, NULL); 1038e05b2d14SJiri Pirko unregister_netdevice(dev); 103992ba1f29SDmytro Linkin if (nsim_dev_port_is_pf(ns->nsim_dev_port)) { 104002b34d03SSabrina Dubroca nsim_macsec_teardown(ns); 1041e05b2d14SJiri Pirko nsim_ipsec_teardown(ns); 1042e05b2d14SJiri Pirko nsim_bpf_uninit(ns); 1043a565dd04SJakub Kicinski nsim_queue_uninit(ns); 104492ba1f29SDmytro Linkin } 1045e05b2d14SJiri Pirko rtnl_unlock(); 104692ba1f29SDmytro Linkin if (nsim_dev_port_is_pf(ns->nsim_dev_port)) 1047ea937f77SJakub Kicinski nsim_exit_netdevsim(ns); 10481580cbcbSJakub Kicinski 10491580cbcbSJakub Kicinski /* Put this intentionally late to exercise the orphaning path */ 10501580cbcbSJakub Kicinski if (ns->page) { 10511580cbcbSJakub Kicinski page_pool_put_full_page(ns->page->pp, ns->page, false); 10521580cbcbSJakub Kicinski ns->page = NULL; 10531580cbcbSJakub Kicinski } 10541580cbcbSJakub Kicinski 1055e05b2d14SJiri Pirko free_netdev(dev); 1056e05b2d14SJiri Pirko } 1057e05b2d14SJiri Pirko 1058f532957dSDavid Wei bool netdev_is_nsim(struct net_device *dev) 1059f532957dSDavid Wei { 1060f532957dSDavid Wei return dev->netdev_ops == &nsim_netdev_ops; 1061f532957dSDavid Wei } 1062f532957dSDavid Wei 1063e05b2d14SJiri Pirko static int nsim_validate(struct nlattr *tb[], struct nlattr *data[], 1064e05b2d14SJiri Pirko struct netlink_ext_ack *extack) 1065e05b2d14SJiri Pirko { 1066d4861fc6SPeilin Ye NL_SET_ERR_MSG_MOD(extack, 1067d4861fc6SPeilin Ye "Please use: echo \"[ID] [PORT_COUNT] [NUM_QUEUES]\" > /sys/bus/netdevsim/new_device"); 1068e05b2d14SJiri Pirko return -EOPNOTSUPP; 1069eeeaaf18SJakub Kicinski } 1070eeeaaf18SJakub Kicinski 107183c9e13aSJakub Kicinski static struct rtnl_link_ops nsim_link_ops __read_mostly = { 107283c9e13aSJakub Kicinski .kind = DRV_NAME, 107383c9e13aSJakub Kicinski .validate = nsim_validate, 107483c9e13aSJakub Kicinski }; 107583c9e13aSJakub Kicinski 107683c9e13aSJakub Kicinski static int __init nsim_module_init(void) 107783c9e13aSJakub Kicinski { 107831d3ad83SJakub Kicinski int err; 107931d3ad83SJakub Kicinski 1080d514f41eSJiri Pirko err = nsim_dev_init(); 1081af9095f0SJiri Pirko if (err) 1082ab1d0cc0SJiri Pirko return err; 1083eeeaaf18SJakub Kicinski 1084925f5afeSJiri Pirko err = nsim_bus_init(); 108531d3ad83SJakub Kicinski if (err) 1086d514f41eSJiri Pirko goto err_dev_exit; 108731d3ad83SJakub Kicinski 1088a5facc4cSJiri Pirko err = rtnl_link_register(&nsim_link_ops); 108979579220SJakub Kicinski if (err) 1090925f5afeSJiri Pirko goto err_bus_exit; 109179579220SJakub Kicinski 109231d3ad83SJakub Kicinski return 0; 109331d3ad83SJakub Kicinski 1094925f5afeSJiri Pirko err_bus_exit: 1095925f5afeSJiri Pirko nsim_bus_exit(); 1096d514f41eSJiri Pirko err_dev_exit: 1097d514f41eSJiri Pirko nsim_dev_exit(); 109831d3ad83SJakub Kicinski return err; 109983c9e13aSJakub Kicinski } 110083c9e13aSJakub Kicinski 110183c9e13aSJakub Kicinski static void __exit nsim_module_exit(void) 110283c9e13aSJakub Kicinski { 110383c9e13aSJakub Kicinski rtnl_link_unregister(&nsim_link_ops); 1104925f5afeSJiri Pirko nsim_bus_exit(); 1105d514f41eSJiri Pirko nsim_dev_exit(); 110683c9e13aSJakub Kicinski } 110783c9e13aSJakub Kicinski 110883c9e13aSJakub Kicinski module_init(nsim_module_init); 110983c9e13aSJakub Kicinski module_exit(nsim_module_exit); 111083c9e13aSJakub Kicinski MODULE_LICENSE("GPL"); 11111fff1f79SJakub Kicinski MODULE_DESCRIPTION("Simulated networking device for testing"); 111283c9e13aSJakub Kicinski MODULE_ALIAS_RTNL_LINK(DRV_NAME); 1113