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)207857710e1SNathan 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 {
339*cdbd5a1dSTiwei Bie struct uml_net *device = container_of(dev, struct uml_net, pdev.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(¨_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 = ¨_netdev_ops;
457f34d9d2dSJeff Dike dev->ethtool_ops = ¨_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, ð_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(ð->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, ð_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(¨_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