blob: da0403efbc0d3c8110853921c97a01e131679671 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
Jeff Dikecd1ae0e2007-10-16 01:27:29 -07002 * Copyright (C) 2002 - 2007 Jeff Dike (jdike@{addtoit,linux.intel}.com)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003 * Licensed under the GPL.
4 */
5
6#include "linux/init.h"
Jeff Dikecd1ae0e2007-10-16 01:27:29 -07007#include <linux/netdevice.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -07008#include "net_kern.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -07009#include "pcap_user.h"
10
11struct pcap_init {
12 char *host_if;
13 int promisc;
14 int optimize;
15 char *filter;
16};
17
18void pcap_init(struct net_device *dev, void *data)
19{
20 struct uml_net_private *pri;
21 struct pcap_data *ppri;
22 struct pcap_init *init = data;
23
24 pri = dev->priv;
25 ppri = (struct pcap_data *) pri->user;
26 ppri->host_if = init->host_if;
27 ppri->promisc = init->promisc;
28 ppri->optimize = init->optimize;
29 ppri->filter = init->filter;
Jeff Dike7d982302007-05-08 00:35:04 -070030
31 printk("pcap backend, host interface %s\n", ppri->host_if);
Linus Torvalds1da177e2005-04-16 15:20:36 -070032}
33
Jeff Dike7d982302007-05-08 00:35:04 -070034static int pcap_read(int fd, struct sk_buff **skb,
Linus Torvalds1da177e2005-04-16 15:20:36 -070035 struct uml_net_private *lp)
36{
37 *skb = ether_adjust_skb(*skb, ETH_HEADER_OTHER);
Jeff Dikecd1ae0e2007-10-16 01:27:29 -070038 if (*skb == NULL)
Jeff Dike7d982302007-05-08 00:35:04 -070039 return -ENOMEM;
40
41 return pcap_user_read(fd, skb_mac_header(*skb),
Linus Torvalds1da177e2005-04-16 15:20:36 -070042 (*skb)->dev->mtu + ETH_HEADER_OTHER,
Jeff Dike7d982302007-05-08 00:35:04 -070043 (struct pcap_data *) &lp->user);
Linus Torvalds1da177e2005-04-16 15:20:36 -070044}
45
46static int pcap_write(int fd, struct sk_buff **skb, struct uml_net_private *lp)
47{
Jeff Dike7d982302007-05-08 00:35:04 -070048 return -EPERM;
Linus Torvalds1da177e2005-04-16 15:20:36 -070049}
50
Jeff Dike5e7672e2006-09-27 01:50:33 -070051static const struct net_kern_info pcap_kern_info = {
Linus Torvalds1da177e2005-04-16 15:20:36 -070052 .init = pcap_init,
53 .protocol = eth_protocol,
54 .read = pcap_read,
55 .write = pcap_write,
56};
57
58int pcap_setup(char *str, char **mac_out, void *data)
59{
60 struct pcap_init *init = data;
61 char *remain, *host_if = NULL, *options[2] = { NULL, NULL };
62 int i;
63
64 *init = ((struct pcap_init)
65 { .host_if = "eth0",
66 .promisc = 1,
67 .optimize = 0,
68 .filter = NULL });
69
Jeff Dike7d982302007-05-08 00:35:04 -070070 remain = split_if_spec(str, &host_if, &init->filter,
Jeff Dike0ba34e12007-05-08 00:35:06 -070071 &options[0], &options[1], mac_out, NULL);
Jeff Dikecd1ae0e2007-10-16 01:27:29 -070072 if (remain != NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -070073 printk(KERN_ERR "pcap_setup - Extra garbage on "
74 "specification : '%s'\n", remain);
Jeff Dike7d982302007-05-08 00:35:04 -070075 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -070076 }
77
Jeff Dikecd1ae0e2007-10-16 01:27:29 -070078 if (host_if != NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -070079 init->host_if = host_if;
80
Jeff Dikecd1ae0e2007-10-16 01:27:29 -070081 for (i = 0; i < ARRAY_SIZE(options); i++) {
82 if (options[i] == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -070083 continue;
Jeff Dikecd1ae0e2007-10-16 01:27:29 -070084 if (!strcmp(options[i], "promisc"))
Linus Torvalds1da177e2005-04-16 15:20:36 -070085 init->promisc = 1;
Jeff Dikecd1ae0e2007-10-16 01:27:29 -070086 else if (!strcmp(options[i], "nopromisc"))
Linus Torvalds1da177e2005-04-16 15:20:36 -070087 init->promisc = 0;
Jeff Dikecd1ae0e2007-10-16 01:27:29 -070088 else if (!strcmp(options[i], "optimize"))
Linus Torvalds1da177e2005-04-16 15:20:36 -070089 init->optimize = 1;
Jeff Dikecd1ae0e2007-10-16 01:27:29 -070090 else if (!strcmp(options[i], "nooptimize"))
Linus Torvalds1da177e2005-04-16 15:20:36 -070091 init->optimize = 0;
Jeff Dike7d982302007-05-08 00:35:04 -070092 else {
Jeff Dikecd1ae0e2007-10-16 01:27:29 -070093 printk(KERN_ERR "pcap_setup : bad option - '%s'\n",
94 options[i]);
Jeff Dike7d982302007-05-08 00:35:04 -070095 return 0;
96 }
Linus Torvalds1da177e2005-04-16 15:20:36 -070097 }
98
Jeff Dike7d982302007-05-08 00:35:04 -070099 return 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700100}
101
102static struct transport pcap_transport = {
103 .list = LIST_HEAD_INIT(pcap_transport.list),
104 .name = "pcap",
105 .setup = pcap_setup,
106 .user = &pcap_user_info,
107 .kern = &pcap_kern_info,
108 .private_size = sizeof(struct pcap_data),
109 .setup_size = sizeof(struct pcap_init),
110};
111
112static int register_pcap(void)
113{
114 register_transport(&pcap_transport);
Jeff Dikef4c57a72006-03-31 02:30:10 -0800115 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700116}
117
Jeff Dike8210fd22006-12-06 20:34:55 -0800118late_initcall(register_pcap);