xref: /openbmc/linux/arch/um/drivers/net_kern.c (revision 857710e1)
1dbddf429SAlex Dewar // SPDX-License-Identifier: GPL-2.0
21da177e4SLinus Torvalds /*
3cd1ae0e4SJeff Dike  * Copyright (C) 2001 - 2007 Jeff Dike (jdike@{addtoit,linux.intel}.com)
41da177e4SLinus Torvalds  * Copyright (C) 2001 Lennert Buytenhek (buytenh@gnu.org) and
51da177e4SLinus Torvalds  * James Leu (jleu@mindspring.net).
61da177e4SLinus Torvalds  * Copyright (C) 2001 by various other people who didn't put their name here.
71da177e4SLinus Torvalds  */
81da177e4SLinus Torvalds 
957c8a661SMike Rapoport #include <linux/memblock.h>
10cd1ae0e4SJeff Dike #include <linux/etherdevice.h>
11cd1ae0e4SJeff Dike #include <linux/ethtool.h>
12cd1ae0e4SJeff Dike #include <linux/inetdevice.h>
13cd1ae0e4SJeff Dike #include <linux/init.h>
14cd1ae0e4SJeff Dike #include <linux/list.h>
15cd1ae0e4SJeff Dike #include <linux/netdevice.h>
16cd1ae0e4SJeff Dike #include <linux/platform_device.h>
17cd1ae0e4SJeff Dike #include <linux/rtnetlink.h>
18cd1ae0e4SJeff Dike #include <linux/skbuff.h>
195a0e3ad6STejun Heo #include <linux/slab.h>
20cd1ae0e4SJeff Dike #include <linux/spinlock.h>
2137185b33SAl Viro #include <init.h>
2237185b33SAl Viro #include <irq_kern.h>
2337185b33SAl Viro #include <irq_user.h>
24cd1ae0e4SJeff Dike #include "mconsole_kern.h"
2537185b33SAl Viro #include <net_kern.h>
2637185b33SAl Viro #include <net_user.h>
271da177e4SLinus Torvalds 
281da177e4SLinus Torvalds #define DRIVER_NAME "uml-netdev"
291da177e4SLinus Torvalds 
301da177e4SLinus Torvalds static DEFINE_SPINLOCK(opened_lock);
319010772cSJeff Dike static LIST_HEAD(opened);
321da177e4SLinus Torvalds 
33605c1e57SJeff Dike /*
34605c1e57SJeff Dike  * The drop_skb is used when we can't allocate an skb.  The
35605c1e57SJeff Dike  * packet is read into drop_skb in order to get the data off the
36605c1e57SJeff Dike  * connection to the host.
37605c1e57SJeff Dike  * It is reallocated whenever a maximum packet size is seen which is
38605c1e57SJeff Dike  * larger than any seen before.  update_drop_skb is called from
39605c1e57SJeff Dike  * eth_configure when a new interface is added.
40605c1e57SJeff Dike  */
41605c1e57SJeff Dike static DEFINE_SPINLOCK(drop_lock);
42605c1e57SJeff Dike static struct sk_buff *drop_skb;
43605c1e57SJeff Dike static int drop_max;
44605c1e57SJeff Dike 
update_drop_skb(int max)45605c1e57SJeff Dike static int update_drop_skb(int max)
46605c1e57SJeff Dike {
47605c1e57SJeff Dike 	struct sk_buff *new;
48605c1e57SJeff Dike 	unsigned long flags;
49605c1e57SJeff Dike 	int err = 0;
50605c1e57SJeff Dike 
51605c1e57SJeff Dike 	spin_lock_irqsave(&drop_lock, flags);
52605c1e57SJeff Dike 
53605c1e57SJeff Dike 	if (max <= drop_max)
54605c1e57SJeff Dike 		goto out;
55605c1e57SJeff Dike 
56605c1e57SJeff Dike 	err = -ENOMEM;
57605c1e57SJeff Dike 	new = dev_alloc_skb(max);
58605c1e57SJeff Dike 	if (new == NULL)
59605c1e57SJeff Dike 		goto out;
60605c1e57SJeff Dike 
61605c1e57SJeff Dike 	skb_put(new, max);
62605c1e57SJeff Dike 
63605c1e57SJeff Dike 	kfree_skb(drop_skb);
64605c1e57SJeff Dike 	drop_skb = new;
65605c1e57SJeff Dike 	drop_max = max;
66605c1e57SJeff Dike 	err = 0;
67605c1e57SJeff Dike out:
68605c1e57SJeff Dike 	spin_unlock_irqrestore(&drop_lock, flags);
69605c1e57SJeff Dike 
70605c1e57SJeff Dike 	return err;
71605c1e57SJeff Dike }
72605c1e57SJeff Dike 
uml_net_rx(struct net_device * dev)731da177e4SLinus Torvalds static int uml_net_rx(struct net_device *dev)
741da177e4SLinus Torvalds {
7517c324faSWang Chen 	struct uml_net_private *lp = netdev_priv(dev);
761da177e4SLinus Torvalds 	int pkt_len;
771da177e4SLinus Torvalds 	struct sk_buff *skb;
781da177e4SLinus Torvalds 
791da177e4SLinus Torvalds 	/* If we can't allocate memory, try again next round. */
80b53f35a8SJeff Dike 	skb = dev_alloc_skb(lp->max_packet);
811da177e4SLinus Torvalds 	if (skb == NULL) {
82605c1e57SJeff Dike 		drop_skb->dev = dev;
83605c1e57SJeff Dike 		/* Read a packet into drop_skb and don't do anything with it. */
84605c1e57SJeff Dike 		(*lp->read)(lp->fd, drop_skb, lp);
85cfa8707aSStephen Hemminger 		dev->stats.rx_dropped++;
861da177e4SLinus Torvalds 		return 0;
871da177e4SLinus Torvalds 	}
881da177e4SLinus Torvalds 
891da177e4SLinus Torvalds 	skb->dev = dev;
90b53f35a8SJeff Dike 	skb_put(skb, lp->max_packet);
91459a98edSArnaldo Carvalho de Melo 	skb_reset_mac_header(skb);
92b53f35a8SJeff Dike 	pkt_len = (*lp->read)(lp->fd, skb, lp);
931da177e4SLinus Torvalds 
941da177e4SLinus Torvalds 	if (pkt_len > 0) {
951da177e4SLinus Torvalds 		skb_trim(skb, pkt_len);
961da177e4SLinus Torvalds 		skb->protocol = (*lp->protocol)(skb);
971da177e4SLinus Torvalds 
98cfa8707aSStephen Hemminger 		dev->stats.rx_bytes += skb->len;
99cfa8707aSStephen Hemminger 		dev->stats.rx_packets++;
100505a41d4SJulia Lawall 		netif_rx(skb);
1011da177e4SLinus Torvalds 		return pkt_len;
1021da177e4SLinus Torvalds 	}
1031da177e4SLinus Torvalds 
1041da177e4SLinus Torvalds 	kfree_skb(skb);
1051da177e4SLinus Torvalds 	return pkt_len;
1061da177e4SLinus Torvalds }
1071da177e4SLinus Torvalds 
uml_dev_close(struct work_struct * work)108eff3b634SPeter Zijlstra static void uml_dev_close(struct work_struct *work)
10971c8d4c3SPaolo 'Blaisorblade' Giarrusso {
110eff3b634SPeter Zijlstra 	struct uml_net_private *lp =
111eff3b634SPeter Zijlstra 		container_of(work, struct uml_net_private, work);
112eff3b634SPeter Zijlstra 	dev_close(lp->dev);
11371c8d4c3SPaolo 'Blaisorblade' Giarrusso }
11471c8d4c3SPaolo 'Blaisorblade' Giarrusso 
uml_net_interrupt(int irq,void * dev_id)115074a0db8SWANG Cong static irqreturn_t uml_net_interrupt(int irq, void *dev_id)
1161da177e4SLinus Torvalds {
1171da177e4SLinus Torvalds 	struct net_device *dev = dev_id;
11817c324faSWang Chen 	struct uml_net_private *lp = netdev_priv(dev);
1191da177e4SLinus Torvalds 	int err;
1201da177e4SLinus Torvalds 
1211da177e4SLinus Torvalds 	if (!netif_running(dev))
122cd1ae0e4SJeff Dike 		return IRQ_NONE;
1231da177e4SLinus Torvalds 
1241da177e4SLinus Torvalds 	spin_lock(&lp->lock);
1251da177e4SLinus Torvalds 	while ((err = uml_net_rx(dev)) > 0) ;
1261da177e4SLinus Torvalds 	if (err < 0) {
1271da177e4SLinus Torvalds 		printk(KERN_ERR
1281da177e4SLinus Torvalds 		       "Device '%s' read returned %d, shutting it down\n",
1291da177e4SLinus Torvalds 		       dev->name, err);
13071c8d4c3SPaolo 'Blaisorblade' Giarrusso 		/* dev_close can't be called in interrupt context, and takes
13171c8d4c3SPaolo 'Blaisorblade' Giarrusso 		 * again lp->lock.
13271c8d4c3SPaolo 'Blaisorblade' Giarrusso 		 * And dev_close() can be safely called multiple times on the
13371c8d4c3SPaolo 'Blaisorblade' Giarrusso 		 * same device, since it tests for (dev->flags & IFF_UP). So
134eff3b634SPeter Zijlstra 		 * there's no harm in delaying the device shutdown.
135eff3b634SPeter Zijlstra 		 * Furthermore, the workqueue will not re-enqueue an already
136eff3b634SPeter Zijlstra 		 * enqueued work item. */
137eff3b634SPeter Zijlstra 		schedule_work(&lp->work);
1381da177e4SLinus Torvalds 		goto out;
1391da177e4SLinus Torvalds 	}
1401da177e4SLinus Torvalds out:
1411da177e4SLinus Torvalds 	spin_unlock(&lp->lock);
1424ea21cd9SJeff Dike 	return IRQ_HANDLED;
1431da177e4SLinus Torvalds }
1441da177e4SLinus Torvalds 
uml_net_open(struct net_device * dev)1451da177e4SLinus Torvalds static int uml_net_open(struct net_device *dev)
1461da177e4SLinus Torvalds {
14717c324faSWang Chen 	struct uml_net_private *lp = netdev_priv(dev);
1481da177e4SLinus Torvalds 	int err;
1491da177e4SLinus Torvalds 
1501da177e4SLinus Torvalds 	if (lp->fd >= 0) {
1511da177e4SLinus Torvalds 		err = -ENXIO;
1521da177e4SLinus Torvalds 		goto out;
1531da177e4SLinus Torvalds 	}
1541da177e4SLinus Torvalds 
1551da177e4SLinus Torvalds 	lp->fd = (*lp->open)(&lp->user);
1561da177e4SLinus Torvalds 	if (lp->fd < 0) {
1571da177e4SLinus Torvalds 		err = lp->fd;
1581da177e4SLinus Torvalds 		goto out;
1591da177e4SLinus Torvalds 	}
1601da177e4SLinus Torvalds 
1611da177e4SLinus Torvalds 	err = um_request_irq(dev->irq, lp->fd, IRQ_READ, uml_net_interrupt,
162c0b79a90SYong Zhang 			     IRQF_SHARED, dev->name, dev);
16336d46a59SJohannes Berg 	if (err < 0) {
1641da177e4SLinus Torvalds 		printk(KERN_ERR "uml_net_open: failed to get irq(%d)\n", err);
1651da177e4SLinus Torvalds 		err = -ENETUNREACH;
16614d9ead0SJeff Dike 		goto out_close;
1671da177e4SLinus Torvalds 	}
1681da177e4SLinus Torvalds 
1691da177e4SLinus Torvalds 	netif_start_queue(dev);
1701da177e4SLinus Torvalds 
1711da177e4SLinus Torvalds 	/* clear buffer - it can happen that the host side of the interface
1721da177e4SLinus Torvalds 	 * is full when we get here.  In this case, new data is never queued,
1731da177e4SLinus Torvalds 	 * SIGIOs never arrive, and the net never works.
1741da177e4SLinus Torvalds 	 */
1751da177e4SLinus Torvalds 	while ((err = uml_net_rx(dev)) > 0) ;
1761da177e4SLinus Torvalds 
17714d9ead0SJeff Dike 	spin_lock(&opened_lock);
17814d9ead0SJeff Dike 	list_add(&lp->list, &opened);
17914d9ead0SJeff Dike 	spin_unlock(&opened_lock);
18014d9ead0SJeff Dike 
18114d9ead0SJeff Dike 	return 0;
18214d9ead0SJeff Dike out_close:
18314d9ead0SJeff Dike 	if (lp->close != NULL) (*lp->close)(lp->fd, &lp->user);
18414d9ead0SJeff Dike 	lp->fd = -1;
1851da177e4SLinus Torvalds out:
18614d9ead0SJeff Dike 	return err;
1871da177e4SLinus Torvalds }
1881da177e4SLinus Torvalds 
uml_net_close(struct net_device * dev)1891da177e4SLinus Torvalds static int uml_net_close(struct net_device *dev)
1901da177e4SLinus Torvalds {
19117c324faSWang Chen 	struct uml_net_private *lp = netdev_priv(dev);
1921da177e4SLinus Torvalds 
1931da177e4SLinus Torvalds 	netif_stop_queue(dev);
1941da177e4SLinus Torvalds 
195fa7a0449SRichard Weinberger 	um_free_irq(dev->irq, dev);
1961da177e4SLinus Torvalds 	if (lp->close != NULL)
1971da177e4SLinus Torvalds 		(*lp->close)(lp->fd, &lp->user);
1981da177e4SLinus Torvalds 	lp->fd = -1;
1991da177e4SLinus Torvalds 
20014d9ead0SJeff Dike 	spin_lock(&opened_lock);
20114d9ead0SJeff Dike 	list_del(&lp->list);
20214d9ead0SJeff Dike 	spin_unlock(&opened_lock);
20314d9ead0SJeff Dike 
2041da177e4SLinus Torvalds 	return 0;
2051da177e4SLinus Torvalds }
2061da177e4SLinus Torvalds 
uml_net_start_xmit(struct sk_buff * skb,struct net_device * dev)207*857710e1SNathan Chancellor static netdev_tx_t uml_net_start_xmit(struct sk_buff *skb, struct net_device *dev)
2081da177e4SLinus Torvalds {
20917c324faSWang Chen 	struct uml_net_private *lp = netdev_priv(dev);
2101da177e4SLinus Torvalds 	unsigned long flags;
2111da177e4SLinus Torvalds 	int len;
2121da177e4SLinus Torvalds 
2131da177e4SLinus Torvalds 	netif_stop_queue(dev);
2141da177e4SLinus Torvalds 
2151da177e4SLinus Torvalds 	spin_lock_irqsave(&lp->lock, flags);
2161da177e4SLinus Torvalds 
217b53f35a8SJeff Dike 	len = (*lp->write)(lp->fd, skb, lp);
21855ea1cfaSPaul Chavent 	skb_tx_timestamp(skb);
2191da177e4SLinus Torvalds 
2201da177e4SLinus Torvalds 	if (len == skb->len) {
221cfa8707aSStephen Hemminger 		dev->stats.tx_packets++;
222cfa8707aSStephen Hemminger 		dev->stats.tx_bytes += skb->len;
223860e9538SFlorian Westphal 		netif_trans_update(dev);
2241da177e4SLinus Torvalds 		netif_start_queue(dev);
2251da177e4SLinus Torvalds 
2261da177e4SLinus Torvalds 		/* this is normally done in the interrupt when tx finishes */
2271da177e4SLinus Torvalds 		netif_wake_queue(dev);
2281da177e4SLinus Torvalds 	}
2291da177e4SLinus Torvalds 	else if (len == 0) {
2301da177e4SLinus Torvalds 		netif_start_queue(dev);
231cfa8707aSStephen Hemminger 		dev->stats.tx_dropped++;
2321da177e4SLinus Torvalds 	}
2331da177e4SLinus Torvalds 	else {
2341da177e4SLinus Torvalds 		netif_start_queue(dev);
2351da177e4SLinus Torvalds 		printk(KERN_ERR "uml_net_start_xmit: failed(%d)\n", len);
2361da177e4SLinus Torvalds 	}
2371da177e4SLinus Torvalds 
2381da177e4SLinus Torvalds 	spin_unlock_irqrestore(&lp->lock, flags);
2391da177e4SLinus Torvalds 
240fabfb91dSEric W. Biederman 	dev_consume_skb_any(skb);
2411da177e4SLinus Torvalds 
2426ed10654SPatrick McHardy 	return NETDEV_TX_OK;
2431da177e4SLinus Torvalds }
2441da177e4SLinus Torvalds 
uml_net_set_multicast_list(struct net_device * dev)2451da177e4SLinus Torvalds static void uml_net_set_multicast_list(struct net_device *dev)
2461da177e4SLinus Torvalds {
247cd1ae0e4SJeff Dike 	return;
2481da177e4SLinus Torvalds }
2491da177e4SLinus Torvalds 
uml_net_tx_timeout(struct net_device * dev,unsigned int txqueue)2500290bd29SMichael S. Tsirkin static void uml_net_tx_timeout(struct net_device *dev, unsigned int txqueue)
2511da177e4SLinus Torvalds {
252860e9538SFlorian Westphal 	netif_trans_update(dev);
2531da177e4SLinus Torvalds 	netif_wake_queue(dev);
2541da177e4SLinus Torvalds }
2551da177e4SLinus Torvalds 
256dd71dc4cSRichard Weinberger #ifdef CONFIG_NET_POLL_CONTROLLER
uml_net_poll_controller(struct net_device * dev)257dd71dc4cSRichard Weinberger static void uml_net_poll_controller(struct net_device *dev)
258dd71dc4cSRichard Weinberger {
259dd71dc4cSRichard Weinberger 	disable_irq(dev->irq);
260dd71dc4cSRichard Weinberger 	uml_net_interrupt(dev->irq, dev);
261dd71dc4cSRichard Weinberger 	enable_irq(dev->irq);
262dd71dc4cSRichard Weinberger }
263dd71dc4cSRichard Weinberger #endif
264dd71dc4cSRichard Weinberger 
uml_net_get_drvinfo(struct net_device * dev,struct ethtool_drvinfo * info)2656d387484SChristoph Hellwig static void uml_net_get_drvinfo(struct net_device *dev,
2666d387484SChristoph Hellwig 				struct ethtool_drvinfo *info)
2671da177e4SLinus Torvalds {
268e6e4d33fSWolfram Sang 	strscpy(info->driver, DRIVER_NAME, sizeof(info->driver));
2696d387484SChristoph Hellwig }
2701da177e4SLinus Torvalds 
2710fc0b732SStephen Hemminger static const struct ethtool_ops uml_net_ethtool_ops = {
2726d387484SChristoph Hellwig 	.get_drvinfo	= uml_net_get_drvinfo,
2736d387484SChristoph Hellwig 	.get_link	= ethtool_op_get_link,
27455ea1cfaSPaul Chavent 	.get_ts_info	= ethtool_op_get_ts_info,
2756d387484SChristoph Hellwig };
2761da177e4SLinus Torvalds 
uml_net_setup_etheraddr(struct net_device * dev,char * str)27749da7e64SAnton Ivanov void uml_net_setup_etheraddr(struct net_device *dev, char *str)
278b10aeeefSJeff Dike {
279ac617341SJakub Kicinski 	u8 addr[ETH_ALEN];
280b10aeeefSJeff Dike 	char *end;
281b10aeeefSJeff Dike 	int i;
282b10aeeefSJeff Dike 
283b10aeeefSJeff Dike 	if (str == NULL)
284b10aeeefSJeff Dike 		goto random;
285b10aeeefSJeff Dike 
286b10aeeefSJeff Dike 	for (i = 0; i < 6; i++) {
287b10aeeefSJeff Dike 		addr[i] = simple_strtoul(str, &end, 16);
288b10aeeefSJeff Dike 		if ((end == str) ||
289b10aeeefSJeff Dike 		   ((*end != ':') && (*end != ',') && (*end != '\0'))) {
290b10aeeefSJeff Dike 			printk(KERN_ERR
291b10aeeefSJeff Dike 			       "setup_etheraddr: failed to parse '%s' "
292b10aeeefSJeff Dike 			       "as an ethernet address\n", str);
293b10aeeefSJeff Dike 			goto random;
294b10aeeefSJeff Dike 		}
295b10aeeefSJeff Dike 		str = end + 1;
296b10aeeefSJeff Dike 	}
297e024715fSPaolo 'Blaisorblade' Giarrusso 	if (is_multicast_ether_addr(addr)) {
298b10aeeefSJeff Dike 		printk(KERN_ERR
299e024715fSPaolo 'Blaisorblade' Giarrusso 		       "Attempt to assign a multicast ethernet address to a "
300b10aeeefSJeff Dike 		       "device disallowed\n");
301b10aeeefSJeff Dike 		goto random;
302b10aeeefSJeff Dike 	}
303e024715fSPaolo 'Blaisorblade' Giarrusso 	if (!is_valid_ether_addr(addr)) {
304e024715fSPaolo 'Blaisorblade' Giarrusso 		printk(KERN_ERR
305e024715fSPaolo 'Blaisorblade' Giarrusso 		       "Attempt to assign an invalid ethernet address to a "
306e024715fSPaolo 'Blaisorblade' Giarrusso 		       "device disallowed\n");
307e024715fSPaolo 'Blaisorblade' Giarrusso 		goto random;
308e024715fSPaolo 'Blaisorblade' Giarrusso 	}
309e024715fSPaolo 'Blaisorblade' Giarrusso 	if (!is_local_ether_addr(addr)) {
310e024715fSPaolo 'Blaisorblade' Giarrusso 		printk(KERN_WARNING
3112278c5acSJeff Dike 		       "Warning: Assigning a globally valid ethernet "
3127d98230aSJeff Dike 		       "address to a device\n");
3132278c5acSJeff Dike 		printk(KERN_WARNING "You should set the 2nd rightmost bit in "
3142278c5acSJeff Dike 		       "the first byte of the MAC,\n");
3157d98230aSJeff Dike 		printk(KERN_WARNING "i.e. %02x:%02x:%02x:%02x:%02x:%02x\n",
3167d98230aSJeff Dike 		       addr[0] | 0x02, addr[1], addr[2], addr[3], addr[4],
3177d98230aSJeff Dike 		       addr[5]);
318e024715fSPaolo 'Blaisorblade' Giarrusso 	}
319ac617341SJakub Kicinski 	eth_hw_addr_set(dev, addr);
320646cbcdaSJiri Pirko 	return;
321b10aeeefSJeff Dike 
322b10aeeefSJeff Dike random:
323e024715fSPaolo 'Blaisorblade' Giarrusso 	printk(KERN_INFO
324646cbcdaSJiri Pirko 	       "Choosing a random ethernet address for device %s\n", dev->name);
325646cbcdaSJiri Pirko 	eth_hw_addr_random(dev);
326b10aeeefSJeff Dike }
327b10aeeefSJeff Dike 
3281da177e4SLinus Torvalds static DEFINE_SPINLOCK(devices_lock);
3299010772cSJeff Dike static LIST_HEAD(devices);
3301da177e4SLinus Torvalds 
3313ae5eaecSRussell King static struct platform_driver uml_net_driver = {
3323ae5eaecSRussell King 	.driver = {
3331da177e4SLinus Torvalds 		.name  = DRIVER_NAME,
3343ae5eaecSRussell King 	},
3351da177e4SLinus Torvalds };
3361da177e4SLinus Torvalds 
net_device_release(struct device * dev)3372e3f5251SJeff Dike static void net_device_release(struct device *dev)
3382e3f5251SJeff Dike {
3398691b97bSGreg Kroah-Hartman 	struct uml_net *device = dev_get_drvdata(dev);
3402e3f5251SJeff Dike 	struct net_device *netdev = device->dev;
34117c324faSWang Chen 	struct uml_net_private *lp = netdev_priv(netdev);
3422e3f5251SJeff Dike 
3432e3f5251SJeff Dike 	if (lp->remove != NULL)
3442e3f5251SJeff Dike 		(*lp->remove)(&lp->user);
3452e3f5251SJeff Dike 	list_del(&device->list);
3462e3f5251SJeff Dike 	kfree(device);
3472e3f5251SJeff Dike 	free_netdev(netdev);
3482e3f5251SJeff Dike }
3492e3f5251SJeff Dike 
3508bb95b39SStephen Hemminger static const struct net_device_ops uml_netdev_ops = {
3518bb95b39SStephen Hemminger 	.ndo_open 		= uml_net_open,
3528bb95b39SStephen Hemminger 	.ndo_stop 		= uml_net_close,
3538bb95b39SStephen Hemminger 	.ndo_start_xmit 	= uml_net_start_xmit,
354afc4b13dSJiri Pirko 	.ndo_set_rx_mode	= uml_net_set_multicast_list,
3558bb95b39SStephen Hemminger 	.ndo_tx_timeout 	= uml_net_tx_timeout,
3569337057dSBoaz Harrosh 	.ndo_set_mac_address	= eth_mac_addr,
3578bb95b39SStephen Hemminger 	.ndo_validate_addr	= eth_validate_addr,
358dd71dc4cSRichard Weinberger #ifdef CONFIG_NET_POLL_CONTROLLER
359dd71dc4cSRichard Weinberger 	.ndo_poll_controller = uml_net_poll_controller,
360dd71dc4cSRichard Weinberger #endif
3618bb95b39SStephen Hemminger };
3628bb95b39SStephen Hemminger 
36380e39311SJeff Dike /*
36480e39311SJeff Dike  * Ensures that platform_driver_register is called only once by
36580e39311SJeff Dike  * eth_configure.  Will be set in an initcall.
36680e39311SJeff Dike  */
36780e39311SJeff Dike static int driver_registered;
36880e39311SJeff Dike 
eth_configure(int n,void * init,char * mac,struct transport * transport,gfp_t gfp_mask)369f34d9d2dSJeff Dike static void eth_configure(int n, void *init, char *mac,
370e17c6d77SSaurabh Sengar 			  struct transport *transport, gfp_t gfp_mask)
3711da177e4SLinus Torvalds {
3721da177e4SLinus Torvalds 	struct uml_net *device;
3731da177e4SLinus Torvalds 	struct net_device *dev;
3741da177e4SLinus Torvalds 	struct uml_net_private *lp;
375c74c69b4SPaolo 'Blaisorblade' Giarrusso 	int err, size;
3761da177e4SLinus Torvalds 
377c74c69b4SPaolo 'Blaisorblade' Giarrusso 	size = transport->private_size + sizeof(struct uml_net_private);
3781da177e4SLinus Torvalds 
379e17c6d77SSaurabh Sengar 	device = kzalloc(sizeof(*device), gfp_mask);
3801da177e4SLinus Torvalds 	if (device == NULL) {
3818c840835SPaolo 'Blaisorblade' Giarrusso 		printk(KERN_ERR "eth_configure failed to allocate struct "
3828c840835SPaolo 'Blaisorblade' Giarrusso 		       "uml_net\n");
383f34d9d2dSJeff Dike 		return;
3841da177e4SLinus Torvalds 	}
3851da177e4SLinus Torvalds 
3868c840835SPaolo 'Blaisorblade' Giarrusso 	dev = alloc_etherdev(size);
3878c840835SPaolo 'Blaisorblade' Giarrusso 	if (dev == NULL) {
3888c840835SPaolo 'Blaisorblade' Giarrusso 		printk(KERN_ERR "eth_configure: failed to allocate struct "
3898c840835SPaolo 'Blaisorblade' Giarrusso 		       "net_device for eth%d\n", n);
3908c840835SPaolo 'Blaisorblade' Giarrusso 		goto out_free_device;
3918c840835SPaolo 'Blaisorblade' Giarrusso 	}
3928c840835SPaolo 'Blaisorblade' Giarrusso 
3931da177e4SLinus Torvalds 	INIT_LIST_HEAD(&device->list);
3941da177e4SLinus Torvalds 	device->index = n;
3951da177e4SLinus Torvalds 
396e024715fSPaolo 'Blaisorblade' Giarrusso 	/* If this name ends up conflicting with an existing registered
397e024715fSPaolo 'Blaisorblade' Giarrusso 	 * netdevice, that is OK, register_netdev{,ice}() will notice this
398e024715fSPaolo 'Blaisorblade' Giarrusso 	 * and fail.
399e024715fSPaolo 'Blaisorblade' Giarrusso 	 */
4008c840835SPaolo 'Blaisorblade' Giarrusso 	snprintf(dev->name, sizeof(dev->name), "eth%d", n);
401e024715fSPaolo 'Blaisorblade' Giarrusso 
40249da7e64SAnton Ivanov 	uml_net_setup_etheraddr(dev, mac);
4031da177e4SLinus Torvalds 
404646cbcdaSJiri Pirko 	printk(KERN_INFO "Netdevice %d (%pM) : ", n, dev->dev_addr);
4051da177e4SLinus Torvalds 
40617c324faSWang Chen 	lp = netdev_priv(dev);
407e56a7885SPaolo 'Blaisorblade' Giarrusso 	/* This points to the transport private data. It's still clear, but we
408e56a7885SPaolo 'Blaisorblade' Giarrusso 	 * must memset it to 0 *now*. Let's help the drivers. */
409e56a7885SPaolo 'Blaisorblade' Giarrusso 	memset(lp, 0, size);
410eff3b634SPeter Zijlstra 	INIT_WORK(&lp->work, uml_dev_close);
411e56a7885SPaolo 'Blaisorblade' Giarrusso 
4121da177e4SLinus Torvalds 	/* sysfs register */
4131da177e4SLinus Torvalds 	if (!driver_registered) {
4143ae5eaecSRussell King 		platform_driver_register(&uml_net_driver);
4151da177e4SLinus Torvalds 		driver_registered = 1;
4161da177e4SLinus Torvalds 	}
4171da177e4SLinus Torvalds 	device->pdev.id = n;
4181da177e4SLinus Torvalds 	device->pdev.name = DRIVER_NAME;
4192e3f5251SJeff Dike 	device->pdev.dev.release = net_device_release;
4208691b97bSGreg Kroah-Hartman 	dev_set_drvdata(&device->pdev.dev, device);
421f34d9d2dSJeff Dike 	if (platform_device_register(&device->pdev))
422f34d9d2dSJeff Dike 		goto out_free_netdev;
4231da177e4SLinus Torvalds 	SET_NETDEV_DEV(dev,&device->pdev.dev);
4241da177e4SLinus Torvalds 
4251da177e4SLinus Torvalds 	device->dev = dev;
4261da177e4SLinus Torvalds 
427f34d9d2dSJeff Dike 	/*
428f34d9d2dSJeff Dike 	 * These just fill in a data structure, so there's no failure
429f34d9d2dSJeff Dike 	 * to be worried about.
430f34d9d2dSJeff Dike 	 */
4311da177e4SLinus Torvalds 	(*transport->kern->init)(dev, init);
4321da177e4SLinus Torvalds 
4331da177e4SLinus Torvalds 	*lp = ((struct uml_net_private)
4341da177e4SLinus Torvalds 		{ .list  		= LIST_HEAD_INIT(lp->list),
4351da177e4SLinus Torvalds 		  .dev 			= dev,
4361da177e4SLinus Torvalds 		  .fd 			= -1,
4371da177e4SLinus Torvalds 		  .mac 			= { 0xfe, 0xfd, 0x0, 0x0, 0x0, 0x0},
438b53f35a8SJeff Dike 		  .max_packet		= transport->user->max_packet,
4391da177e4SLinus Torvalds 		  .protocol 		= transport->kern->protocol,
4401da177e4SLinus Torvalds 		  .open 		= transport->user->open,
4411da177e4SLinus Torvalds 		  .close 		= transport->user->close,
4421da177e4SLinus Torvalds 		  .remove 		= transport->user->remove,
4431da177e4SLinus Torvalds 		  .read 		= transport->kern->read,
4441da177e4SLinus Torvalds 		  .write 		= transport->kern->write,
4451da177e4SLinus Torvalds 		  .add_address 		= transport->user->add_address,
446b53f35a8SJeff Dike 		  .delete_address  	= transport->user->delete_address });
4471da177e4SLinus Torvalds 
4481da177e4SLinus Torvalds 	spin_lock_init(&lp->lock);
449646cbcdaSJiri Pirko 	memcpy(lp->mac, dev->dev_addr, sizeof(lp->mac));
4501da177e4SLinus Torvalds 
451f34d9d2dSJeff Dike 	if ((transport->user->init != NULL) &&
452f34d9d2dSJeff Dike 	    ((*transport->user->init)(&lp->user, dev) != 0))
453f34d9d2dSJeff Dike 		goto out_unregister;
4541da177e4SLinus Torvalds 
455b53f35a8SJeff Dike 	dev->mtu = transport->user->mtu;
4568bb95b39SStephen Hemminger 	dev->netdev_ops = &uml_netdev_ops;
457f34d9d2dSJeff Dike 	dev->ethtool_ops = &uml_net_ethtool_ops;
458f34d9d2dSJeff Dike 	dev->watchdog_timeo = (HZ >> 1);
459f34d9d2dSJeff Dike 	dev->irq = UM_ETH_IRQ;
4601da177e4SLinus Torvalds 
461605c1e57SJeff Dike 	err = update_drop_skb(lp->max_packet);
462605c1e57SJeff Dike 	if (err)
463605c1e57SJeff Dike 		goto out_undo_user_init;
464605c1e57SJeff Dike 
465f34d9d2dSJeff Dike 	rtnl_lock();
466f34d9d2dSJeff Dike 	err = register_netdevice(dev);
467f34d9d2dSJeff Dike 	rtnl_unlock();
468f34d9d2dSJeff Dike 	if (err)
469f34d9d2dSJeff Dike 		goto out_undo_user_init;
470f34d9d2dSJeff Dike 
471f34d9d2dSJeff Dike 	spin_lock(&devices_lock);
472f34d9d2dSJeff Dike 	list_add(&device->list, &devices);
473f34d9d2dSJeff Dike 	spin_unlock(&devices_lock);
474f34d9d2dSJeff Dike 
475f34d9d2dSJeff Dike 	return;
476f34d9d2dSJeff Dike 
477f34d9d2dSJeff Dike out_undo_user_init:
4788c840835SPaolo 'Blaisorblade' Giarrusso 	if (transport->user->remove != NULL)
479f34d9d2dSJeff Dike 		(*transport->user->remove)(&lp->user);
480f34d9d2dSJeff Dike out_unregister:
481f34d9d2dSJeff Dike 	platform_device_unregister(&device->pdev);
4827d98230aSJeff Dike 	return; /* platform_device_unregister frees dev and device */
483f34d9d2dSJeff Dike out_free_netdev:
484f34d9d2dSJeff Dike 	free_netdev(dev);
4858c840835SPaolo 'Blaisorblade' Giarrusso out_free_device:
486f34d9d2dSJeff Dike 	kfree(device);
4871da177e4SLinus Torvalds }
4881da177e4SLinus Torvalds 
find_device(int n)4891da177e4SLinus Torvalds static struct uml_net *find_device(int n)
4901da177e4SLinus Torvalds {
4911da177e4SLinus Torvalds 	struct uml_net *device;
4921da177e4SLinus Torvalds 	struct list_head *ele;
4931da177e4SLinus Torvalds 
4941da177e4SLinus Torvalds 	spin_lock(&devices_lock);
4951da177e4SLinus Torvalds 	list_for_each(ele, &devices) {
4961da177e4SLinus Torvalds 		device = list_entry(ele, struct uml_net, list);
4971da177e4SLinus Torvalds 		if (device->index == n)
4981da177e4SLinus Torvalds 			goto out;
4991da177e4SLinus Torvalds 	}
5001da177e4SLinus Torvalds 	device = NULL;
5011da177e4SLinus Torvalds  out:
5021da177e4SLinus Torvalds 	spin_unlock(&devices_lock);
5034ea21cd9SJeff Dike 	return device;
5041da177e4SLinus Torvalds }
5051da177e4SLinus Torvalds 
eth_parse(char * str,int * index_out,char ** str_out,char ** error_out)506f28169d2SJeff Dike static int eth_parse(char *str, int *index_out, char **str_out,
507f28169d2SJeff Dike 		     char **error_out)
5081da177e4SLinus Torvalds {
5091da177e4SLinus Torvalds 	char *end;
510a419aef8SJoe Perches 	int n, err = -EINVAL;
5111da177e4SLinus Torvalds 
5121da177e4SLinus Torvalds 	n = simple_strtoul(str, &end, 0);
5131da177e4SLinus Torvalds 	if (end == str) {
514f28169d2SJeff Dike 		*error_out = "Bad device number";
515f28169d2SJeff Dike 		return err;
5161da177e4SLinus Torvalds 	}
517f28169d2SJeff Dike 
5181da177e4SLinus Torvalds 	str = end;
5191da177e4SLinus Torvalds 	if (*str != '=') {
520f28169d2SJeff Dike 		*error_out = "Expected '=' after device number";
521f28169d2SJeff Dike 		return err;
5221da177e4SLinus Torvalds 	}
523f28169d2SJeff Dike 
5241da177e4SLinus Torvalds 	str++;
5251da177e4SLinus Torvalds 	if (find_device(n)) {
526f28169d2SJeff Dike 		*error_out = "Device already configured";
527f28169d2SJeff Dike 		return err;
5281da177e4SLinus Torvalds 	}
529f28169d2SJeff Dike 
530f28169d2SJeff Dike 	*index_out = n;
5311da177e4SLinus Torvalds 	*str_out = str;
532f28169d2SJeff Dike 	return 0;
5331da177e4SLinus Torvalds }
5341da177e4SLinus Torvalds 
5351da177e4SLinus Torvalds struct eth_init {
5361da177e4SLinus Torvalds 	struct list_head list;
5371da177e4SLinus Torvalds 	char *init;
5381da177e4SLinus Torvalds 	int index;
5391da177e4SLinus Torvalds };
5401da177e4SLinus Torvalds 
541d3b7f69dSJeff Dike static DEFINE_SPINLOCK(transports_lock);
542d3b7f69dSJeff Dike static LIST_HEAD(transports);
5431da177e4SLinus Torvalds 
5441da177e4SLinus Torvalds /* Filled in during early boot */
545c862fc32SJeff Dike static LIST_HEAD(eth_cmd_line);
5461da177e4SLinus Torvalds 
check_transport(struct transport * transport,char * eth,int n,void ** init_out,char ** mac_out,gfp_t gfp_mask)5471da177e4SLinus Torvalds static int check_transport(struct transport *transport, char *eth, int n,
548e17c6d77SSaurabh Sengar 			   void **init_out, char **mac_out, gfp_t gfp_mask)
5491da177e4SLinus Torvalds {
5501da177e4SLinus Torvalds 	int len;
5511da177e4SLinus Torvalds 
5521da177e4SLinus Torvalds 	len = strlen(transport->name);
5531da177e4SLinus Torvalds 	if (strncmp(eth, transport->name, len))
5544ea21cd9SJeff Dike 		return 0;
5551da177e4SLinus Torvalds 
5561da177e4SLinus Torvalds 	eth += len;
5571da177e4SLinus Torvalds 	if (*eth == ',')
5581da177e4SLinus Torvalds 		eth++;
5591da177e4SLinus Torvalds 	else if (*eth != '\0')
5604ea21cd9SJeff Dike 		return 0;
5611da177e4SLinus Torvalds 
562e17c6d77SSaurabh Sengar 	*init_out = kmalloc(transport->setup_size, gfp_mask);
5631da177e4SLinus Torvalds 	if (*init_out == NULL)
5644ea21cd9SJeff Dike 		return 1;
5651da177e4SLinus Torvalds 
5661da177e4SLinus Torvalds 	if (!transport->setup(eth, mac_out, *init_out)) {
5671da177e4SLinus Torvalds 		kfree(*init_out);
5681da177e4SLinus Torvalds 		*init_out = NULL;
5691da177e4SLinus Torvalds 	}
5704ea21cd9SJeff Dike 	return 1;
5711da177e4SLinus Torvalds }
5721da177e4SLinus Torvalds 
register_transport(struct transport * new)5731da177e4SLinus Torvalds void register_transport(struct transport *new)
5741da177e4SLinus Torvalds {
5751da177e4SLinus Torvalds 	struct list_head *ele, *next;
5761da177e4SLinus Torvalds 	struct eth_init *eth;
5771da177e4SLinus Torvalds 	void *init;
5781da177e4SLinus Torvalds 	char *mac = NULL;
5791da177e4SLinus Torvalds 	int match;
5801da177e4SLinus Torvalds 
581d3b7f69dSJeff Dike 	spin_lock(&transports_lock);
582d3b7f69dSJeff Dike 	BUG_ON(!list_empty(&new->list));
5831da177e4SLinus Torvalds 	list_add(&new->list, &transports);
584d3b7f69dSJeff Dike 	spin_unlock(&transports_lock);
5851da177e4SLinus Torvalds 
5861da177e4SLinus Torvalds 	list_for_each_safe(ele, next, &eth_cmd_line) {
5871da177e4SLinus Torvalds 		eth = list_entry(ele, struct eth_init, list);
5881da177e4SLinus Torvalds 		match = check_transport(new, eth->init, eth->index, &init,
589e17c6d77SSaurabh Sengar 					&mac, GFP_KERNEL);
5901da177e4SLinus Torvalds 		if (!match)
5911da177e4SLinus Torvalds 			continue;
5921da177e4SLinus Torvalds 		else if (init != NULL) {
593e17c6d77SSaurabh Sengar 			eth_configure(eth->index, init, mac, new, GFP_KERNEL);
5941da177e4SLinus Torvalds 			kfree(init);
5951da177e4SLinus Torvalds 		}
5961da177e4SLinus Torvalds 		list_del(&eth->list);
5971da177e4SLinus Torvalds 	}
5981da177e4SLinus Torvalds }
5991da177e4SLinus Torvalds 
eth_setup_common(char * str,int index)6001da177e4SLinus Torvalds static int eth_setup_common(char *str, int index)
6011da177e4SLinus Torvalds {
6021da177e4SLinus Torvalds 	struct list_head *ele;
6031da177e4SLinus Torvalds 	struct transport *transport;
6041da177e4SLinus Torvalds 	void *init;
6051da177e4SLinus Torvalds 	char *mac = NULL;
606c862fc32SJeff Dike 	int found = 0;
6071da177e4SLinus Torvalds 
608c862fc32SJeff Dike 	spin_lock(&transports_lock);
6091da177e4SLinus Torvalds 	list_for_each(ele, &transports) {
6101da177e4SLinus Torvalds 		transport = list_entry(ele, struct transport, list);
611e17c6d77SSaurabh Sengar 	        if (!check_transport(transport, str, index, &init,
612e17c6d77SSaurabh Sengar 					&mac, GFP_ATOMIC))
6131da177e4SLinus Torvalds 			continue;
6141da177e4SLinus Torvalds 		if (init != NULL) {
615e17c6d77SSaurabh Sengar 			eth_configure(index, init, mac, transport, GFP_ATOMIC);
6161da177e4SLinus Torvalds 			kfree(init);
6171da177e4SLinus Torvalds 		}
618c862fc32SJeff Dike 		found = 1;
619c862fc32SJeff Dike 		break;
6201da177e4SLinus Torvalds 	}
621c862fc32SJeff Dike 
622c862fc32SJeff Dike 	spin_unlock(&transports_lock);
623c862fc32SJeff Dike 	return found;
6241da177e4SLinus Torvalds }
6251da177e4SLinus Torvalds 
eth_setup(char * str)62697a1fcbbSJeff Dike static int __init eth_setup(char *str)
6271da177e4SLinus Torvalds {
6281da177e4SLinus Torvalds 	struct eth_init *new;
629f28169d2SJeff Dike 	char *error;
6301da177e4SLinus Torvalds 	int n, err;
6311da177e4SLinus Torvalds 
632f28169d2SJeff Dike 	err = eth_parse(str, &n, &str, &error);
633f28169d2SJeff Dike 	if (err) {
634f28169d2SJeff Dike 		printk(KERN_ERR "eth_setup - Couldn't parse '%s' : %s\n",
635f28169d2SJeff Dike 		       str, error);
6361183dc94SJeff Dike 		return 1;
637f28169d2SJeff Dike 	}
6381da177e4SLinus Torvalds 
6397e1c4e27SMike Rapoport 	new = memblock_alloc(sizeof(*new), SMP_CACHE_BYTES);
6408a7f97b9SMike Rapoport 	if (!new)
6418a7f97b9SMike Rapoport 		panic("%s: Failed to allocate %zu bytes\n", __func__,
6428a7f97b9SMike Rapoport 		      sizeof(*new));
6431da177e4SLinus Torvalds 
6441da177e4SLinus Torvalds 	INIT_LIST_HEAD(&new->list);
6451da177e4SLinus Torvalds 	new->index = n;
6461da177e4SLinus Torvalds 	new->init = str;
6471da177e4SLinus Torvalds 
6481da177e4SLinus Torvalds 	list_add_tail(&new->list, &eth_cmd_line);
6491183dc94SJeff Dike 	return 1;
6501da177e4SLinus Torvalds }
6511da177e4SLinus Torvalds 
6521da177e4SLinus Torvalds __setup("eth", eth_setup);
6531da177e4SLinus Torvalds __uml_help(eth_setup,
6541da177e4SLinus Torvalds "eth[0-9]+=<transport>,<options>\n"
6551da177e4SLinus Torvalds "    Configure a network device.\n\n"
6561da177e4SLinus Torvalds );
6571da177e4SLinus Torvalds 
net_config(char * str,char ** error_out)658f28169d2SJeff Dike static int net_config(char *str, char **error_out)
6591da177e4SLinus Torvalds {
6601da177e4SLinus Torvalds 	int n, err;
6611da177e4SLinus Torvalds 
662f28169d2SJeff Dike 	err = eth_parse(str, &n, &str, error_out);
663f28169d2SJeff Dike 	if (err)
664f28169d2SJeff Dike 		return err;
6651da177e4SLinus Torvalds 
666f28169d2SJeff Dike 	/* This string is broken up and the pieces used by the underlying
667f28169d2SJeff Dike 	 * driver.  So, it is freed only if eth_setup_common fails.
668f28169d2SJeff Dike 	 */
669970d6e3aSJeff Dike 	str = kstrdup(str, GFP_KERNEL);
6701da177e4SLinus Torvalds 	if (str == NULL) {
671f28169d2SJeff Dike 	        *error_out = "net_config failed to strdup string";
672f28169d2SJeff Dike 		return -ENOMEM;
6731da177e4SLinus Torvalds 	}
6741da177e4SLinus Torvalds 	err = !eth_setup_common(str, n);
6751da177e4SLinus Torvalds 	if (err)
6761da177e4SLinus Torvalds 		kfree(str);
677cd1ae0e4SJeff Dike 	return err;
6781da177e4SLinus Torvalds }
6791da177e4SLinus Torvalds 
net_id(char ** str,int * start_out,int * end_out)68029d56cfeSJeff Dike static int net_id(char **str, int *start_out, int *end_out)
68129d56cfeSJeff Dike {
68229d56cfeSJeff Dike 	char *end;
68329d56cfeSJeff Dike 	int n;
68429d56cfeSJeff Dike 
68529d56cfeSJeff Dike 	n = simple_strtoul(*str, &end, 0);
68629d56cfeSJeff Dike 	if ((*end != '\0') || (end == *str))
68729d56cfeSJeff Dike 		return -1;
68829d56cfeSJeff Dike 
68929d56cfeSJeff Dike 	*start_out = n;
69029d56cfeSJeff Dike 	*end_out = n;
69129d56cfeSJeff Dike 	*str = end;
69229d56cfeSJeff Dike 	return n;
69329d56cfeSJeff Dike }
69429d56cfeSJeff Dike 
net_remove(int n,char ** error_out)695f28169d2SJeff Dike static int net_remove(int n, char **error_out)
6961da177e4SLinus Torvalds {
6971da177e4SLinus Torvalds 	struct uml_net *device;
6981da177e4SLinus Torvalds 	struct net_device *dev;
6991da177e4SLinus Torvalds 	struct uml_net_private *lp;
7001da177e4SLinus Torvalds 
7011da177e4SLinus Torvalds 	device = find_device(n);
7021da177e4SLinus Torvalds 	if (device == NULL)
70329d56cfeSJeff Dike 		return -ENODEV;
7041da177e4SLinus Torvalds 
7051da177e4SLinus Torvalds 	dev = device->dev;
70617c324faSWang Chen 	lp = netdev_priv(dev);
70729d56cfeSJeff Dike 	if (lp->fd > 0)
70829d56cfeSJeff Dike 		return -EBUSY;
7091da177e4SLinus Torvalds 	unregister_netdev(dev);
7101da177e4SLinus Torvalds 	platform_device_unregister(&device->pdev);
7111da177e4SLinus Torvalds 
71229d56cfeSJeff Dike 	return 0;
7131da177e4SLinus Torvalds }
7141da177e4SLinus Torvalds 
7151da177e4SLinus Torvalds static struct mc_device net_mc = {
71684f48d4fSJeff Dike 	.list		= LIST_HEAD_INIT(net_mc.list),
7171da177e4SLinus Torvalds 	.name		= "eth",
7181da177e4SLinus Torvalds 	.config		= net_config,
7191da177e4SLinus Torvalds 	.get_config	= NULL,
72029d56cfeSJeff Dike 	.id		= net_id,
7211da177e4SLinus Torvalds 	.remove		= net_remove,
7221da177e4SLinus Torvalds };
7231da177e4SLinus Torvalds 
72432f862c3SJeff Dike #ifdef CONFIG_INET
uml_inetaddr_event(struct notifier_block * this,unsigned long event,void * ptr)7251da177e4SLinus Torvalds static int uml_inetaddr_event(struct notifier_block *this, unsigned long event,
7261da177e4SLinus Torvalds 			      void *ptr)
7271da177e4SLinus Torvalds {
7281da177e4SLinus Torvalds 	struct in_ifaddr *ifa = ptr;
7291da177e4SLinus Torvalds 	struct net_device *dev = ifa->ifa_dev->dev;
7301da177e4SLinus Torvalds 	struct uml_net_private *lp;
7311da177e4SLinus Torvalds 	void (*proc)(unsigned char *, unsigned char *, void *);
7321da177e4SLinus Torvalds 	unsigned char addr_buf[4], netmask_buf[4];
7331da177e4SLinus Torvalds 
734ebe28bb4SMiklos Szeredi 	if (dev->netdev_ops->ndo_open != uml_net_open)
7354ea21cd9SJeff Dike 		return NOTIFY_DONE;
7361da177e4SLinus Torvalds 
73717c324faSWang Chen 	lp = netdev_priv(dev);
7381da177e4SLinus Torvalds 
7391da177e4SLinus Torvalds 	proc = NULL;
7401da177e4SLinus Torvalds 	switch (event) {
7411da177e4SLinus Torvalds 	case NETDEV_UP:
7421da177e4SLinus Torvalds 		proc = lp->add_address;
7431da177e4SLinus Torvalds 		break;
7441da177e4SLinus Torvalds 	case NETDEV_DOWN:
7451da177e4SLinus Torvalds 		proc = lp->delete_address;
7461da177e4SLinus Torvalds 		break;
7471da177e4SLinus Torvalds 	}
7481da177e4SLinus Torvalds 	if (proc != NULL) {
7490e76422cSBodo Stroesser 		memcpy(addr_buf, &ifa->ifa_address, sizeof(addr_buf));
7500e76422cSBodo Stroesser 		memcpy(netmask_buf, &ifa->ifa_mask, sizeof(netmask_buf));
7511da177e4SLinus Torvalds 		(*proc)(addr_buf, netmask_buf, &lp->user);
7521da177e4SLinus Torvalds 	}
7534ea21cd9SJeff Dike 	return NOTIFY_DONE;
7541da177e4SLinus Torvalds }
7551da177e4SLinus Torvalds 
756c862fc32SJeff Dike /* uml_net_init shouldn't be called twice on two CPUs at the same time */
757074a0db8SWANG Cong static struct notifier_block uml_inetaddr_notifier = {
7581da177e4SLinus Torvalds 	.notifier_call		= uml_inetaddr_event,
7591da177e4SLinus Torvalds };
7601da177e4SLinus Torvalds 
inet_register(void)76132f862c3SJeff Dike static void inet_register(void)
7621da177e4SLinus Torvalds {
7631da177e4SLinus Torvalds 	struct list_head *ele;
7641da177e4SLinus Torvalds 	struct uml_net_private *lp;
7651da177e4SLinus Torvalds 	struct in_device *ip;
7661da177e4SLinus Torvalds 	struct in_ifaddr *in;
7671da177e4SLinus Torvalds 
7681da177e4SLinus Torvalds 	register_inetaddr_notifier(&uml_inetaddr_notifier);
7691da177e4SLinus Torvalds 
7701da177e4SLinus Torvalds 	/* Devices may have been opened already, so the uml_inetaddr_notifier
7711da177e4SLinus Torvalds 	 * didn't get a chance to run for them.  This fakes it so that
7721da177e4SLinus Torvalds 	 * addresses which have already been set up get handled properly.
7731da177e4SLinus Torvalds 	 */
774c862fc32SJeff Dike 	spin_lock(&opened_lock);
7751da177e4SLinus Torvalds 	list_for_each(ele, &opened) {
7761da177e4SLinus Torvalds 		lp = list_entry(ele, struct uml_net_private, list);
7771da177e4SLinus Torvalds 		ip = lp->dev->ip_ptr;
778c862fc32SJeff Dike 		if (ip == NULL)
779c862fc32SJeff Dike 			continue;
7801da177e4SLinus Torvalds 		in = ip->ifa_list;
7811da177e4SLinus Torvalds 		while (in != NULL) {
7821da177e4SLinus Torvalds 			uml_inetaddr_event(NULL, NETDEV_UP, in);
7831da177e4SLinus Torvalds 			in = in->ifa_next;
7841da177e4SLinus Torvalds 		}
7851da177e4SLinus Torvalds 	}
786c862fc32SJeff Dike 	spin_unlock(&opened_lock);
78732f862c3SJeff Dike }
78832f862c3SJeff Dike #else
inet_register(void)78932f862c3SJeff Dike static inline void inet_register(void)
79032f862c3SJeff Dike {
79132f862c3SJeff Dike }
79232f862c3SJeff Dike #endif
7931da177e4SLinus Torvalds 
uml_net_init(void)79432f862c3SJeff Dike static int uml_net_init(void)
79532f862c3SJeff Dike {
79632f862c3SJeff Dike 	mconsole_register_dev(&net_mc);
79732f862c3SJeff Dike 	inet_register();
798c862fc32SJeff Dike 	return 0;
7991da177e4SLinus Torvalds }
8001da177e4SLinus Torvalds 
8011da177e4SLinus Torvalds __initcall(uml_net_init);
8021da177e4SLinus Torvalds 
close_devices(void)8031da177e4SLinus Torvalds static void close_devices(void)
8041da177e4SLinus Torvalds {
8051da177e4SLinus Torvalds 	struct list_head *ele;
8061da177e4SLinus Torvalds 	struct uml_net_private *lp;
8071da177e4SLinus Torvalds 
808c862fc32SJeff Dike 	spin_lock(&opened_lock);
8091da177e4SLinus Torvalds 	list_for_each(ele, &opened) {
8101da177e4SLinus Torvalds 		lp = list_entry(ele, struct uml_net_private, list);
811fa7a0449SRichard Weinberger 		um_free_irq(lp->dev->irq, lp->dev);
8121da177e4SLinus Torvalds 		if ((lp->close != NULL) && (lp->fd >= 0))
8131da177e4SLinus Torvalds 			(*lp->close)(lp->fd, &lp->user);
814c862fc32SJeff Dike 		if (lp->remove != NULL)
815c862fc32SJeff Dike 			(*lp->remove)(&lp->user);
8161da177e4SLinus Torvalds 	}
817c862fc32SJeff Dike 	spin_unlock(&opened_lock);
8181da177e4SLinus Torvalds }
8191da177e4SLinus Torvalds 
8201da177e4SLinus Torvalds __uml_exitcall(close_devices);
8211da177e4SLinus Torvalds 
iter_addresses(void * d,void (* cb)(unsigned char *,unsigned char *,void *),void * arg)8221da177e4SLinus Torvalds void iter_addresses(void *d, void (*cb)(unsigned char *, unsigned char *,
8231da177e4SLinus Torvalds 					void *),
8241da177e4SLinus Torvalds 		    void *arg)
8251da177e4SLinus Torvalds {
8261da177e4SLinus Torvalds 	struct net_device *dev = d;
8271da177e4SLinus Torvalds 	struct in_device *ip = dev->ip_ptr;
8281da177e4SLinus Torvalds 	struct in_ifaddr *in;
8291da177e4SLinus Torvalds 	unsigned char address[4], netmask[4];
8301da177e4SLinus Torvalds 
8311da177e4SLinus Torvalds 	if (ip == NULL) return;
8321da177e4SLinus Torvalds 	in = ip->ifa_list;
8331da177e4SLinus Torvalds 	while (in != NULL) {
8340e76422cSBodo Stroesser 		memcpy(address, &in->ifa_address, sizeof(address));
8350e76422cSBodo Stroesser 		memcpy(netmask, &in->ifa_mask, sizeof(netmask));
8361da177e4SLinus Torvalds 		(*cb)(address, netmask, arg);
8371da177e4SLinus Torvalds 		in = in->ifa_next;
8381da177e4SLinus Torvalds 	}
8391da177e4SLinus Torvalds }
8401da177e4SLinus Torvalds 
dev_netmask(void * d,void * m)8411da177e4SLinus Torvalds int dev_netmask(void *d, void *m)
8421da177e4SLinus Torvalds {
8431da177e4SLinus Torvalds 	struct net_device *dev = d;
8441da177e4SLinus Torvalds 	struct in_device *ip = dev->ip_ptr;
8451da177e4SLinus Torvalds 	struct in_ifaddr *in;
846a144ea4bSAl Viro 	__be32 *mask_out = m;
8471da177e4SLinus Torvalds 
8481da177e4SLinus Torvalds 	if (ip == NULL)
849cd1ae0e4SJeff Dike 		return 1;
8501da177e4SLinus Torvalds 
8511da177e4SLinus Torvalds 	in = ip->ifa_list;
8521da177e4SLinus Torvalds 	if (in == NULL)
853cd1ae0e4SJeff Dike 		return 1;
8541da177e4SLinus Torvalds 
8551da177e4SLinus Torvalds 	*mask_out = in->ifa_mask;
856cd1ae0e4SJeff Dike 	return 0;
8571da177e4SLinus Torvalds }
8581da177e4SLinus Torvalds 
get_output_buffer(int * len_out)8591da177e4SLinus Torvalds void *get_output_buffer(int *len_out)
8601da177e4SLinus Torvalds {
8611da177e4SLinus Torvalds 	void *ret;
8621da177e4SLinus Torvalds 
8631da177e4SLinus Torvalds 	ret = (void *) __get_free_pages(GFP_KERNEL, 0);
8641da177e4SLinus Torvalds 	if (ret) *len_out = PAGE_SIZE;
8651da177e4SLinus Torvalds 	else *len_out = 0;
8664ea21cd9SJeff Dike 	return ret;
8671da177e4SLinus Torvalds }
8681da177e4SLinus Torvalds 
free_output_buffer(void * buffer)8691da177e4SLinus Torvalds void free_output_buffer(void *buffer)
8701da177e4SLinus Torvalds {
8711da177e4SLinus Torvalds 	free_pages((unsigned long) buffer, 0);
8721da177e4SLinus Torvalds }
8731da177e4SLinus Torvalds 
tap_setup_common(char * str,char * type,char ** dev_name,char ** mac_out,char ** gate_addr)8741da177e4SLinus Torvalds int tap_setup_common(char *str, char *type, char **dev_name, char **mac_out,
8751da177e4SLinus Torvalds 		     char **gate_addr)
8761da177e4SLinus Torvalds {
8771da177e4SLinus Torvalds 	char *remain;
8781da177e4SLinus Torvalds 
8791da177e4SLinus Torvalds 	remain = split_if_spec(str, dev_name, mac_out, gate_addr, NULL);
8801da177e4SLinus Torvalds 	if (remain != NULL) {
881cd1ae0e4SJeff Dike 		printk(KERN_ERR "tap_setup_common - Extra garbage on "
882cd1ae0e4SJeff Dike 		       "specification : '%s'\n", remain);
883cd1ae0e4SJeff Dike 		return 1;
8841da177e4SLinus Torvalds 	}
8851da177e4SLinus Torvalds 
886cd1ae0e4SJeff Dike 	return 0;
8871da177e4SLinus Torvalds }
8881da177e4SLinus Torvalds 
eth_protocol(struct sk_buff * skb)8891da177e4SLinus Torvalds unsigned short eth_protocol(struct sk_buff *skb)
8901da177e4SLinus Torvalds {
891cd1ae0e4SJeff Dike 	return eth_type_trans(skb, skb->dev);
8921da177e4SLinus Torvalds }
893