Blame view

net/phonet/pn_dev.c 10.2 KB
f8ff60283   Remi Denis-Courmont   Phonet: network d...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
  /*
   * File: pn_dev.c
   *
   * Phonet network device
   *
   * Copyright (C) 2008 Nokia Corporation.
   *
   * Contact: Remi Denis-Courmont <remi.denis-courmont@nokia.com>
   * Original author: Sakari Ailus <sakari.ailus@nokia.com>
   *
   * This program is free software; you can redistribute it and/or
   * modify it under the terms of the GNU General Public License
   * version 2 as published by the Free Software Foundation.
   *
   * This program is distributed in the hope that it will be useful, but
   * WITHOUT ANY WARRANTY; without even the implied warranty of
   * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
   * General Public License for more details.
   *
   * You should have received a copy of the GNU General Public License
   * along with this program; if not, write to the Free Software
   * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
   * 02110-1301 USA
   */
  
  #include <linux/kernel.h>
  #include <linux/net.h>
5a0e3ad6a   Tejun Heo   include cleanup: ...
28
  #include <linux/slab.h>
f8ff60283   Remi Denis-Courmont   Phonet: network d...
29
30
  #include <linux/netdevice.h>
  #include <linux/phonet.h>
421d20a3d   David S. Miller   phonet: Fix build.
31
  #include <linux/proc_fs.h>
f5bb1c558   Rémi Denis-Courmont   Phonet: back-end ...
32
  #include <linux/if_arp.h>
f8ff60283   Remi Denis-Courmont   Phonet: network d...
33
  #include <net/sock.h>
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
34
  #include <net/netns/generic.h>
f8ff60283   Remi Denis-Courmont   Phonet: network d...
35
  #include <net/phonet/pn_dev.h>
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
36
  struct phonet_routes {
888801357   Rémi Denis-Courmont   Phonet: convert r...
37
  	struct mutex		lock;
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
38
39
  	struct net_device	*table[64];
  };
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
40
41
  struct phonet_net {
  	struct phonet_device_list pndevs;
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
42
  	struct phonet_routes routes;
f8ff60283   Remi Denis-Courmont   Phonet: network d...
43
  };
f99189b18   Eric Dumazet   netns: net_identi...
44
  int phonet_net_id __read_mostly;
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
45

0db3f0f49   Jiri Pirko   phonet: use phone...
46
47
48
49
50
51
  static struct phonet_net *phonet_pernet(struct net *net)
  {
  	BUG_ON(!net);
  
  	return net_generic(net, phonet_net_id);
  }
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
52
53
  struct phonet_device_list *phonet_device_list(struct net *net)
  {
0db3f0f49   Jiri Pirko   phonet: use phone...
54
  	struct phonet_net *pnn = phonet_pernet(net);
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
55
56
  	return &pnn->pndevs;
  }
f8ff60283   Remi Denis-Courmont   Phonet: network d...
57
58
59
  /* Allocate new Phonet device. */
  static struct phonet_device *__phonet_device_alloc(struct net_device *dev)
  {
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
60
  	struct phonet_device_list *pndevs = phonet_device_list(dev_net(dev));
f8ff60283   Remi Denis-Courmont   Phonet: network d...
61
62
63
64
65
  	struct phonet_device *pnd = kmalloc(sizeof(*pnd), GFP_ATOMIC);
  	if (pnd == NULL)
  		return NULL;
  	pnd->netdev = dev;
  	bitmap_zero(pnd->addrs, 64);
eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
66
67
  	BUG_ON(!mutex_is_locked(&pndevs->lock));
  	list_add_rcu(&pnd->list, &pndevs->list);
f8ff60283   Remi Denis-Courmont   Phonet: network d...
68
69
70
71
72
  	return pnd;
  }
  
  static struct phonet_device *__phonet_get(struct net_device *dev)
  {
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
73
  	struct phonet_device_list *pndevs = phonet_device_list(dev_net(dev));
f8ff60283   Remi Denis-Courmont   Phonet: network d...
74
  	struct phonet_device *pnd;
eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
75
  	BUG_ON(!mutex_is_locked(&pndevs->lock));
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
76
  	list_for_each_entry(pnd, &pndevs->list, list) {
f8ff60283   Remi Denis-Courmont   Phonet: network d...
77
78
79
80
81
  		if (pnd->netdev == dev)
  			return pnd;
  	}
  	return NULL;
  }
eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
82
83
84
85
86
87
88
89
90
91
92
  static struct phonet_device *__phonet_get_rcu(struct net_device *dev)
  {
  	struct phonet_device_list *pndevs = phonet_device_list(dev_net(dev));
  	struct phonet_device *pnd;
  
  	list_for_each_entry_rcu(pnd, &pndevs->list, list) {
  		if (pnd->netdev == dev)
  			return pnd;
  	}
  	return NULL;
  }
2be6fa4c7   Rémi Denis-Courmont   Phonet: generate ...
93
  static void phonet_device_destroy(struct net_device *dev)
f8ff60283   Remi Denis-Courmont   Phonet: network d...
94
  {
2be6fa4c7   Rémi Denis-Courmont   Phonet: generate ...
95
96
97
98
  	struct phonet_device_list *pndevs = phonet_device_list(dev_net(dev));
  	struct phonet_device *pnd;
  
  	ASSERT_RTNL();
eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
99
  	mutex_lock(&pndevs->lock);
2be6fa4c7   Rémi Denis-Courmont   Phonet: generate ...
100
101
  	pnd = __phonet_get(dev);
  	if (pnd)
eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
102
103
  		list_del_rcu(&pnd->list);
  	mutex_unlock(&pndevs->lock);
2be6fa4c7   Rémi Denis-Courmont   Phonet: generate ...
104
105
106
  
  	if (pnd) {
  		u8 addr;
a1ca14ac5   Akinobu Mita   phonet: use for_e...
107
  		for_each_set_bit(addr, pnd->addrs, 64)
2be6fa4c7   Rémi Denis-Courmont   Phonet: generate ...
108
109
110
  			phonet_address_notify(RTM_DELADDR, dev, addr);
  		kfree(pnd);
  	}
f8ff60283   Remi Denis-Courmont   Phonet: network d...
111
112
113
114
  }
  
  struct net_device *phonet_device_get(struct net *net)
  {
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
115
  	struct phonet_device_list *pndevs = phonet_device_list(net);
f8ff60283   Remi Denis-Courmont   Phonet: network d...
116
  	struct phonet_device *pnd;
59e57f441   Eric Dumazet   phonet: phonet_de...
117
  	struct net_device *dev = NULL;
f8ff60283   Remi Denis-Courmont   Phonet: network d...
118

eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
119
120
  	rcu_read_lock();
  	list_for_each_entry_rcu(pnd, &pndevs->list, list) {
f8ff60283   Remi Denis-Courmont   Phonet: network d...
121
122
  		dev = pnd->netdev;
  		BUG_ON(!dev);
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
123
  		if ((dev->reg_state == NETREG_REGISTERED) &&
f8ff60283   Remi Denis-Courmont   Phonet: network d...
124
125
126
127
128
129
  			((pnd->netdev->flags & IFF_UP)) == IFF_UP)
  			break;
  		dev = NULL;
  	}
  	if (dev)
  		dev_hold(dev);
eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
130
  	rcu_read_unlock();
f8ff60283   Remi Denis-Courmont   Phonet: network d...
131
132
133
134
135
  	return dev;
  }
  
  int phonet_address_add(struct net_device *dev, u8 addr)
  {
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
136
  	struct phonet_device_list *pndevs = phonet_device_list(dev_net(dev));
f8ff60283   Remi Denis-Courmont   Phonet: network d...
137
138
  	struct phonet_device *pnd;
  	int err = 0;
eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
139
  	mutex_lock(&pndevs->lock);
f8ff60283   Remi Denis-Courmont   Phonet: network d...
140
141
142
143
144
145
146
147
  	/* Find or create Phonet-specific device data */
  	pnd = __phonet_get(dev);
  	if (pnd == NULL)
  		pnd = __phonet_device_alloc(dev);
  	if (unlikely(pnd == NULL))
  		err = -ENOMEM;
  	else if (test_and_set_bit(addr >> 2, pnd->addrs))
  		err = -EEXIST;
eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
148
  	mutex_unlock(&pndevs->lock);
f8ff60283   Remi Denis-Courmont   Phonet: network d...
149
150
151
152
153
  	return err;
  }
  
  int phonet_address_del(struct net_device *dev, u8 addr)
  {
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
154
  	struct phonet_device_list *pndevs = phonet_device_list(dev_net(dev));
f8ff60283   Remi Denis-Courmont   Phonet: network d...
155
156
  	struct phonet_device *pnd;
  	int err = 0;
eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
157
  	mutex_lock(&pndevs->lock);
f8ff60283   Remi Denis-Courmont   Phonet: network d...
158
  	pnd = __phonet_get(dev);
eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
159
  	if (!pnd || !test_and_clear_bit(addr >> 2, pnd->addrs)) {
f8ff60283   Remi Denis-Courmont   Phonet: network d...
160
  		err = -EADDRNOTAVAIL;
eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
161
162
163
164
165
166
  		pnd = NULL;
  	} else if (bitmap_empty(pnd->addrs, 64))
  		list_del_rcu(&pnd->list);
  	else
  		pnd = NULL;
  	mutex_unlock(&pndevs->lock);
88e7594a9   Jiri Pirko   phonet: use call_...
167
  	if (pnd)
7e113a9c7   Lai Jiangshan   net,rcu: convert ...
168
  		kfree_rcu(pnd, rcu);
88e7594a9   Jiri Pirko   phonet: use call_...
169

f8ff60283   Remi Denis-Courmont   Phonet: network d...
170
171
172
173
  	return err;
  }
  
  /* Gets a source address toward a destination, through a interface. */
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
174
  u8 phonet_address_get(struct net_device *dev, u8 daddr)
