diff options
Diffstat (limited to 'net/l2tp/l2tp_eth.c')
-rw-r--r-- | net/l2tp/l2tp_eth.c | 347 |
1 files changed, 347 insertions, 0 deletions
diff --git a/net/l2tp/l2tp_eth.c b/net/l2tp/l2tp_eth.c new file mode 100644 index 000000000000..755c29729b6f --- /dev/null +++ b/net/l2tp/l2tp_eth.c | |||
@@ -0,0 +1,347 @@ | |||
1 | /* | ||
2 | * L2TPv3 ethernet pseudowire driver | ||
3 | * | ||
4 | * Copyright (c) 2008,2009,2010 Katalix Systems Ltd | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or | ||
7 | * modify it under the terms of the GNU General Public License | ||
8 | * as published by the Free Software Foundation; either version | ||
9 | * 2 of the License, or (at your option) any later version. | ||
10 | */ | ||
11 | |||
12 | #include <linux/module.h> | ||
13 | #include <linux/skbuff.h> | ||
14 | #include <linux/socket.h> | ||
15 | #include <linux/hash.h> | ||
16 | #include <linux/l2tp.h> | ||
17 | #include <linux/in.h> | ||
18 | #include <linux/etherdevice.h> | ||
19 | #include <linux/spinlock.h> | ||
20 | #include <net/sock.h> | ||
21 | #include <net/ip.h> | ||
22 | #include <net/icmp.h> | ||
23 | #include <net/udp.h> | ||
24 | #include <net/inet_common.h> | ||
25 | #include <net/inet_hashtables.h> | ||
26 | #include <net/tcp_states.h> | ||
27 | #include <net/protocol.h> | ||
28 | #include <net/xfrm.h> | ||
29 | #include <net/net_namespace.h> | ||
30 | #include <net/netns/generic.h> | ||
31 | |||
32 | #include "l2tp_core.h" | ||
33 | |||
34 | /* Default device name. May be overridden by name specified by user */ | ||
35 | #define L2TP_ETH_DEV_NAME "l2tpeth%d" | ||
36 | |||
37 | /* via netdev_priv() */ | ||
38 | struct l2tp_eth { | ||
39 | struct net_device *dev; | ||
40 | struct sock *tunnel_sock; | ||
41 | struct l2tp_session *session; | ||
42 | struct list_head list; | ||
43 | }; | ||
44 | |||
45 | /* via l2tp_session_priv() */ | ||
46 | struct l2tp_eth_sess { | ||
47 | struct net_device *dev; | ||
48 | }; | ||
49 | |||
50 | /* per-net private data for this module */ | ||
51 | static unsigned int l2tp_eth_net_id; | ||
52 | struct l2tp_eth_net { | ||
53 | struct list_head l2tp_eth_dev_list; | ||
54 | spinlock_t l2tp_eth_lock; | ||
55 | }; | ||
56 | |||
57 | static inline struct l2tp_eth_net *l2tp_eth_pernet(struct net *net) | ||
58 | { | ||
59 | return net_generic(net, l2tp_eth_net_id); | ||
60 | } | ||
61 | |||
62 | static int l2tp_eth_dev_init(struct net_device *dev) | ||
63 | { | ||
64 | struct l2tp_eth *priv = netdev_priv(dev); | ||
65 | |||
66 | priv->dev = dev; | ||
67 | random_ether_addr(dev->dev_addr); | ||
68 | memset(&dev->broadcast[0], 0xff, 6); | ||
69 | |||
70 | return 0; | ||
71 | } | ||
72 | |||
73 | static void l2tp_eth_dev_uninit(struct net_device *dev) | ||
74 | { | ||
75 | struct l2tp_eth *priv = netdev_priv(dev); | ||
76 | struct l2tp_eth_net *pn = l2tp_eth_pernet(dev_net(dev)); | ||
77 | |||
78 | spin_lock(&pn->l2tp_eth_lock); | ||
79 | list_del_init(&priv->list); | ||
80 | spin_unlock(&pn->l2tp_eth_lock); | ||
81 | dev_put(dev); | ||
82 | } | ||
83 | |||
84 | static int l2tp_eth_dev_xmit(struct sk_buff *skb, struct net_device *dev) | ||
85 | { | ||
86 | struct l2tp_eth *priv = netdev_priv(dev); | ||
87 | struct l2tp_session *session = priv->session; | ||
88 | |||
89 | l2tp_xmit_skb(session, skb, session->hdr_len); | ||
90 | |||
91 | dev->stats.tx_bytes += skb->len; | ||
92 | dev->stats.tx_packets++; | ||
93 | |||
94 | return 0; | ||
95 | } | ||
96 | |||
97 | static struct net_device_ops l2tp_eth_netdev_ops = { | ||
98 | .ndo_init = l2tp_eth_dev_init, | ||
99 | .ndo_uninit = l2tp_eth_dev_uninit, | ||
100 | .ndo_start_xmit = l2tp_eth_dev_xmit, | ||
101 | }; | ||
102 | |||
103 | static void l2tp_eth_dev_setup(struct net_device *dev) | ||
104 | { | ||
105 | ether_setup(dev); | ||
106 | |||
107 | dev->netdev_ops = &l2tp_eth_netdev_ops; | ||
108 | dev->destructor = free_netdev; | ||
109 | } | ||
110 | |||
111 | static void l2tp_eth_dev_recv(struct l2tp_session *session, struct sk_buff *skb, int data_len) | ||
112 | { | ||
113 | struct l2tp_eth_sess *spriv = l2tp_session_priv(session); | ||
114 | struct net_device *dev = spriv->dev; | ||
115 | |||
116 | if (session->debug & L2TP_MSG_DATA) { | ||
117 | unsigned int length; | ||
118 | int offset; | ||
119 | u8 *ptr = skb->data; | ||
120 | |||
121 | length = min(32u, skb->len); | ||
122 | if (!pskb_may_pull(skb, length)) | ||
123 | goto error; | ||
124 | |||
125 | printk(KERN_DEBUG "%s: eth recv: ", session->name); | ||
126 | |||
127 | offset = 0; | ||
128 | do { | ||
129 | printk(" %02X", ptr[offset]); | ||
130 | } while (++offset < length); | ||
131 | |||
132 | printk("\n"); | ||
133 | } | ||
134 | |||
135 | if (data_len < ETH_HLEN) | ||
136 | goto error; | ||
137 | |||
138 | secpath_reset(skb); | ||
139 | |||
140 | /* checksums verified by L2TP */ | ||
141 | skb->ip_summed = CHECKSUM_NONE; | ||
142 | |||
143 | skb_dst_drop(skb); | ||
144 | nf_reset(skb); | ||
145 | |||
146 | if (dev_forward_skb(dev, skb) == NET_RX_SUCCESS) { | ||
147 | dev->last_rx = jiffies; | ||
148 | dev->stats.rx_packets++; | ||
149 | dev->stats.rx_bytes += data_len; | ||
150 | } else | ||
151 | dev->stats.rx_errors++; | ||
152 | |||
153 | return; | ||
154 | |||
155 | error: | ||
156 | dev->stats.rx_errors++; | ||
157 | kfree_skb(skb); | ||
158 | } | ||
159 | |||
160 | static void l2tp_eth_delete(struct l2tp_session *session) | ||
161 | { | ||
162 | struct l2tp_eth_sess *spriv; | ||
163 | struct net_device *dev; | ||
164 | |||
165 | if (session) { | ||
166 | spriv = l2tp_session_priv(session); | ||
167 | dev = spriv->dev; | ||
168 | if (dev) { | ||
169 | unregister_netdev(dev); | ||
170 | spriv->dev = NULL; | ||
171 | } | ||
172 | } | ||
173 | } | ||
174 | |||
175 | static int l2tp_eth_create(struct net *net, u32 tunnel_id, u32 session_id, u32 peer_session_id, struct l2tp_session_cfg *cfg) | ||
176 | { | ||
177 | struct net_device *dev; | ||
178 | char name[IFNAMSIZ]; | ||
179 | struct l2tp_tunnel *tunnel; | ||
180 | struct l2tp_session *session; | ||
181 | struct l2tp_eth *priv; | ||
182 | struct l2tp_eth_sess *spriv; | ||
183 | int rc; | ||
184 | struct l2tp_eth_net *pn; | ||
185 | |||
186 | tunnel = l2tp_tunnel_find(net, tunnel_id); | ||
187 | if (!tunnel) { | ||
188 | rc = -ENODEV; | ||
189 | goto out; | ||
190 | } | ||
191 | |||
192 | session = l2tp_session_find(net, tunnel, session_id); | ||
193 | if (session) { | ||
194 | rc = -EEXIST; | ||
195 | goto out; | ||
196 | } | ||
197 | |||
198 | if (cfg->ifname) { | ||
199 | dev = dev_get_by_name(net, cfg->ifname); | ||
200 | if (dev) { | ||
201 | dev_put(dev); | ||
202 | rc = -EEXIST; | ||
203 | goto out; | ||
204 | } | ||
205 | strlcpy(name, cfg->ifname, IFNAMSIZ); | ||
206 | } else | ||
207 | strcpy(name, L2TP_ETH_DEV_NAME); | ||
208 | |||
209 | session = l2tp_session_create(sizeof(*spriv), tunnel, session_id, | ||
210 | peer_session_id, cfg); | ||
211 | if (!session) { | ||
212 | rc = -ENOMEM; | ||
213 | goto out; | ||
214 | } | ||
215 | |||
216 | dev = alloc_netdev(sizeof(*priv), name, l2tp_eth_dev_setup); | ||
217 | if (!dev) { | ||
218 | rc = -ENOMEM; | ||
219 | goto out_del_session; | ||
220 | } | ||
221 | |||
222 | dev_net_set(dev, net); | ||
223 | if (session->mtu == 0) | ||
224 | session->mtu = dev->mtu - session->hdr_len; | ||
225 | dev->mtu = session->mtu; | ||
226 | dev->needed_headroom += session->hdr_len; | ||
227 | |||
228 | priv = netdev_priv(dev); | ||
229 | priv->dev = dev; | ||
230 | priv->session = session; | ||
231 | INIT_LIST_HEAD(&priv->list); | ||
232 | |||
233 | priv->tunnel_sock = tunnel->sock; | ||
234 | session->recv_skb = l2tp_eth_dev_recv; | ||
235 | session->session_close = l2tp_eth_delete; | ||
236 | |||
237 | spriv = l2tp_session_priv(session); | ||
238 | spriv->dev = dev; | ||
239 | |||
240 | rc = register_netdev(dev); | ||
241 | if (rc < 0) | ||
242 | goto out_del_dev; | ||
243 | |||
244 | /* Must be done after register_netdev() */ | ||
245 | strlcpy(session->ifname, dev->name, IFNAMSIZ); | ||
246 | |||
247 | dev_hold(dev); | ||
248 | pn = l2tp_eth_pernet(dev_net(dev)); | ||
249 | spin_lock(&pn->l2tp_eth_lock); | ||
250 | list_add(&priv->list, &pn->l2tp_eth_dev_list); | ||
251 | spin_unlock(&pn->l2tp_eth_lock); | ||
252 | |||
253 | return 0; | ||
254 | |||
255 | out_del_dev: | ||
256 | free_netdev(dev); | ||
257 | out_del_session: | ||
258 | l2tp_session_delete(session); | ||
259 | out: | ||
260 | return rc; | ||
261 | } | ||
262 | |||
263 | static __net_init int l2tp_eth_init_net(struct net *net) | ||
264 | { | ||
265 | struct l2tp_eth_net *pn; | ||
266 | int err; | ||
267 | |||
268 | pn = kzalloc(sizeof(*pn), GFP_KERNEL); | ||
269 | if (!pn) | ||
270 | return -ENOMEM; | ||
271 | |||
272 | INIT_LIST_HEAD(&pn->l2tp_eth_dev_list); | ||
273 | spin_lock_init(&pn->l2tp_eth_lock); | ||
274 | |||
275 | err = net_assign_generic(net, l2tp_eth_net_id, pn); | ||
276 | if (err) | ||
277 | goto out; | ||
278 | |||
279 | return 0; | ||
280 | |||
281 | out: | ||
282 | kfree(pn); | ||
283 | return err; | ||
284 | } | ||
285 | |||
286 | static __net_exit void l2tp_eth_exit_net(struct net *net) | ||
287 | { | ||
288 | struct l2tp_eth_net *pn; | ||
289 | |||
290 | pn = net_generic(net, l2tp_eth_net_id); | ||
291 | /* | ||
292 | * if someone has cached our net then | ||
293 | * further net_generic call will return NULL | ||
294 | */ | ||
295 | net_assign_generic(net, l2tp_eth_net_id, NULL); | ||
296 | kfree(pn); | ||
297 | } | ||
298 | |||
299 | static __net_initdata struct pernet_operations l2tp_eth_net_ops = { | ||
300 | .init = l2tp_eth_init_net, | ||
301 | .exit = l2tp_eth_exit_net, | ||
302 | .id = &l2tp_eth_net_id, | ||
303 | .size = sizeof(struct l2tp_eth_net), | ||
304 | }; | ||
305 | |||
306 | |||
307 | static const struct l2tp_nl_cmd_ops l2tp_eth_nl_cmd_ops = { | ||
308 | .session_create = l2tp_eth_create, | ||
309 | .session_delete = l2tp_session_delete, | ||
310 | }; | ||
311 | |||
312 | |||
313 | static int __init l2tp_eth_init(void) | ||
314 | { | ||
315 | int err = 0; | ||
316 | |||
317 | err = l2tp_nl_register_ops(L2TP_PWTYPE_ETH, &l2tp_eth_nl_cmd_ops); | ||
318 | if (err) | ||
319 | goto out; | ||
320 | |||
321 | err = register_pernet_device(&l2tp_eth_net_ops); | ||
322 | if (err) | ||
323 | goto out_unreg; | ||
324 | |||
325 | printk(KERN_INFO "L2TP ethernet pseudowire support (L2TPv3)\n"); | ||
326 | |||
327 | return 0; | ||
328 | |||
329 | out_unreg: | ||
330 | l2tp_nl_unregister_ops(L2TP_PWTYPE_ETH); | ||
331 | out: | ||
332 | return err; | ||
333 | } | ||
334 | |||
335 | static void __exit l2tp_eth_exit(void) | ||
336 | { | ||
337 | unregister_pernet_device(&l2tp_eth_net_ops); | ||
338 | l2tp_nl_unregister_ops(L2TP_PWTYPE_ETH); | ||
339 | } | ||
340 | |||
341 | module_init(l2tp_eth_init); | ||
342 | module_exit(l2tp_eth_exit); | ||
343 | |||
344 | MODULE_LICENSE("GPL"); | ||
345 | MODULE_AUTHOR("James Chapman <jchapman@katalix.com>"); | ||
346 | MODULE_DESCRIPTION("L2TP ethernet pseudowire driver"); | ||
347 | MODULE_VERSION("1.0"); | ||