mirror of
				https://github.com/FreeRTOS/FreeRTOS-Kernel.git
				synced 2025-10-24 13:47:47 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			171 lines
		
	
	
	
		
			4.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			171 lines
		
	
	
	
		
			4.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2001, Swedish Institute of Computer Science.
 | |
|  * All rights reserved. 
 | |
|  *
 | |
|  * Redistribution and use in source and binary forms, with or without 
 | |
|  * modification, are permitted provided that the following conditions 
 | |
|  * are met: 
 | |
|  *
 | |
|  * 1. Redistributions of source code must retain the above copyright 
 | |
|  *    notice, this list of conditions and the following disclaimer. 
 | |
|  *
 | |
|  * 2. Redistributions in binary form must reproduce the above copyright 
 | |
|  *    notice, this list of conditions and the following disclaimer in the 
 | |
|  *    documentation and/or other materials provided with the distribution. 
 | |
|  *
 | |
|  * 3. Neither the name of the Institute nor the names of its contributors 
 | |
|  *    may be used to endorse or promote products derived from this software 
 | |
|  *    without specific prior written permission. 
 | |
|  *
 | |
|  * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND 
 | |
|  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 
 | |
|  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 
 | |
|  * ARE DISCLAIMED.  IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE 
 | |
|  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 
 | |
|  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 
 | |
|  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 
 | |
|  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 
 | |
|  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 
 | |
|  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 
 | |
|  * SUCH DAMAGE. 
 | |
|  *
 | |
|  * Author: Adam Dunkels <adam@sics.se>
 | |
|  *
 | |
|  * $Id: tapdev.c,v 1.7.2.1 2003/10/07 13:23:19 adam Exp $
 | |
|  */
 | |
| 
 | |
| 
 | |
| #include <fcntl.h>
 | |
| #include <stdlib.h>
 | |
| #include <stdio.h>
 | |
| #include <unistd.h>
 | |
| #include <string.h>
 | |
| #include <sys/ioctl.h>
 | |
| #include <sys/socket.h>
 | |
| #include <sys/types.h>
 | |
| #include <sys/time.h>
 | |
| #include <sys/uio.h>
 | |
| #include <sys/socket.h>
 | |
| 
 | |
| #ifdef linux
 | |
| #include <sys/ioctl.h>
 | |
| #include <linux/if.h>
 | |
| #include <linux/if_tun.h>
 | |
| #define DEVTAP "/dev/net/tun"
 | |
| #else  /* linux */
 | |
| #define DEVTAP "/dev/tap0"
 | |
| #endif /* linux */
 | |
| 
 | |
| #include "uip.h"
 | |
| 
 | |
| static int fd;
 | |
| 
 | |
| static unsigned long lasttime;
 | |
| static struct timezone tz;
 | |
| 
 | |
| /*-----------------------------------------------------------------------------------*/
 | |
| void
 | |
| tapdev_init(void)
 | |
| {
 | |
|   char buf[1024];
 | |
|   
 | |
|   fd = open(DEVTAP, O_RDWR);
 | |
|   if(fd == -1) {
 | |
|     perror("tapdev: tapdev_init: open");
 | |
|     exit(1);
 | |
|   }
 | |
| 
 | |
| #ifdef linux
 | |
|   {
 | |
|     struct ifreq ifr;
 | |
|     memset(&ifr, 0, sizeof(ifr));
 | |
|     ifr.ifr_flags = IFF_TAP|IFF_NO_PI;
 | |
|     if (ioctl(fd, TUNSETIFF, (void *) &ifr) < 0) {
 | |
|       perror(buf);
 | |
|       exit(1);
 | |
|     }
 | |
|   }
 | |
| #endif /* Linux */
 | |
| 
 | |
|   snprintf(buf, sizeof(buf), "ifconfig tap0 inet %d.%d.%d.%d",
 | |
| 	   UIP_DRIPADDR0, UIP_DRIPADDR1, UIP_DRIPADDR2, UIP_DRIPADDR3);
 | |
|   system(buf);
 | |
| 
 | |
|   lasttime = 0;
 | |
| }
 | |
| /*-----------------------------------------------------------------------------------*/
 | |
| unsigned int
 | |
| tapdev_read(void)
 | |
| {
 | |
|   fd_set fdset;
 | |
|   struct timeval tv, now;
 | |
|   int ret;
 | |
|   
 | |
|   if(lasttime >= 500000) {
 | |
|     lasttime = 0;
 | |
|     return 0;
 | |
|   }
 | |
|   
 | |
|   tv.tv_sec = 0;
 | |
|   tv.tv_usec = 500000 - lasttime;
 | |
| 
 | |
| 
 | |
|   FD_ZERO(&fdset);
 | |
|   FD_SET(fd, &fdset);
 | |
| 
 | |
|   gettimeofday(&now, &tz);  
 | |
|   ret = select(fd + 1, &fdset, NULL, NULL, &tv);
 | |
|   if(ret == 0) {
 | |
|     lasttime = 0;    
 | |
|     return 0;
 | |
|   } 
 | |
|   ret = read(fd, uip_buf, UIP_BUFSIZE);  
 | |
|   if(ret == -1) {
 | |
|     perror("tap_dev: tapdev_read: read");
 | |
|   }
 | |
|   gettimeofday(&tv, &tz);
 | |
|   lasttime += (tv.tv_sec - now.tv_sec) * 1000000 + (tv.tv_usec - now.tv_usec);
 | |
| 
 | |
|   return ret;
 | |
| }
 | |
| /*-----------------------------------------------------------------------------------*/
 | |
| void
 | |
| tapdev_send(void)
 | |
| {
 | |
|   int ret;
 | |
|   struct iovec iov[2];
 | |
|   
 | |
| #ifdef linux
 | |
|   {
 | |
|     char tmpbuf[UIP_BUFSIZE];
 | |
|     int i;
 | |
| 
 | |
|     for(i = 0; i < 40 + UIP_LLH_LEN; i++) {
 | |
|       tmpbuf[i] = uip_buf[i];
 | |
|     }
 | |
|     
 | |
|     for(; i < uip_len; i++) {
 | |
|       tmpbuf[i] = uip_appdata[i - 40 - UIP_LLH_LEN];
 | |
|     }
 | |
|     
 | |
|     ret = write(fd, tmpbuf, uip_len);
 | |
|   }  
 | |
| #else 
 | |
| 
 | |
|   if(uip_len < 40 + UIP_LLH_LEN) {
 | |
|     ret = write(fd, uip_buf, uip_len + UIP_LLH_LEN);
 | |
|   } else {
 | |
|     iov[0].iov_base = uip_buf;
 | |
|     iov[0].iov_len = 40 + UIP_LLH_LEN;
 | |
|     iov[1].iov_base = (char *)uip_appdata;
 | |
|     iov[1].iov_len = uip_len - (40 + UIP_LLH_LEN);  
 | |
|     
 | |
|     ret = writev(fd, iov, 2);
 | |
|   }
 | |
| #endif
 | |
|   if(ret == -1) {
 | |
|     perror("tap_dev: tapdev_send: writev");
 | |
|     exit(1);
 | |
|   }
 | |
| }  
 | |
| /*-----------------------------------------------------------------------------------*/
 |