f8ff60283   Remi Denis-Courmont   Phonet: network d...
175
176
  {
  	struct phonet_device *pnd;
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
177
  	u8 saddr;
f8ff60283   Remi Denis-Courmont   Phonet: network d...
178

eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
179
180
  	rcu_read_lock();
  	pnd = __phonet_get_rcu(dev);
f8ff60283   Remi Denis-Courmont   Phonet: network d...
181
182
183
184
  	if (pnd) {
  		BUG_ON(bitmap_empty(pnd->addrs, 64));
  
  		/* Use same source address as destination, if possible */
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
185
186
187
188
  		if (test_bit(daddr >> 2, pnd->addrs))
  			saddr = daddr;
  		else
  			saddr = find_first_bit(pnd->addrs, 64) << 2;
f8ff60283   Remi Denis-Courmont   Phonet: network d...
189
  	} else
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
190
  		saddr = PN_NO_ADDR;
eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
191
  	rcu_read_unlock();
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
192
193
194
195
196
197
198
199
200
201
202
203
204
  
  	if (saddr == PN_NO_ADDR) {
  		/* Fallback to another device */
  		struct net_device *def_dev;
  
  		def_dev = phonet_device_get(dev_net(dev));
  		if (def_dev) {
  			if (def_dev != dev)
  				saddr = phonet_address_get(def_dev, daddr);
  			dev_put(def_dev);
  		}
  	}
  	return saddr;
f8ff60283   Remi Denis-Courmont   Phonet: network d...
205
  }
524048819   Rémi Denis-Courmont   Phonet: basic net...
206
  int phonet_address_lookup(struct net *net, u8 addr)
f8ff60283   Remi Denis-Courmont   Phonet: network d...
207
  {
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
208
  	struct phonet_device_list *pndevs = phonet_device_list(net);
f8ff60283   Remi Denis-Courmont   Phonet: network d...
209
  	struct phonet_device *pnd;
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
210
  	int err = -EADDRNOTAVAIL;
f8ff60283   Remi Denis-Courmont   Phonet: network d...
211

eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
212
213
  	rcu_read_lock();
  	list_for_each_entry_rcu(pnd, &pndevs->list, list) {
f8ff60283   Remi Denis-Courmont   Phonet: network d...
214
215
216
217
218
219
  		/* Don't allow unregistering devices! */
  		if ((pnd->netdev->reg_state != NETREG_REGISTERED) ||
  				((pnd->netdev->flags & IFF_UP)) != IFF_UP)
  			continue;
  
  		if (test_bit(addr >> 2, pnd->addrs)) {
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
220
221
  			err = 0;
  			goto found;
f8ff60283   Remi Denis-Courmont   Phonet: network d...
222
223
  		}
  	}
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
224
  found:
eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
225
  	rcu_read_unlock();
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
226
  	return err;
f8ff60283   Remi Denis-Courmont   Phonet: network d...
227
  }
