]> git.proxmox.com Git - mirror_ubuntu-jammy-kernel.git/blame - drivers/net/dummy.c
net: Remove net_rwsem from {, un}register_netdevice_notifier()
[mirror_ubuntu-jammy-kernel.git] / drivers / net / dummy.c
CommitLineData
1da177e4
LT
1/* dummy.c: a dummy net driver
2
3 The purpose of this driver is to provide a device to point a
4 route through, but not to actually transmit packets.
5
6 Why? If you have a machine whose only connection is an occasional
7 PPP/SLIP/PLIP link, you can only connect to your own hostname
8 when the link is up. Otherwise you have to use localhost.
9 This isn't very consistent.
10
11 One solution is to set up a dummy link using PPP/SLIP/PLIP,
12 but this seems (to me) too much overhead for too little gain.
13 This driver provides a small alternative. Thus you can do
6aa20a22 14
1da177e4
LT
15 [when not running slip]
16 ifconfig dummy slip.addr.ess.here up
17 [to go to slip]
18 ifconfig dummy down
19 dip whatever
20
21 This was written by looking at Donald Becker's skeleton driver
22 and the loopback driver. I then threw away anything that didn't
23 apply! Thanks to Alan Cox for the key clue on what to do with
24 misguided packets.
25
26 Nick Holloway, 27th May 1994
27 [I tweaked this explanation a little but that's all]
28 Alan Cox, 30th May 1994
29*/
30
1da177e4
LT
31#include <linux/module.h>
32#include <linux/kernel.h>
33#include <linux/netdevice.h>
34#include <linux/etherdevice.h>
35#include <linux/init.h>
36#include <linux/moduleparam.h>
206c9fb2 37#include <linux/rtnetlink.h>
6df014cf 38#include <linux/net_tstamp.h>
5d5cb173 39#include <net/rtnetlink.h>
6d81f41c 40#include <linux/u64_stats_sync.h>
206c9fb2 41
c19be735
FL
42#define DRV_NAME "dummy"
43#define DRV_VERSION "1.0"
44
1da177e4
LT
45static int numdummies = 1;
46
1da177e4
LT
47/* fake multicast ability */
48static void set_multicast_list(struct net_device *dev)
49{
50}
51
6d81f41c
ED
52struct pcpu_dstats {
53 u64 tx_packets;
54 u64 tx_bytes;
55 struct u64_stats_sync syncp;
56};
57
bc1f4470 58static void dummy_get_stats64(struct net_device *dev,
59 struct rtnl_link_stats64 *stats)
6d81f41c
ED
60{
61 int i;
62
63 for_each_possible_cpu(i) {
64 const struct pcpu_dstats *dstats;
65 u64 tbytes, tpackets;
66 unsigned int start;
67
68 dstats = per_cpu_ptr(dev->dstats, i);
69 do {
57a7744e 70 start = u64_stats_fetch_begin_irq(&dstats->syncp);
6d81f41c
ED
71 tbytes = dstats->tx_bytes;
72 tpackets = dstats->tx_packets;
57a7744e 73 } while (u64_stats_fetch_retry_irq(&dstats->syncp, start));
6d81f41c
ED
74 stats->tx_bytes += tbytes;
75 stats->tx_packets += tpackets;
76 }
6d81f41c 77}
424efe9c
SH
78
79static netdev_tx_t dummy_xmit(struct sk_buff *skb, struct net_device *dev)
80{
6d81f41c
ED
81 struct pcpu_dstats *dstats = this_cpu_ptr(dev->dstats);
82
83 u64_stats_update_begin(&dstats->syncp);
84 dstats->tx_packets++;
85 dstats->tx_bytes += skb->len;
86 u64_stats_update_end(&dstats->syncp);
424efe9c 87
6df014cf 88 skb_tx_timestamp(skb);
424efe9c
SH
89 dev_kfree_skb(skb);
90 return NETDEV_TX_OK;
91}
92
6d81f41c
ED
93static int dummy_dev_init(struct net_device *dev)
94{
1c213bd2 95 dev->dstats = netdev_alloc_pcpu_stats(struct pcpu_dstats);
6d81f41c
ED
96 if (!dev->dstats)
97 return -ENOMEM;
98
99 return 0;
100}
101
890fdf2a 102static void dummy_dev_uninit(struct net_device *dev)
6d81f41c
ED
103{
104 free_percpu(dev->dstats);
6d81f41c
ED
105}
106
210ab665
JP
107static int dummy_change_carrier(struct net_device *dev, bool new_carrier)
108{
109 if (new_carrier)
110 netif_carrier_on(dev);
111 else
112 netif_carrier_off(dev);
113 return 0;
114}
115
aa18e9e8 116static const struct net_device_ops dummy_netdev_ops = {
6d81f41c 117 .ndo_init = dummy_dev_init,
890fdf2a 118 .ndo_uninit = dummy_dev_uninit,
aa18e9e8
SH
119 .ndo_start_xmit = dummy_xmit,
120 .ndo_validate_addr = eth_validate_addr,
afc4b13d 121 .ndo_set_rx_mode = set_multicast_list,
0d1632b4 122 .ndo_set_mac_address = eth_mac_addr,
6d81f41c 123 .ndo_get_stats64 = dummy_get_stats64,
210ab665 124 .ndo_change_carrier = dummy_change_carrier,
aa18e9e8
SH
125};
126
c19be735
FL
127static void dummy_get_drvinfo(struct net_device *dev,
128 struct ethtool_drvinfo *info)
129{
130 strlcpy(info->driver, DRV_NAME, sizeof(info->driver));
131 strlcpy(info->version, DRV_VERSION, sizeof(info->version));
132}
133
6df014cf
ELG
134static int dummy_get_ts_info(struct net_device *dev,
135 struct ethtool_ts_info *ts_info)
136{
137 ts_info->so_timestamping = SOF_TIMESTAMPING_TX_SOFTWARE |
138 SOF_TIMESTAMPING_RX_SOFTWARE |
139 SOF_TIMESTAMPING_SOFTWARE;
140
141 ts_info->phc_index = -1;
142
143 return 0;
144};
145
c19be735
FL
146static const struct ethtool_ops dummy_ethtool_ops = {
147 .get_drvinfo = dummy_get_drvinfo,
6df014cf 148 .get_ts_info = dummy_get_ts_info,
c19be735
FL
149};
150
5d5cb173 151static void dummy_setup(struct net_device *dev)
1da177e4 152{
aa18e9e8
SH
153 ether_setup(dev);
154
1da177e4 155 /* Initialize the device structure. */
aa18e9e8 156 dev->netdev_ops = &dummy_netdev_ops;
c19be735 157 dev->ethtool_ops = &dummy_ethtool_ops;
cf124db5 158 dev->needs_free_netdev = true;
1da177e4
LT
159
160 /* Fill in device structure with ethernet-generic values. */
1da177e4
LT
161 dev->flags |= IFF_NOARP;
162 dev->flags &= ~IFF_MULTICAST;
ff42c02c 163 dev->priv_flags |= IFF_LIVE_ADDR_CHANGE | IFF_NO_QUEUE;
8f3af277 164 dev->features |= NETIF_F_SG | NETIF_F_FRAGLIST;
2082499a 165 dev->features |= NETIF_F_ALL_TSO;
34324dc2 166 dev->features |= NETIF_F_HW_CSUM | NETIF_F_HIGHDMA | NETIF_F_LLTX;
8f3af277
ED
167 dev->features |= NETIF_F_GSO_ENCAP_ALL;
168 dev->hw_features |= dev->features;
169 dev->hw_enc_features |= dev->features;
7ce5d222 170 eth_hw_addr_random(dev);
25e3e84b
ZS
171
172 dev->min_mtu = 0;
e94cd811 173 dev->max_mtu = 0;
1da177e4 174}
6d81f41c 175
a8b8a889
MS
176static int dummy_validate(struct nlattr *tb[], struct nlattr *data[],
177 struct netlink_ext_ack *extack)
0e06877c
PM
178{
179 if (tb[IFLA_ADDRESS]) {
180 if (nla_len(tb[IFLA_ADDRESS]) != ETH_ALEN)
181 return -EINVAL;
182 if (!is_valid_ether_addr(nla_data(tb[IFLA_ADDRESS])))
183 return -EADDRNOTAVAIL;
184 }
185 return 0;
186}
187
5d5cb173 188static struct rtnl_link_ops dummy_link_ops __read_mostly = {
c19be735 189 .kind = DRV_NAME,
5d5cb173 190 .setup = dummy_setup,
0e06877c 191 .validate = dummy_validate,
5d5cb173
PM
192};
193
1da177e4
LT
194/* Number of dummy devices to be set up by this module. */
195module_param(numdummies, int, 0);
196MODULE_PARM_DESC(numdummies, "Number of dummy pseudo devices");
197
206c9fb2 198static int __init dummy_init_one(void)
1da177e4
LT
199{
200 struct net_device *dev_dummy;
201 int err;
202
c3361610 203 dev_dummy = alloc_netdev(0, "dummy%d", NET_NAME_ENUM, dummy_setup);
1da177e4
LT
204 if (!dev_dummy)
205 return -ENOMEM;
206
5d5cb173
PM
207 dev_dummy->rtnl_link_ops = &dummy_link_ops;
208 err = register_netdevice(dev_dummy);
209 if (err < 0)
210 goto err;
5d5cb173 211 return 0;
206c9fb2 212
5d5cb173
PM
213err:
214 free_netdev(dev_dummy);
215 return err;
6aa20a22 216}
1da177e4
LT
217
218static int __init dummy_init_module(void)
6aa20a22 219{
1da177e4 220 int i, err = 0;
206c9fb2 221
5d5cb173
PM
222 rtnl_lock();
223 err = __rtnl_link_register(&dummy_link_ops);
2c8a0189 224 if (err < 0)
225 goto out;
5d5cb173 226
16b0dc29 227 for (i = 0; i < numdummies && !err; i++) {
206c9fb2 228 err = dummy_init_one();
16b0dc29
ED
229 cond_resched();
230 }
2d85cba2 231 if (err < 0)
5d5cb173 232 __rtnl_link_unregister(&dummy_link_ops);
2c8a0189 233
234out:
5d5cb173
PM
235 rtnl_unlock();
236
1da177e4 237 return err;
6aa20a22 238}
1da177e4
LT
239
240static void __exit dummy_cleanup_module(void)
241{
2d85cba2 242 rtnl_link_unregister(&dummy_link_ops);
1da177e4
LT
243}
244
245module_init(dummy_init_module);
246module_exit(dummy_cleanup_module);
247MODULE_LICENSE("GPL");
c19be735 248MODULE_ALIAS_RTNL_LINK(DRV_NAME);
6c702fab 249MODULE_VERSION(DRV_VERSION);