This source file includes following definitions.
- loopback_xmit
- get_stats
- loopback_open
- loopback_init
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 #include <linux/config.h>
27 #include <linux/kernel.h>
28 #include <linux/sched.h>
29 #include <linux/interrupt.h>
30 #include <linux/fs.h>
31 #include <linux/types.h>
32 #include <linux/string.h>
33 #include <linux/socket.h>
34 #include <linux/errno.h>
35 #include <linux/fcntl.h>
36 #include <linux/in.h>
37
38 #include <asm/system.h>
39 #include <asm/segment.h>
40 #include <asm/io.h>
41
42 #include <linux/inet.h>
43 #include <linux/netdevice.h>
44 #include <linux/etherdevice.h>
45 #include <linux/skbuff.h>
46 #include <net/sock.h>
47 #include <linux/if_ether.h>
48 #include <linux/if_arp.h>
49
50 #define LOOPBACK_MTU (PAGE_SIZE*7/8)
51
52
53
54
55
56 static int loopback_xmit(struct sk_buff *skb, struct device *dev)
57 {
58 struct enet_statistics *stats = (struct enet_statistics *)dev->priv;
59 int unlock=1;
60
61 if (skb == NULL || dev == NULL)
62 return(0);
63
64
65
66
67
68
69 if(skb->free==0)
70 {
71 struct sk_buff *skb2=skb;
72 skb=skb_clone(skb, GFP_ATOMIC);
73 if(skb==NULL)
74 return 1;
75 dev_kfree_skb(skb2, FREE_WRITE);
76 unlock=0;
77 }
78 else if(skb->sk)
79 {
80
81
82
83
84 atomic_sub(skb->truesize, &skb->sk->wmem_alloc);
85 skb->sk->write_space(skb->sk);
86 }
87
88 skb->protocol=eth_type_trans(skb,dev);
89 skb->dev=dev;
90 #ifndef LOOPBACK_MUST_CHECKSUM
91 skb->ip_summed = CHECKSUM_UNNECESSARY;
92 #endif
93 netif_rx(skb);
94 if(unlock)
95 skb_device_unlock(skb);
96
97 stats->rx_packets++;
98 stats->tx_packets++;
99
100 return(0);
101 }
102
103 static struct enet_statistics *get_stats(struct device *dev)
104 {
105 return (struct enet_statistics *)dev->priv;
106 }
107
108 static int loopback_open(struct device *dev)
109 {
110 dev->flags|=IFF_LOOPBACK;
111 return 0;
112 }
113
114
115 int loopback_init(struct device *dev)
116 {
117 int i;
118
119 dev->mtu = LOOPBACK_MTU;
120 dev->tbusy = 0;
121 dev->hard_start_xmit = loopback_xmit;
122 dev->open = NULL;
123 dev->hard_header = eth_header;
124 dev->hard_header_len = ETH_HLEN;
125 dev->addr_len = ETH_ALEN;
126 dev->tx_queue_len = 50000;
127 dev->type = ARPHRD_LOOPBACK;
128 dev->rebuild_header = eth_rebuild_header;
129 dev->open = loopback_open;
130 dev->flags = IFF_LOOPBACK|IFF_BROADCAST;
131 dev->family = AF_INET;
132 #ifdef CONFIG_INET
133 dev->pa_addr = in_aton("127.0.0.1");
134 dev->pa_brdaddr = in_aton("127.255.255.255");
135 dev->pa_mask = in_aton("255.0.0.0");
136 dev->pa_alen = 4;
137 #endif
138 dev->priv = kmalloc(sizeof(struct enet_statistics), GFP_KERNEL);
139 if (dev->priv == NULL)
140 return -ENOMEM;
141 memset(dev->priv, 0, sizeof(struct enet_statistics));
142 dev->get_stats = get_stats;
143
144
145
146
147
148 for (i = 0; i < DEV_NUMBUFFS; i++)
149 skb_queue_head_init(&dev->buffs[i]);
150
151 return(0);
152 };