f5bb1c558   Rémi Denis-Courmont   Phonet: back-end ...
228
229
230
231
232
233
234
235
236
237
238
239
240
  /* automatically configure a Phonet device, if supported */
  static int phonet_device_autoconf(struct net_device *dev)
  {
  	struct if_phonet_req req;
  	int ret;
  
  	if (!dev->netdev_ops->ndo_do_ioctl)
  		return -EOPNOTSUPP;
  
  	ret = dev->netdev_ops->ndo_do_ioctl(dev, (struct ifreq *)&req,
  						SIOCPNGAUTOCONF);
  	if (ret < 0)
  		return ret;
b11b5165a   Rémi Denis-Courmont   Phonet: Netlink e...
241
242
243
244
245
246
247
248
  
  	ASSERT_RTNL();
  	ret = phonet_address_add(dev, req.ifr_phonet_autoconf.device);
  	if (ret)
  		return ret;
  	phonet_address_notify(RTM_NEWADDR, dev,
  				req.ifr_phonet_autoconf.device);
  	return 0;
f5bb1c558   Rémi Denis-Courmont   Phonet: back-end ...
249
  }
f062f41d0   Rémi Denis-Courmont   Phonet: routing t...
250
251
  static void phonet_route_autodel(struct net_device *dev)
  {
0db3f0f49   Jiri Pirko   phonet: use phone...
252
  	struct phonet_net *pnn = phonet_pernet(dev_net(dev));
f062f41d0   Rémi Denis-Courmont   Phonet: routing t...
253
254
255
256
257
  	unsigned i;
  	DECLARE_BITMAP(deleted, 64);
  
  	/* Remove left-over Phonet routes */
  	bitmap_zero(deleted, 64);
888801357   Rémi Denis-Courmont   Phonet: convert r...
258
  	mutex_lock(&pnn->routes.lock);
f062f41d0   Rémi Denis-Courmont   Phonet: routing t...
259
260
  	for (i = 0; i < 64; i++)
  		if (dev == pnn->routes.table[i]) {
a9b3cd7f3   Stephen Hemminger   rcu: convert uses...
261
  			RCU_INIT_POINTER(pnn->routes.table[i], NULL);
f062f41d0   Rémi Denis-Courmont   Phonet: routing t...
262
  			set_bit(i, deleted);
f062f41d0   Rémi Denis-Courmont   Phonet: routing t...
263
  		}
888801357   Rémi Denis-Courmont   Phonet: convert r...
264
265
266
267
268
  	mutex_unlock(&pnn->routes.lock);
  
  	if (bitmap_empty(deleted, 64))
  		return; /* short-circuit RCU */
  	synchronize_rcu();
6a499b242   Akinobu Mita   phonet: use for_e...
269
  	for_each_set_bit(i, deleted, 64) {
f062f41d0   Rémi Denis-Courmont   Phonet: routing t...
270
  		rtm_phonet_notify(RTM_DELROUTE, dev, i);
888801357   Rémi Denis-Courmont   Phonet: convert r...
271
272
  		dev_put(dev);
  	}
f062f41d0   Rémi Denis-Courmont   Phonet: routing t...
273
  }
f8ff60283   Remi Denis-Courmont   Phonet: network d...
274
275
276
277
278
  /* notify Phonet of device events */
  static int phonet_device_notify(struct notifier_block *me, unsigned long what,
  				void *arg)
  {
  	struct net_device *dev = arg;
f5bb1c558   Rémi Denis-Courmont   Phonet: back-end ...
279
280
281
282
283
284
  	switch (what) {
  	case NETDEV_REGISTER:
  		if (dev->type == ARPHRD_PHONET)
  			phonet_device_autoconf(dev);
  		break;
  	case NETDEV_UNREGISTER:
2be6fa4c7   Rémi Denis-Courmont   Phonet: generate ...
285
  		phonet_device_destroy(dev);
f062f41d0   Rémi Denis-Courmont   Phonet: routing t...
286
  		phonet_route_autodel(dev);
f5bb1c558   Rémi Denis-Courmont   Phonet: back-end ...
287
288
  		break;
  	}
f8ff60283   Remi Denis-Courmont   Phonet: network d...
289
290
291
292
293
294
295
296
  	return 0;
  
  }
  
  static struct notifier_block phonet_device_notifier = {
  	.notifier_call = phonet_device_notify,
  	.priority = 0,
  };
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
297
  /* Per-namespace Phonet devices handling */
