Contiki-NG
Loading...
Searching...
No Matches
tun-bridge.c
1/*
2 * Copyright (c) 2011, Swedish Institute of Computer Science.
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * 3. Neither the name of the Institute nor the names of its contributors
14 * may be used to endorse or promote products derived from this software
15 * without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
18 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20 * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
21 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27 * SUCH DAMAGE.
28 */
29
30/**
31 * \author
32 * Niclas Finne <nfi@sics.se>
33 * Joakim Eriksson <joakime@sics.se>
34 */
35
36#include "net/ipv6/uip.h"
37#include "net/ipv6/uip-ds6.h"
38#include <stdio.h>
39#include <stdlib.h>
40#include <stdarg.h>
41#include <string.h>
42#include <sys/time.h>
43#include <sys/types.h>
44
45#include <unistd.h>
46#include <errno.h>
47#include <fcntl.h>
48#include <signal.h>
49#include <termios.h>
50#include <sys/ioctl.h>
51#include <sys/socket.h>
52
53/*---------------------------------------------------------------------------*/
54/* Log configuration */
55#include "sys/log.h"
56#define LOG_MODULE "BR"
57#define LOG_LEVEL LOG_LEVEL_NONE
58
59#ifdef linux
60#include <linux/if.h>
61#include <linux/if_tun.h>
62#endif
63
64#include <err.h>
65#include "net/netstack.h"
66#include "net/packetbuf.h"
67#include "cmd.h"
68#include "border-router.h"
69
70extern const char *slip_config_ipaddr;
71extern char slip_config_tundev[32];
72extern uint16_t slip_config_basedelay;
73
74static int tunfd;
75
76static int set_fd(fd_set *rset, fd_set *wset);
77static void handle_fd(fd_set *rset, fd_set *wset);
78static const struct select_callback tun_select_callback = {
79 set_fd,
80 handle_fd
81};
82
83int ssystem(const char *fmt, ...)
84 __attribute__((__format__ (__printf__, 1, 2)));
85int
86ssystem(const char *fmt, ...) __attribute__((__format__ (__printf__, 1, 2)));
87
88int
89ssystem(const char *fmt, ...)
90{
91 char cmd[128];
92 va_list ap;
93 va_start(ap, fmt);
94 vsnprintf(cmd, sizeof(cmd), fmt, ap);
95 va_end(ap);
96 printf("%s\n", cmd);
97 fflush(stdout);
98 return system(cmd);
99}
100/*---------------------------------------------------------------------------*/
101void
102cleanup(void)
103{
104 ssystem("ifconfig %s down", slip_config_tundev);
105#ifndef linux
106 ssystem("sysctl -w net.ipv6.conf.all.forwarding=1");
107#endif
108 ssystem("netstat -nr"
109 " | awk '{ if ($2 == \"%s\") print \"route delete -net \"$1; }'"
110 " | sh",
111 slip_config_tundev);
112}
113/*---------------------------------------------------------------------------*/
114void
115sigcleanup(int signo)
116{
117 fprintf(stderr, "signal %d\n", signo);
118 exit(0); /* exit(0) will call cleanup() */
119}
120/*---------------------------------------------------------------------------*/
121void
122ifconf(const char *tundev, const char *ipaddr)
123{
124#ifdef linux
125 ssystem("ifconfig %s inet `hostname` up", tundev);
126 ssystem("ifconfig %s add %s", tundev, ipaddr);
127#elif defined(__APPLE__)
128 ssystem("ifconfig %s inet6 %s up", tundev, ipaddr);
129 ssystem("sysctl -w net.inet.ip.forwarding=1");
130#else
131 ssystem("ifconfig %s inet `hostname` %s up", tundev, ipaddr);
132 ssystem("sysctl -w net.inet.ip.forwarding=1");
133#endif /* !linux */
134
135 /* Print the configuration to the console. */
136 ssystem("ifconfig %s\n", tundev);
137}
138/*---------------------------------------------------------------------------*/
139int
140devopen(const char *dev, int flags)
141{
142 char t[32];
143 strcpy(t, "/dev/");
144 strncat(t, dev, sizeof(t) - 5);
145 return open(t, flags);
146}
147/*---------------------------------------------------------------------------*/
148#ifdef linux
149int
150tun_alloc(char *dev)
151{
152 struct ifreq ifr;
153 int fd, err;
154
155 if((fd = open("/dev/net/tun", O_RDWR)) < 0) {
156 return -1;
157 }
158
159 memset(&ifr, 0, sizeof(ifr));
160
161 /* Flags: IFF_TUN - TUN device (no Ethernet headers)
162 * IFF_NO_PI - Do not provide packet information
163 */
164 ifr.ifr_flags = IFF_TUN | IFF_NO_PI;
165 if(*dev != 0) {
166 strncpy(ifr.ifr_name, dev, IFNAMSIZ - 1);
167 }
168
169 if((err = ioctl(fd, TUNSETIFF, (void *)&ifr)) < 0) {
170 close(fd);
171 return err;
172 }
173 strcpy(dev, ifr.ifr_name);
174 return fd;
175}
176#else
177int
178tun_alloc(char *dev)
179{
180 return devopen(dev, O_RDWR);
181}
182#endif
183
184static uint16_t delaymsec = 0;
185static uint32_t delaystartsec, delaystartmsec;
186
187/*---------------------------------------------------------------------------*/
188void
189tun_init()
190{
191 setvbuf(stdout, NULL, _IOLBF, 0); /* Line buffered output. */
192
193 slip_init();
194
195 LOG_INFO("Opening tun interface:%s\n", slip_config_tundev);
196
197 tunfd = tun_alloc(slip_config_tundev);
198
199 if(tunfd == -1) {
200 err(1, "tun_init: tun_alloc failed");
201 }
202
203 select_set_callback(tunfd, &tun_select_callback);
204
205 fprintf(stderr, "opened %s device ``/dev/%s''\n",
206 "tun", slip_config_tundev);
207
208 atexit(cleanup);
209 signal(SIGHUP, sigcleanup);
210 signal(SIGTERM, sigcleanup);
211 signal(SIGINT, sigcleanup);
212 ifconf(slip_config_tundev, slip_config_ipaddr);
213}
214/*---------------------------------------------------------------------------*/
215static int
216tun_output(uint8_t *data, int len)
217{
218 /* fprintf(stderr, "*** Writing to tun...%d\n", len); */
219 if(write(tunfd, data, len) != len) {
220 err(1, "serial_to_tun: write");
221 return -1;
222 }
223 return 0;
224}
225/*---------------------------------------------------------------------------*/
226int
227tun_input(unsigned char *data, int maxlen)
228{
229 int size;
230 if((size = read(tunfd, data, maxlen)) == -1) {
231 err(1, "tun_input: read");
232 }
233 return size;
234}
235/*---------------------------------------------------------------------------*/
236static void
237init(void)
238{
239}
240/*---------------------------------------------------------------------------*/
241static int
242output(void)
243{
244 LOG_DBG("SUT: %u\n", uip_len);
245 if(uip_len > 0) {
246 return tun_output(uip_buf, uip_len);
247 }
248 return 0;
249}
250const struct uip_fallback_interface rpl_interface = {
251 init, output
252};
253
254/*---------------------------------------------------------------------------*/
255/* tun and slip select callback */
256/*---------------------------------------------------------------------------*/
257static int
258set_fd(fd_set *rset, fd_set *wset)
259{
260 FD_SET(tunfd, rset);
261 return 1;
262}
263/*---------------------------------------------------------------------------*/
264
265static void
266handle_fd(fd_set *rset, fd_set *wset)
267{
268 /* Optional delay between outgoing packets */
269 /* Base delay times number of 6lowpan fragments to be sent */
270 /* delaymsec = 10; */
271 if(delaymsec) {
272 struct timeval tv;
273 int dmsec;
274 gettimeofday(&tv, NULL);
275 dmsec = (tv.tv_sec - delaystartsec) * 1000 + tv.tv_usec / 1000 - delaystartmsec;
276 if(dmsec < 0) {
277 delaymsec = 0;
278 }
279 if(dmsec > delaymsec) {
280 delaymsec = 0;
281 }
282 }
283
284 if(delaymsec == 0) {
285 int size;
286
287 if(FD_ISSET(tunfd, rset)) {
288 size = tun_input(uip_buf, sizeof(uip_buf));
289 /* printf("TUN data incoming read:%d\n", size); */
290 uip_len = size;
291 tcpip_input();
292
293 if(slip_config_basedelay) {
294 struct timeval tv;
295 gettimeofday(&tv, NULL);
296 delaymsec = slip_config_basedelay;
297 delaystartsec = tv.tv_sec;
298 delaystartmsec = tv.tv_usec / 1000;
299 }
300 }
301 }
302}
303/*---------------------------------------------------------------------------*/
Border router header file.
Simple command handler.
void tcpip_input(void)
Deliver an incoming packet to the TCP/IP stack.
Definition tcpip.c:433
#define uip_buf
Macro to access uip_aligned_buf as an array of bytes.
Definition uip.h:465
uint16_t uip_len
The length of the packet in the uip_buf buffer.
Definition uip6.c:159
Header file for the logging system.
Include file for the Contiki low-layer network stack (NETSTACK)
Header file for the Packet buffer (packetbuf) management.
Header file for IPv6-related data structures.
static uip_ipaddr_t ipaddr
Pointer to prefix information option in uip_buf.
Definition uip-nd6.c:116
Header file for the uIP TCP/IP stack.