source: azure_iot_hub_f767zi/trunk/asp_baseplatform/lwip/contrib-2.1.0/ports/unix/port/netif/tapif.c@ 457

Last change on this file since 457 was 457, checked in by coas-nagasima, 4 years ago

ファイルを追加

  • Property svn:eol-style set to native
  • Property svn:mime-type set to text/x-csrc;charset=UTF-8
File size: 11.6 KB
Line 
1/*
2 * Copyright (c) 2001-2003 Swedish Institute of Computer Science.
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without modification,
6 * are permitted provided that the following conditions are met:
7 *
8 * 1. Redistributions of source code must retain the above copyright notice,
9 * this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright notice,
11 * this list of conditions and the following disclaimer in the documentation
12 * and/or other materials provided with the distribution.
13 * 3. The name of the author may not be used to endorse or promote products
14 * derived from this software without specific prior written permission.
15 *
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
17 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
18 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT
19 * SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
20 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
21 * OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
22 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
23 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
24 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY
25 * OF SUCH DAMAGE.
26 *
27 * This file is part of the lwIP TCP/IP stack.
28 *
29 * Author: Adam Dunkels <adam@sics.se>
30 *
31 */
32
33#include <fcntl.h>
34#include <stdlib.h>
35#include <stdio.h>
36#include <unistd.h>
37#include <string.h>
38#include <sys/ioctl.h>
39#include <sys/socket.h>
40#include <sys/types.h>
41#include <sys/time.h>
42#include <sys/uio.h>
43#include <sys/socket.h>
44
45#include "lwip/opt.h"
46
47#include "lwip/debug.h"
48#include "lwip/def.h"
49#include "lwip/ip.h"
50#include "lwip/mem.h"
51#include "lwip/stats.h"
52#include "lwip/snmp.h"
53#include "lwip/pbuf.h"
54#include "lwip/sys.h"
55#include "lwip/timeouts.h"
56#include "netif/etharp.h"
57#include "lwip/ethip6.h"
58
59#include "netif/tapif.h"
60
61#define IFCONFIG_BIN "/sbin/ifconfig "
62
63#if defined(LWIP_UNIX_LINUX)
64#include <sys/ioctl.h>
65#include <linux/if.h>
66#include <linux/if_tun.h>
67/*
68 * Creating a tap interface requires special privileges. If the interfaces
69 * is created in advance with `tunctl -u <user>` it can be opened as a regular
70 * user. The network must already be configured. If DEVTAP_IF is defined it
71 * will be opened instead of creating a new tap device.
72 *
73 * You can also use PRECONFIGURED_TAPIF environment variable to do so.
74 */
75#ifndef DEVTAP_DEFAULT_IF
76#define DEVTAP_DEFAULT_IF "tap0"
77#endif
78#ifndef DEVTAP
79#define DEVTAP "/dev/net/tun"
80#endif
81#define NETMASK_ARGS "netmask %d.%d.%d.%d"
82#define IFCONFIG_ARGS "tap0 inet %d.%d.%d.%d " NETMASK_ARGS
83#elif defined(LWIP_UNIX_OPENBSD)
84#define DEVTAP "/dev/tun0"
85#define NETMASK_ARGS "netmask %d.%d.%d.%d"
86#define IFCONFIG_ARGS "tun0 inet %d.%d.%d.%d " NETMASK_ARGS " link0"
87#else /* others */
88#define DEVTAP "/dev/tap0"
89#define NETMASK_ARGS "netmask %d.%d.%d.%d"
90#define IFCONFIG_ARGS "tap0 inet %d.%d.%d.%d " NETMASK_ARGS
91#endif
92
93/* Define those to better describe your network interface. */
94#define IFNAME0 't'
95#define IFNAME1 'p'
96
97#ifndef TAPIF_DEBUG
98#define TAPIF_DEBUG LWIP_DBG_OFF
99#endif
100
101struct tapif {
102 /* Add whatever per-interface state that is needed here. */
103 int fd;
104};
105
106/* Forward declarations. */
107static void tapif_input(struct netif *netif);
108#if !NO_SYS
109static void tapif_thread(void *arg);
110#endif /* !NO_SYS */
111
112/*-----------------------------------------------------------------------------------*/
113static void
114low_level_init(struct netif *netif)
115{
116 struct tapif *tapif;
117#if LWIP_IPV4
118 int ret;
119 char buf[1024];
120#endif /* LWIP_IPV4 */
121 char *preconfigured_tapif = getenv("PRECONFIGURED_TAPIF");
122
123 tapif = (struct tapif *)netif->state;
124
125 /* Obtain MAC address from network interface. */
126
127 /* (We just fake an address...) */
128 netif->hwaddr[0] = 0x02;
129 netif->hwaddr[1] = 0x12;
130 netif->hwaddr[2] = 0x34;
131 netif->hwaddr[3] = 0x56;
132 netif->hwaddr[4] = 0x78;
133 netif->hwaddr[5] = 0xab;
134 netif->hwaddr_len = 6;
135
136 /* device capabilities */
137 netif->flags = NETIF_FLAG_BROADCAST | NETIF_FLAG_ETHARP | NETIF_FLAG_IGMP;
138
139 tapif->fd = open(DEVTAP, O_RDWR);
140 LWIP_DEBUGF(TAPIF_DEBUG, ("tapif_init: fd %d\n", tapif->fd));
141 if (tapif->fd == -1) {
142#ifdef LWIP_UNIX_LINUX
143 perror("tapif_init: try running \"modprobe tun\" or rebuilding your kernel with CONFIG_TUN; cannot open "DEVTAP);
144#else /* LWIP_UNIX_LINUX */
145 perror("tapif_init: cannot open "DEVTAP);
146#endif /* LWIP_UNIX_LINUX */
147 exit(1);
148 }
149
150#ifdef LWIP_UNIX_LINUX
151 {
152 struct ifreq ifr;
153 memset(&ifr, 0, sizeof(ifr));
154
155 if (preconfigured_tapif) {
156 strncpy(ifr.ifr_name, preconfigured_tapif, sizeof(ifr.ifr_name));
157 } else {
158 strncpy(ifr.ifr_name, DEVTAP_DEFAULT_IF, sizeof(ifr.ifr_name));
159 }
160 ifr.ifr_name[sizeof(ifr.ifr_name)-1] = 0; /* ensure \0 termination */
161
162 ifr.ifr_flags = IFF_TAP|IFF_NO_PI;
163 if (ioctl(tapif->fd, TUNSETIFF, (void *) &ifr) < 0) {
164 perror("tapif_init: "DEVTAP" ioctl TUNSETIFF");
165 exit(1);
166 }
167 }
168#endif /* LWIP_UNIX_LINUX */
169
170 netif_set_link_up(netif);
171
172 if (preconfigured_tapif == NULL) {
173#if LWIP_IPV4
174 snprintf(buf, 1024, IFCONFIG_BIN IFCONFIG_ARGS,
175 ip4_addr1(netif_ip4_gw(netif)),
176 ip4_addr2(netif_ip4_gw(netif)),
177 ip4_addr3(netif_ip4_gw(netif)),
178 ip4_addr4(netif_ip4_gw(netif))
179#ifdef NETMASK_ARGS
180 ,
181 ip4_addr1(netif_ip4_netmask(netif)),
182 ip4_addr2(netif_ip4_netmask(netif)),
183 ip4_addr3(netif_ip4_netmask(netif)),
184 ip4_addr4(netif_ip4_netmask(netif))
185#endif /* NETMASK_ARGS */
186 );
187
188 LWIP_DEBUGF(TAPIF_DEBUG, ("tapif_init: system(\"%s\");\n", buf));
189 ret = system(buf);
190 if (ret < 0) {
191 perror("ifconfig failed");
192 exit(1);
193 }
194 if (ret != 0) {
195 printf("ifconfig returned %d\n", ret);
196 }
197#else /* LWIP_IPV4 */
198 perror("todo: support IPv6 support for non-preconfigured tapif");
199 exit(1);
200#endif /* LWIP_IPV4 */
201 }
202
203#if !NO_SYS
204 sys_thread_new("tapif_thread", tapif_thread, netif, DEFAULT_THREAD_STACKSIZE, DEFAULT_THREAD_PRIO);
205#endif /* !NO_SYS */
206}
207/*-----------------------------------------------------------------------------------*/
208/*
209 * low_level_output():
210 *
211 * Should do the actual transmission of the packet. The packet is
212 * contained in the pbuf that is passed to the function. This pbuf
213 * might be chained.
214 *
215 */
216/*-----------------------------------------------------------------------------------*/
217
218static err_t
219low_level_output(struct netif *netif, struct pbuf *p)
220{
221 struct tapif *tapif = (struct tapif *)netif->state;
222 char buf[1518]; /* max packet size including VLAN excluding CRC */
223 ssize_t written;
224
225#if 0
226 if (((double)rand()/(double)RAND_MAX) < 0.2) {
227 printf("drop output\n");
228 return ERR_OK; /* ERR_OK because we simulate packet loss on cable */
229 }
230#endif
231
232 if (p->tot_len > sizeof(buf)) {
233 MIB2_STATS_NETIF_INC(netif, ifoutdiscards);
234 perror("tapif: packet too large");
235 return ERR_IF;
236 }
237
238 /* initiate transfer(); */
239 pbuf_copy_partial(p, buf, p->tot_len, 0);
240
241 /* signal that packet should be sent(); */
242 written = write(tapif->fd, buf, p->tot_len);
243 if (written < p->tot_len) {
244 MIB2_STATS_NETIF_INC(netif, ifoutdiscards);
245 perror("tapif: write");
246 return ERR_IF;
247 } else {
248 MIB2_STATS_NETIF_ADD(netif, ifoutoctets, (u32_t)written);
249 return ERR_OK;
250 }
251}
252/*-----------------------------------------------------------------------------------*/
253/*
254 * low_level_input():
255 *
256 * Should allocate a pbuf and transfer the bytes of the incoming
257 * packet from the interface into the pbuf.
258 *
259 */
260/*-----------------------------------------------------------------------------------*/
261static struct pbuf *
262low_level_input(struct netif *netif)
263{
264 struct pbuf *p;
265 u16_t len;
266 ssize_t readlen;
267 char buf[1518]; /* max packet size including VLAN excluding CRC */
268 struct tapif *tapif = (struct tapif *)netif->state;
269
270 /* Obtain the size of the packet and put it into the "len"
271 variable. */
272 readlen = read(tapif->fd, buf, sizeof(buf));
273 if (readlen < 0) {
274 perror("read returned -1");
275 exit(1);
276 }
277 len = (u16_t)readlen;
278
279 MIB2_STATS_NETIF_ADD(netif, ifinoctets, len);
280
281#if 0
282 if (((double)rand()/(double)RAND_MAX) < 0.2) {
283 printf("drop\n");
284 return NULL;
285 }
286#endif
287
288 /* We allocate a pbuf chain of pbufs from the pool. */
289 p = pbuf_alloc(PBUF_RAW, len, PBUF_POOL);
290 if (p != NULL) {
291 pbuf_take(p, buf, len);
292 /* acknowledge that packet has been read(); */
293 } else {
294 /* drop packet(); */
295 MIB2_STATS_NETIF_INC(netif, ifindiscards);
296 LWIP_DEBUGF(NETIF_DEBUG, ("tapif_input: could not allocate pbuf\n"));
297 }
298
299 return p;
300}
301
302/*-----------------------------------------------------------------------------------*/
303/*
304 * tapif_input():
305 *
306 * This function should be called when a packet is ready to be read
307 * from the interface. It uses the function low_level_input() that
308 * should handle the actual reception of bytes from the network
309 * interface.
310 *
311 */
312/*-----------------------------------------------------------------------------------*/
313static void
314tapif_input(struct netif *netif)
315{
316 struct pbuf *p = low_level_input(netif);
317
318 if (p == NULL) {
319#if LINK_STATS
320 LINK_STATS_INC(link.recv);
321#endif /* LINK_STATS */
322 LWIP_DEBUGF(TAPIF_DEBUG, ("tapif_input: low_level_input returned NULL\n"));
323 return;
324 }
325
326 if (netif->input(p, netif) != ERR_OK) {
327 LWIP_DEBUGF(NETIF_DEBUG, ("tapif_input: netif input error\n"));
328 pbuf_free(p);
329 }
330}
331/*-----------------------------------------------------------------------------------*/
332/*
333 * tapif_init():
334 *
335 * Should be called at the beginning of the program to set up the
336 * network interface. It calls the function low_level_init() to do the
337 * actual setup of the hardware.
338 *
339 */
340/*-----------------------------------------------------------------------------------*/
341err_t
342tapif_init(struct netif *netif)
343{
344 struct tapif *tapif = (struct tapif *)mem_malloc(sizeof(struct tapif));
345
346 if (tapif == NULL) {
347 LWIP_DEBUGF(NETIF_DEBUG, ("tapif_init: out of memory for tapif\n"));
348 return ERR_MEM;
349 }
350 netif->state = tapif;
351 MIB2_INIT_NETIF(netif, snmp_ifType_other, 100000000);
352
353 netif->name[0] = IFNAME0;
354 netif->name[1] = IFNAME1;
355#if LWIP_IPV4
356 netif->output = etharp_output;
357#endif /* LWIP_IPV4 */
358#if LWIP_IPV6
359 netif->output_ip6 = ethip6_output;
360#endif /* LWIP_IPV6 */
361 netif->linkoutput = low_level_output;
362 netif->mtu = 1500;
363
364 low_level_init(netif);
365
366 return ERR_OK;
367}
368
369
370/*-----------------------------------------------------------------------------------*/
371void
372tapif_poll(struct netif *netif)
373{
374 tapif_input(netif);
375}
376
377#if NO_SYS
378
379int
380tapif_select(struct netif *netif)
381{
382 fd_set fdset;
383 int ret;
384 struct timeval tv;
385 struct tapif *tapif;
386 u32_t msecs = sys_timeouts_sleeptime();
387
388 tapif = (struct tapif *)netif->state;
389
390 tv.tv_sec = msecs / 1000;
391 tv.tv_usec = (msecs % 1000) * 1000;
392
393 FD_ZERO(&fdset);
394 FD_SET(tapif->fd, &fdset);
395
396 ret = select(tapif->fd + 1, &fdset, NULL, NULL, &tv);
397 if (ret > 0) {
398 tapif_input(netif);
399 }
400 return ret;
401}
402
403#else /* NO_SYS */
404
405static void
406tapif_thread(void *arg)
407{
408 struct netif *netif;
409 struct tapif *tapif;
410 fd_set fdset;
411 int ret;
412
413 netif = (struct netif *)arg;
414 tapif = (struct tapif *)netif->state;
415
416 while(1) {
417 FD_ZERO(&fdset);
418 FD_SET(tapif->fd, &fdset);
419
420 /* Wait for a packet to arrive. */
421 ret = select(tapif->fd + 1, &fdset, NULL, NULL, NULL);
422
423 if(ret == 1) {
424 /* Handle incoming packet. */
425 tapif_input(netif);
426 } else if(ret == -1) {
427 perror("tapif_thread: select");
428 }
429 }
430}
431
432#endif /* NO_SYS */
Note: See TracBrowser for help on using the repository browser.