2c8c1e729   Alexey Dobriyan   net: spread __net...
298
  static int __net_init phonet_init_net(struct net *net)
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
299
  {
0db3f0f49   Jiri Pirko   phonet: use phone...
300
  	struct phonet_net *pnn = phonet_pernet(net);
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
301

d2b3eb630   Eric W. Biederman   net: Simplify pho...
302
  	if (!proc_net_fops_create(net, "phonet", 0, &pn_sock_seq_fops))
c1dc13e9d   Rémi Denis-Courmont   Phonet: sockets l...
303
  		return -ENOMEM;
c1dc13e9d   Rémi Denis-Courmont   Phonet: sockets l...
304

9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
305
  	INIT_LIST_HEAD(&pnn->pndevs.list);
eeb74a9d4   Rémi Denis-Courmont   Phonet: convert d...
306
  	mutex_init(&pnn->pndevs.lock);
888801357   Rémi Denis-Courmont   Phonet: convert r...
307
  	mutex_init(&pnn->routes.lock);
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
308
309
  	return 0;
  }
2c8c1e729   Alexey Dobriyan   net: spread __net...
310
  static void __net_exit phonet_exit_net(struct net *net)
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
311
  {
0db3f0f49   Jiri Pirko   phonet: use phone...
312
  	struct phonet_net *pnn = phonet_pernet(net);
2be6fa4c7   Rémi Denis-Courmont   Phonet: generate ...
313
  	struct net_device *dev;
f062f41d0   Rémi Denis-Courmont   Phonet: routing t...
314
  	unsigned i;
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
315

2be6fa4c7   Rémi Denis-Courmont   Phonet: generate ...
316
317
318
  	rtnl_lock();
  	for_each_netdev(net, dev)
  		phonet_device_destroy(dev);
f062f41d0   Rémi Denis-Courmont   Phonet: routing t...
319
320
321
322
323
324
325
326
  
  	for (i = 0; i < 64; i++) {
  		dev = pnn->routes.table[i];
  		if (dev) {
  			rtm_phonet_notify(RTM_DELROUTE, dev, i);
  			dev_put(dev);
  		}
  	}
2be6fa4c7   Rémi Denis-Courmont   Phonet: generate ...
327
  	rtnl_unlock();
c1dc13e9d   Rémi Denis-Courmont   Phonet: sockets l...
328
329
  
  	proc_net_remove(net, "phonet");
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
330
331
332
333
334
  }
  
  static struct pernet_operations phonet_net_ops = {
  	.init = phonet_init_net,
  	.exit = phonet_exit_net,
d2b3eb630   Eric W. Biederman   net: Simplify pho...
335
336
  	.id   = &phonet_net_id,
  	.size = sizeof(struct phonet_net),
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
337
  };
f8ff60283   Remi Denis-Courmont   Phonet: network d...
338
  /* Initialize Phonet devices list */
76e02cf69   remi.denis-courmont@nokia   Phonet: allow pho...
339
  int __init phonet_device_init(void)
f8ff60283   Remi Denis-Courmont   Phonet: network d...
340
  {
d2b3eb630   Eric W. Biederman   net: Simplify pho...
341
  	int err = register_pernet_device(&phonet_net_ops);
9a3b7a42b   remi.denis-courmont@nokia   Phonet: use per-n...
342
343
  	if (err)
  		return err;
660f706d9   remi.denis-courmont@nokia   Phonet: handle rt...
344

507215f8d   Rémi Denis-Courmont   Phonet: list subs...
345
  	proc_net_fops_create(&init_net, "pnresource", 0, &pn_res_seq_fops);
f8ff60283   Remi Denis-Courmont   Phonet: network d...
346
  	register_netdevice_notifier(&phonet_device_notifier);
660f706d9   remi.denis-courmont@nokia   Phonet: handle rt...
347
348
349
350
  	err = phonet_netlink_register();
  	if (err)
  		phonet_device_exit();
  	return err;
f8ff60283   Remi Denis-Courmont   Phonet: network d...
351
352
353
354
  }
  
  void phonet_device_exit(void)
  {
f8ff60283   Remi Denis-Courmont   Phonet: network d...
355
  	rtnl_unregister_all(PF_PHONET);
6530e0fee   remi.denis-courmont@nokia   Phonet: remove us...
356
  	unregister_netdevice_notifier(&phonet_device_notifier);
d2b3eb630   Eric W. Biederman   net: Simplify pho...
357
  	unregister_pernet_device(&phonet_net_ops);
507215f8d   Rémi Denis-Courmont   Phonet: list subs...
358
  	proc_net_remove(&init_net, "pnresource");
f8ff60283   Remi Denis-Courmont   Phonet: network d...
359
  }
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
360
361
362
  
  int phonet_route_add(struct net_device *dev, u8 daddr)
  {
0db3f0f49   Jiri Pirko   phonet: use phone...
363
  	struct phonet_net *pnn = phonet_pernet(dev_net(dev));
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
364
365
366
367
  	struct phonet_routes *routes = &pnn->routes;
  	int err = -EEXIST;
  
  	daddr = daddr >> 2;
888801357   Rémi Denis-Courmont   Phonet: convert r...
368
  	mutex_lock(&routes->lock);
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
369
  	if (routes->table[daddr] == NULL) {
cf778b00e   Eric Dumazet   net: reintroduce ...
370
  		rcu_assign_pointer(routes->table[daddr], dev);
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
371
372
373
  		dev_hold(dev);
  		err = 0;
  	}
888801357   Rémi Denis-Courmont   Phonet: convert r...
374
  	mutex_unlock(&routes->lock);
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
375
376
377
378
379
  	return err;
  }
  
  int phonet_route_del(struct net_device *dev, u8 daddr)
  {
0db3f0f49   Jiri Pirko   phonet: use phone...
380
  	struct phonet_net *pnn = phonet_pernet(dev_net(dev));
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
381
  	struct phonet_routes *routes = &pnn->routes;
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
382
383
  
  	daddr = daddr >> 2;
888801357   Rémi Denis-Courmont   Phonet: convert r...
384
385
  	mutex_lock(&routes->lock);
  	if (dev == routes->table[daddr])
a9b3cd7f3   Stephen Hemminger   rcu: convert uses...
386
  		RCU_INIT_POINTER(routes->table[daddr], NULL);
888801357   Rémi Denis-Courmont   Phonet: convert r...
387
388
389
390
391
392
393
394
395
  	else
  		dev = NULL;
  	mutex_unlock(&routes->lock);
  
  	if (!dev)
  		return -ENOENT;
  	synchronize_rcu();
  	dev_put(dev);
  	return 0;
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
396
  }
e67f88dd1   Eric Dumazet   net: dont hold rt...
397
  struct net_device *phonet_route_get_rcu(struct net *net, u8 daddr)
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
398
  {
0db3f0f49   Jiri Pirko   phonet: use phone...
399
  	struct phonet_net *pnn = phonet_pernet(net);
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
400
401
  	struct phonet_routes *routes = &pnn->routes;
  	struct net_device *dev;
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
402
  	daddr >>= 2;
888801357   Rémi Denis-Courmont   Phonet: convert r...
403
  	dev = rcu_dereference(routes->table[daddr]);
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
404
405
406
407
408
  	return dev;
  }
  
  struct net_device *phonet_route_output(struct net *net, u8 daddr)
  {
0db3f0f49   Jiri Pirko   phonet: use phone...
409
  	struct phonet_net *pnn = phonet_pernet(net);
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
410
411
  	struct phonet_routes *routes = &pnn->routes;
  	struct net_device *dev;
888801357   Rémi Denis-Courmont   Phonet: convert r...
412
413
414
  	daddr >>= 2;
  	rcu_read_lock();
  	dev = rcu_dereference(routes->table[daddr]);
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
415
416
  	if (dev)
  		dev_hold(dev);
888801357   Rémi Denis-Courmont   Phonet: convert r...
417
  	rcu_read_unlock();
55748ac04   Rémi Denis-Courmont   Phonet: routing t...
418
419
420
421
422
  
  	if (!dev)
  		dev = phonet_device_get(net); /* Default route */
  	return dev;
  }