2010-10-25 10:42:40 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2001, Adam Dunkels.
|
|
|
|
* Copyright (c) 2009, Joakim Eriksson, Niclas Finne.
|
2011-01-17 09:16:55 +00:00
|
|
|
* Copyright (c) 2011, STMicroelectronics.
|
2010-10-25 10:42:40 +00:00
|
|
|
* 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. The name of the author may not be used to endorse or promote
|
|
|
|
* products derived from this software without specific prior
|
|
|
|
* written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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.
|
|
|
|
*
|
|
|
|
* This file is part of the uIP TCP/IP stack.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \file
|
|
|
|
* A driver for the uip6-bridge customized for STM32W that works
|
|
|
|
* under Windows (using cygwin dll). Thanks to Adam Dunkels, Joakim
|
|
|
|
* Eriksson, Niclas Finne for the original code.
|
|
|
|
* \author
|
|
|
|
* Salvatore Pitrulli <salvopitru@users.sourceforge.net>
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#ifdef __CYGWIN__
|
|
|
|
#include <alloca.h>
|
|
|
|
#else /* __CYGWIN__ */
|
|
|
|
#include <malloc.h>
|
|
|
|
#endif /* __CYGWIN__ */
|
|
|
|
#include <signal.h>
|
|
|
|
//#include <stdarg.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <sys/select.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
//#include <termios.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <err.h>
|
|
|
|
|
|
|
|
//#include <netinet/in.h>
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
|
|
|
|
|
|
|
|
#include <windows.h>
|
|
|
|
|
|
|
|
|
|
|
|
#include "net/uip_arp.h"
|
|
|
|
|
2011-05-24 08:31:05 +02:00
|
|
|
#include "ip-process.h"
|
2010-10-25 10:42:40 +00:00
|
|
|
|
|
|
|
char * wpcap_start(struct uip_eth_addr *addr, int log);
|
|
|
|
|
|
|
|
void wpcap_send(void *buf, int len);
|
|
|
|
|
|
|
|
uint16_t wpcap_poll(char *buf);
|
|
|
|
|
|
|
|
void wpcap_exit(void);
|
|
|
|
|
|
|
|
int ssystem(const char *fmt, ...)
|
|
|
|
__attribute__((__format__ (__printf__, 1, 2)));
|
|
|
|
void write_to_serial(void *inbuf, int len);
|
|
|
|
|
2011-08-24 11:05:58 -04:00
|
|
|
#define PRINTF(...) if(verbose)fprintf(stderr,__VA_ARGS__)
|
2010-10-25 10:42:40 +00:00
|
|
|
|
|
|
|
//#define PROGRESS(s) fprintf(stderr, s)
|
|
|
|
#define PROGRESS(s) do { } while (0)
|
|
|
|
|
|
|
|
#define REQUEST_MAC_TIMEOUT 3
|
|
|
|
|
|
|
|
typedef enum {
|
|
|
|
false = 0,
|
|
|
|
true = 1,
|
|
|
|
} bool;
|
|
|
|
|
2011-01-17 09:16:55 +00:00
|
|
|
const char *prog;
|
|
|
|
/* Local adapter IP address. */
|
|
|
|
static const char *local_ipaddr = NULL;
|
|
|
|
/* Attached device's IP address. */
|
|
|
|
static char rem_ipaddr[INET6_ADDRSTRLEN];
|
2010-10-25 10:42:40 +00:00
|
|
|
static char *ipprefix = NULL;
|
2011-01-17 09:16:55 +00:00
|
|
|
static char autoconf_addr[INET6_ADDRSTRLEN] = {0};
|
2010-10-25 10:42:40 +00:00
|
|
|
static bool autoconf = false;
|
|
|
|
static bool verbose = false;
|
2011-01-17 09:16:55 +00:00
|
|
|
static bool tun = false;
|
|
|
|
static bool clean_addr = false;
|
|
|
|
static bool clean_route = false;
|
|
|
|
static bool clean_neighb = false;
|
|
|
|
static struct uip_eth_addr adapter_eth_addr;
|
2010-10-25 10:42:40 +00:00
|
|
|
static char * if_name;
|
2011-08-24 11:05:58 -04:00
|
|
|
static int timestamp = 1;
|
2011-03-17 11:02:42 +01:00
|
|
|
|
2010-10-25 10:42:40 +00:00
|
|
|
OSVERSIONINFO osVersionInfo;
|
|
|
|
|
2011-01-17 09:16:55 +00:00
|
|
|
/* Fictitious Ethernet address of the attached device (used in tun mode). */
|
2011-03-22 12:48:02 +01:00
|
|
|
static struct uip_eth_addr dev_eth_addr = {{0x00,0x00,0x00,0x00,0x00,0x00}};
|
2011-01-17 09:16:55 +00:00
|
|
|
|
|
|
|
|
|
|
|
static bool request_mac = true;
|
|
|
|
static bool send_mac = true;
|
|
|
|
static bool set_sniffer_mode = true;
|
|
|
|
static bool set_channel = true;
|
|
|
|
static bool send_prefix = false;
|
|
|
|
/* Network prefix for border router. */
|
2011-01-19 09:22:23 +00:00
|
|
|
char * br_prefix = NULL;
|
2010-10-25 10:42:40 +00:00
|
|
|
|
|
|
|
static int sniffer_mode = 0;
|
|
|
|
static int channel = 0;
|
|
|
|
|
|
|
|
static bool mac_received = false;
|
|
|
|
|
|
|
|
int
|
|
|
|
ssystem(const char *fmt, ...) __attribute__((__format__ (__printf__, 1, 2)));
|
|
|
|
|
|
|
|
void addAddress(const char * ifname, const char * ipaddr);
|
|
|
|
void delAddress(const char * ifname, const char * ipaddr);
|
2011-01-17 09:16:55 +00:00
|
|
|
void addLoWPANRoute(const char * ifname, const char * net, const char * gw);
|
|
|
|
void delLoWPANRoute(const char * ifname, const char * net);
|
|
|
|
void addNeighbor(const char * ifname, const char * neighb, const char * neighb_mac);
|
|
|
|
void delNeighbor(const char * ifname, const char * neighb);
|
|
|
|
int IPAddrFromPrefix(char * ipaddr, const char * ipprefix, const char * mac);
|
|
|
|
|
|
|
|
void print_help()
|
|
|
|
{
|
|
|
|
fprintf(stderr, "usage: %s -s siodev [options] <local interface mac address>\r\n\n", prog);
|
|
|
|
fprintf(stderr, "Options:\r\n");
|
|
|
|
fprintf(stderr, "-s siodev\tDevice that identifies the bridge or the boder router.\r\n");
|
|
|
|
fprintf(stderr, "-B baudrate\tBaudrate of the serial port (default:115200).\r\n");
|
2011-08-24 11:05:58 -04:00
|
|
|
fprintf(stderr, "-L Log time\t-L1 HH:MM:SS -L2 HH:MM:SS.xxx -L3 SSSS -L4 SSSS.xxxx\r\n");
|
2011-01-17 09:16:55 +00:00
|
|
|
fprintf(stderr, " One between:\n");
|
|
|
|
fprintf(stderr, " -a ipaddress/[prefixlen] The address to be assigned to the local interface.\r\n");
|
|
|
|
fprintf(stderr, "\t\tadapter.\r\n");
|
|
|
|
fprintf(stderr, " -p 64bit-prefix Automatic assignment of the IPv6 address from the specified\r\n");
|
|
|
|
fprintf(stderr, "\t\tsubnet prefix, based on bridge's MAC address. It may be\r\n");
|
|
|
|
fprintf(stderr, "\t\tfollowed by the prefix length.\r\n");
|
|
|
|
fprintf(stderr, "\t\tNot allowed in Border Router mode.\r\n");
|
|
|
|
fprintf(stderr, "-c channel\t802.15.4 radio channel.\r\n");
|
|
|
|
//fprintf(stderr, "-r\t\tSet sniffer mode. \r\n");
|
|
|
|
fprintf(stderr, "-t\t\tUse tun interface, i.e., send bare IP packets to device.\r\n");
|
|
|
|
fprintf(stderr, "-b 64bit-prefix\tAttached device is an RPL Border Router (-t option forced).\r\n");
|
|
|
|
fprintf(stderr, "\t\t64bit-prefix is the prefix the border router has to announce.\r\n");
|
|
|
|
fprintf(stderr, "-v\t\tVerbose. Print more info.\r\n");
|
|
|
|
fprintf(stderr, "-h\t\tShow this help.\r\n");
|
|
|
|
fprintf(stderr, "\r\n<local interface mac address>\tMAC address of the local interface that will\r\n");
|
|
|
|
fprintf(stderr, "\t\tbe used by wcapslip6.\r\n");
|
|
|
|
|
|
|
|
exit(0);
|
|
|
|
}
|
2010-10-25 10:42:40 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
ssystem(const char *fmt, ...)
|
|
|
|
{
|
|
|
|
char cmd[128];
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, fmt);
|
|
|
|
vsnprintf(cmd, sizeof(cmd), fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
printf("%s\n", cmd);
|
|
|
|
fflush(stdout);
|
|
|
|
return system(cmd);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
execProcess(LPDWORD exitCode,const char *fmt, ...)
|
|
|
|
{
|
|
|
|
char cmd[128];
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, fmt);
|
|
|
|
vsnprintf(cmd, sizeof(cmd), fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
printf("%s\n", cmd);
|
|
|
|
fflush(stdout);
|
|
|
|
//return system(cmd);
|
|
|
|
|
|
|
|
STARTUPINFO si;
|
|
|
|
PROCESS_INFORMATION pi;
|
|
|
|
|
|
|
|
|
|
|
|
ZeroMemory( &si, sizeof(si) );
|
|
|
|
si.cb = sizeof(si);
|
|
|
|
ZeroMemory( &pi, sizeof(pi) );
|
|
|
|
|
|
|
|
// Start the child process.
|
|
|
|
if( !CreateProcess( NULL, // No module name (use command line)
|
|
|
|
cmd, // Command line
|
|
|
|
NULL, // Process handle not inheritable
|
|
|
|
NULL, // Thread handle not inheritable
|
|
|
|
FALSE, // Set handle inheritance to FALSE
|
|
|
|
0, // No creation flags
|
|
|
|
NULL, // Use parent's environment block
|
|
|
|
NULL, // Use parent's starting directory
|
|
|
|
&si, // Pointer to STARTUPINFO structure
|
|
|
|
&pi ) // Pointer to PROCESS_INFORMATION structure
|
|
|
|
)
|
|
|
|
{
|
|
|
|
//printf( "CreateProcess failed (%d).\n", (int)GetLastError() );
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait until child process exits.
|
|
|
|
WaitForSingleObject( pi.hProcess, INFINITE );
|
|
|
|
|
|
|
|
if(exitCode!=NULL){
|
|
|
|
GetExitCodeProcess(pi.hProcess,exitCode);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close process and thread handles.
|
|
|
|
CloseHandle( pi.hProcess );
|
|
|
|
CloseHandle( pi.hThread );
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-08-24 11:05:58 -04:00
|
|
|
void
|
|
|
|
stamptime(void)
|
|
|
|
{
|
|
|
|
static long startsecs=0,startmsecs=0;
|
|
|
|
long secs,msecs;
|
|
|
|
struct timeval tv;
|
|
|
|
time_t t;
|
|
|
|
struct tm *tmp;
|
|
|
|
char timec[20];
|
|
|
|
|
|
|
|
gettimeofday(&tv, NULL) ;
|
|
|
|
msecs=tv.tv_usec/1000;
|
|
|
|
secs=tv.tv_sec;
|
|
|
|
if (startsecs) {
|
|
|
|
if (timestamp<3) {
|
|
|
|
t=time(NULL);
|
|
|
|
tmp=localtime(&t);
|
|
|
|
strftime(timec,sizeof(timec),"%T",tmp);
|
|
|
|
if (timestamp==2) {
|
|
|
|
fprintf(stderr,"%s.%03lu ",timec,msecs);
|
|
|
|
} else {
|
|
|
|
fprintf(stderr,"%s ",timec);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
secs -=startsecs;
|
|
|
|
msecs-=startmsecs;
|
|
|
|
if (msecs<0) {secs--;msecs+=1000;}
|
|
|
|
if (timestamp==3) {
|
|
|
|
fprintf(stderr,"%04lu ", secs);
|
|
|
|
} else {
|
|
|
|
fprintf(stderr,"%04lu.%03lu ", secs, msecs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
startsecs=secs;
|
|
|
|
startmsecs=msecs;
|
|
|
|
t=time(NULL);
|
|
|
|
tmp=localtime(&t);
|
|
|
|
strftime(timec,sizeof(timec),"%T",tmp);
|
|
|
|
if ((timestamp==2) || (timestamp>3)) {
|
|
|
|
fprintf(stderr,"\n%s.%03lu ",timec,msecs);
|
|
|
|
} else {
|
|
|
|
fprintf(stderr,"\n%s ",timec);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-10-25 10:42:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
#define SLIP_END 0300
|
|
|
|
#define SLIP_ESC 0333
|
|
|
|
#define SLIP_ESC_END 0334
|
|
|
|
#define SLIP_ESC_ESC 0335
|
|
|
|
|
|
|
|
/*static void
|
|
|
|
print_packet(u_int8_t *p, int len) {
|
2011-01-17 09:16:55 +00:00
|
|
|
int i;
|
|
|
|
printf("\n");
|
|
|
|
for(i = 0; i < len; i++) {
|
|
|
|
printf("%02x", p[i]);
|
|
|
|
if ((i & 3) == 3)
|
|
|
|
printf(" ");
|
|
|
|
if ((i & 15) == 15)
|
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
printf("\n");
|
2010-10-25 10:42:40 +00:00
|
|
|
}*/
|
2011-08-24 11:05:58 -04:00
|
|
|
#if 0
|
2010-10-25 10:42:40 +00:00
|
|
|
int
|
|
|
|
is_sensible_string(const unsigned char *s, int len)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for(i = 1; i < len; i++) {
|
|
|
|
if(s[i] == 0 || s[i] == '\r' || s[i] == '\n' || s[i] == '\t') {
|
|
|
|
continue;
|
|
|
|
} else if(s[i] < ' ' || '~' < s[i]) {
|
2011-08-24 11:05:58 -04:00
|
|
|
printf("\nbad character at %d:%x\n",i,s[i]);
|
2010-10-25 10:42:40 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
2011-08-24 11:05:58 -04:00
|
|
|
#endif
|
2010-10-25 10:42:40 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Read from serial, when we have a packet write it to tun. No output
|
|
|
|
* buffering, input buffered by stdio.
|
|
|
|
*/
|
2011-01-17 09:16:55 +00:00
|
|
|
|
|
|
|
#define BUF_SIZE 2000
|
|
|
|
|
2010-10-25 10:42:40 +00:00
|
|
|
void
|
|
|
|
serial_to_wpcap(FILE *inslip)
|
|
|
|
{
|
2012-02-20 19:45:47 +00:00
|
|
|
uint16_t buf_aligned[BUF_SIZE/2 + 42]; //extra for possible eth_hdr and ip_process expansion
|
|
|
|
uint8_t *buf = (uint8_t *)buf_aligned;
|
2010-10-25 10:42:40 +00:00
|
|
|
|
2011-08-24 11:05:58 -04:00
|
|
|
static int inbufptr = 0, issensiblestring=1;
|
2011-01-17 09:16:55 +00:00
|
|
|
int ret;
|
2010-10-25 10:42:40 +00:00
|
|
|
unsigned char c;
|
|
|
|
|
2011-01-17 09:16:55 +00:00
|
|
|
unsigned char * inpktbuf;
|
|
|
|
|
|
|
|
if(tun){
|
|
|
|
inpktbuf = buf + sizeof(struct uip_eth_hdr);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
inpktbuf = buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-10-25 10:42:40 +00:00
|
|
|
|
|
|
|
read_more:
|
2011-01-17 09:16:55 +00:00
|
|
|
if(inbufptr >= BUF_SIZE) {
|
2010-10-25 10:42:40 +00:00
|
|
|
inbufptr = 0;
|
|
|
|
}
|
|
|
|
ret = fread(&c, 1, 1, inslip);
|
2011-01-17 09:16:55 +00:00
|
|
|
|
2010-10-25 10:42:40 +00:00
|
|
|
if(ret == -1) {
|
|
|
|
err(1, "serial_to_tun: read");
|
|
|
|
}
|
|
|
|
if(ret == 0) {
|
|
|
|
clearerr(inslip);
|
|
|
|
return;
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2010-10-25 10:42:40 +00:00
|
|
|
fprintf(stderr, "serial_to_tun: EOF\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
/* fprintf(stderr, ".");*/
|
|
|
|
switch(c) {
|
|
|
|
case SLIP_END:
|
|
|
|
if(inbufptr > 0) {
|
2011-01-17 09:16:55 +00:00
|
|
|
if(inpktbuf[0] == '!') {
|
|
|
|
if (inpktbuf[1] == 'M' && inbufptr == 18) {
|
2010-10-25 10:42:40 +00:00
|
|
|
/* Read gateway MAC address and autoconfigure tap0 interface */
|
2011-03-22 12:48:02 +01:00
|
|
|
char macs64[24], macs48[18];
|
|
|
|
int addr_bytes[8];
|
|
|
|
int i, pos;
|
2010-10-25 10:42:40 +00:00
|
|
|
for(i = 0, pos = 0; i < 16; i++) {
|
2011-03-22 12:48:02 +01:00
|
|
|
macs64[pos++] = inpktbuf[2 + i];
|
2010-10-25 10:42:40 +00:00
|
|
|
if ((i & 1) == 1 && i < 14) {
|
2011-03-22 12:48:02 +01:00
|
|
|
macs64[pos++] = '-';
|
2010-10-25 10:42:40 +00:00
|
|
|
}
|
|
|
|
}
|
2011-03-22 12:48:02 +01:00
|
|
|
macs64[pos] = '\0';
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
|
|
|
fprintf(stderr, "*** Gateway's MAC address: %s\n", macs64);
|
2010-10-25 10:42:40 +00:00
|
|
|
mac_received = true;
|
2011-03-22 12:48:02 +01:00
|
|
|
|
|
|
|
sscanf(macs64, "%2X-%2X-%2X-%2X-%2X-%2X-%2X-%2X",
|
|
|
|
&addr_bytes[0],
|
|
|
|
&addr_bytes[1],
|
|
|
|
&addr_bytes[2],
|
|
|
|
&addr_bytes[3],
|
|
|
|
&addr_bytes[4],
|
|
|
|
&addr_bytes[5],
|
|
|
|
&addr_bytes[6],
|
|
|
|
&addr_bytes[7]);
|
|
|
|
|
|
|
|
/* Form a fictitious MAC address for the attached device from its EUI-64 (2 middle bytes elided) */
|
|
|
|
addr_bytes[0] |= 0x02;
|
|
|
|
for(i=0;i<3;i++){
|
|
|
|
dev_eth_addr.addr[i] = addr_bytes[i];
|
|
|
|
}
|
|
|
|
for(i=3;i<6;i++){
|
|
|
|
dev_eth_addr.addr[i] = addr_bytes[i+2];
|
|
|
|
}
|
|
|
|
sprintf(macs48,"%02X-%02X-%02X-%02X-%02X-%02X",
|
|
|
|
dev_eth_addr.addr[0],
|
|
|
|
dev_eth_addr.addr[1],
|
|
|
|
dev_eth_addr.addr[2],
|
|
|
|
dev_eth_addr.addr[3],
|
|
|
|
dev_eth_addr.addr[4],
|
|
|
|
dev_eth_addr.addr[5]);
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
|
|
|
fprintf(stderr,"Fictitious MAC-48: %s\n", macs48);
|
2010-10-25 10:42:40 +00:00
|
|
|
|
|
|
|
if(autoconf){
|
|
|
|
|
2011-03-22 12:48:02 +01:00
|
|
|
if(IPAddrFromPrefix(autoconf_addr, ipprefix, macs64)!=0){
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2011-01-17 09:16:55 +00:00
|
|
|
fprintf(stderr, "Invalid IPv6 address.\n");
|
2010-10-25 10:42:40 +00:00
|
|
|
exit(1);
|
2011-01-17 09:16:55 +00:00
|
|
|
}
|
|
|
|
local_ipaddr = autoconf_addr;
|
|
|
|
addAddress(if_name,local_ipaddr);
|
|
|
|
|
|
|
|
}
|
|
|
|
if(br_prefix != NULL){
|
|
|
|
/* RPL Border Router mode. Add route towards LoWPAN. */
|
2010-10-25 10:42:40 +00:00
|
|
|
|
2011-03-22 12:48:02 +01:00
|
|
|
if(IPAddrFromPrefix(rem_ipaddr, br_prefix, macs64)!=0){
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2011-01-17 09:16:55 +00:00
|
|
|
fprintf(stderr, "Invalid IPv6 address.\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2010-10-25 10:42:40 +00:00
|
|
|
|
2011-01-17 09:16:55 +00:00
|
|
|
addLoWPANRoute(if_name, br_prefix, rem_ipaddr);
|
2011-03-22 12:48:02 +01:00
|
|
|
addNeighbor(if_name, rem_ipaddr, macs48);
|
2011-01-17 09:16:55 +00:00
|
|
|
}
|
2010-10-25 10:42:40 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
#define DEBUG_LINE_MARKER '\r'
|
|
|
|
}
|
2011-01-17 09:16:55 +00:00
|
|
|
else if(inpktbuf[0] == '?') {
|
|
|
|
if (inpktbuf[1] == 'M') {
|
2010-10-25 10:42:40 +00:00
|
|
|
/* Send our MAC address. */
|
|
|
|
|
2011-01-17 09:16:55 +00:00
|
|
|
send_mac = true;
|
|
|
|
set_sniffer_mode = true;
|
|
|
|
set_channel = true;
|
|
|
|
}
|
|
|
|
else if (inpktbuf[1] == 'P') {
|
|
|
|
/* Send LoWPAN network prefix to the border router. */
|
|
|
|
send_prefix = true;
|
2010-10-25 10:42:40 +00:00
|
|
|
}
|
|
|
|
}
|
2011-01-17 09:16:55 +00:00
|
|
|
else if(inpktbuf[0] == DEBUG_LINE_MARKER) {
|
2011-08-24 11:05:58 -04:00
|
|
|
/* Dont insert timestamp on wireshark packet dumps starting with 0000 */
|
|
|
|
if (timestamp) {
|
|
|
|
if (inpktbuf[0]!='0' || inpktbuf[1]!=0 || inpktbuf[2]!=0 || inpktbuf[3]!=0) stamptime();
|
|
|
|
}
|
2011-01-17 09:16:55 +00:00
|
|
|
fwrite(inpktbuf + 1, inbufptr - 1, 1, stderr);
|
2011-08-24 11:05:58 -04:00
|
|
|
issensiblestring=1;
|
2010-10-25 10:42:40 +00:00
|
|
|
}
|
2011-08-24 11:05:58 -04:00
|
|
|
#if 0
|
2011-01-17 09:16:55 +00:00
|
|
|
else if(is_sensible_string(inpktbuf, inbufptr)) {
|
2011-08-24 11:05:58 -04:00
|
|
|
/* Dont insert timestamp on wireshark packet dumps starting with 0000 */
|
|
|
|
if (timestamp) {
|
|
|
|
if (inpktbuf[0]!='0' || inpktbuf[1]!=0 || inpktbuf[2]!=0 || inpktbuf[3]!=0) stamptime();
|
|
|
|
}
|
2011-01-17 09:16:55 +00:00
|
|
|
fwrite(inpktbuf, inbufptr, 1, stderr);
|
2010-10-25 10:42:40 +00:00
|
|
|
}
|
2011-08-24 11:05:58 -04:00
|
|
|
#else
|
|
|
|
else if(issensiblestring) {
|
|
|
|
/* Dont insert timestamp on wireshark packet dumps starting with 0000 */
|
|
|
|
if (timestamp) {
|
|
|
|
if (inpktbuf[0]!='0' || inpktbuf[1]!=0 || inpktbuf[2]!=0 || inpktbuf[3]!=0) stamptime();
|
|
|
|
}
|
|
|
|
fwrite(inpktbuf, inbufptr, 1, stderr);
|
|
|
|
|
|
|
|
}
|
|
|
|
#endif
|
2010-10-25 10:42:40 +00:00
|
|
|
else {
|
2011-01-17 09:16:55 +00:00
|
|
|
|
|
|
|
PRINTF("Sending to wpcap\n");
|
|
|
|
|
|
|
|
if(tun){
|
|
|
|
|
|
|
|
//Ethernet header to be inserted before IP packet
|
|
|
|
struct uip_eth_hdr * eth_hdr = (struct uip_eth_hdr *)buf;
|
|
|
|
|
|
|
|
memcpy(ð_hdr->dest, &adapter_eth_addr, sizeof(struct uip_eth_addr));
|
|
|
|
memcpy(ð_hdr->src, &dev_eth_addr, sizeof(struct uip_eth_addr));
|
|
|
|
|
|
|
|
eth_hdr->type = htons(UIP_ETHTYPE_IPV6);
|
|
|
|
inbufptr += sizeof(struct uip_eth_hdr);
|
2011-05-24 08:31:05 +02:00
|
|
|
|
|
|
|
// Process incoming packets to transform link layer addresses inside ICMP packets.
|
|
|
|
// Try to do processing if we did not succeed to add the neighbor.
|
|
|
|
if(clean_neighb == false){
|
|
|
|
inbufptr = ip_process(buf, inbufptr);
|
|
|
|
}
|
2011-01-17 09:16:55 +00:00
|
|
|
}
|
|
|
|
//print_packet(inpktbuf, inbufptr);
|
|
|
|
|
|
|
|
wpcap_send(buf, inbufptr);
|
2010-10-25 10:42:40 +00:00
|
|
|
/* printf("After sending to wpcap\n");*/
|
|
|
|
}
|
|
|
|
inbufptr = 0;
|
2011-08-24 11:05:58 -04:00
|
|
|
issensiblestring=1;
|
2010-10-25 10:42:40 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SLIP_ESC:
|
|
|
|
if(fread(&c, 1, 1, inslip) != 1) {
|
|
|
|
clearerr(inslip);
|
|
|
|
/* Put ESC back and give up! */
|
|
|
|
ungetc(SLIP_ESC, inslip);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(c) {
|
|
|
|
case SLIP_ESC_END:
|
|
|
|
c = SLIP_END;
|
|
|
|
break;
|
|
|
|
case SLIP_ESC_ESC:
|
|
|
|
c = SLIP_ESC;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* FALLTHROUGH */
|
|
|
|
default:
|
2011-01-17 09:16:55 +00:00
|
|
|
inpktbuf[inbufptr++] = c;
|
2011-08-24 11:05:58 -04:00
|
|
|
if (issensiblestring) {
|
|
|
|
if(c == '\n') {
|
|
|
|
/* Dont insert timestamp on wireshark packet dumps starting with 0000 */
|
|
|
|
if (timestamp) {
|
|
|
|
if (inpktbuf[0]!='0' || inpktbuf[1]!=0 || inpktbuf[2]!=0 || inpktbuf[3]!=0) stamptime();
|
|
|
|
}
|
2011-08-25 14:39:56 -04:00
|
|
|
/* This could be a proper debug string starting with CR just a print to stdout */
|
|
|
|
/* Trap the CR which would cause overwriting of the timestamp */
|
|
|
|
//{int i;for (i=0;i<inbufptr;i++) fprintf(stderr,"%2x ",inpktbuf[i]);}
|
|
|
|
if(inpktbuf[0] == DEBUG_LINE_MARKER) {
|
|
|
|
fwrite(inpktbuf + 1, inbufptr - 1, 1, stderr);
|
|
|
|
} else {
|
|
|
|
fwrite(inpktbuf, inbufptr, 1, stderr);
|
|
|
|
}
|
|
|
|
inbufptr=0;
|
2011-08-24 11:05:58 -04:00
|
|
|
} else if (c == 0 || c == '\t' || c == '\r') {
|
|
|
|
} else if(c < ' ' || '~' < c) {
|
|
|
|
issensiblestring=0;
|
|
|
|
}
|
|
|
|
}
|
2010-10-25 10:42:40 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
goto read_more;
|
|
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
unsigned char slip_buf[2000];
|
|
|
|
int slip_end, slip_begin;
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
void
|
|
|
|
slip_send(unsigned char c)
|
|
|
|
{
|
|
|
|
if(slip_end >= sizeof(slip_buf)) {
|
|
|
|
err(1, "slip_send overflow");
|
|
|
|
}
|
|
|
|
slip_buf[slip_end] = c;
|
|
|
|
slip_end++;
|
|
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*/
|
2011-01-17 09:16:55 +00:00
|
|
|
void
|
|
|
|
slip_send_char(unsigned char c)
|
|
|
|
{
|
|
|
|
switch(c) {
|
|
|
|
case SLIP_END:
|
|
|
|
slip_send(SLIP_ESC);
|
|
|
|
slip_send(SLIP_ESC_END);
|
|
|
|
break;
|
|
|
|
case SLIP_ESC:
|
|
|
|
slip_send(SLIP_ESC);
|
|
|
|
slip_send(SLIP_ESC_ESC);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
slip_send(c);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*/
|
2010-10-25 10:42:40 +00:00
|
|
|
int
|
|
|
|
slip_empty()
|
|
|
|
{
|
|
|
|
return slip_end == 0;
|
|
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
void
|
|
|
|
slip_flushbuf(int fd)
|
|
|
|
{
|
|
|
|
int n;
|
|
|
|
|
|
|
|
if (slip_empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
n = write(fd, slip_buf + slip_begin, (slip_end - slip_begin));
|
|
|
|
|
|
|
|
if(n == -1 && errno != EAGAIN) {
|
|
|
|
err(1, "slip_flushbuf write failed");
|
|
|
|
} else if(n == -1) {
|
|
|
|
PROGRESS("Q"); /* Outqueueis full! */
|
|
|
|
} else {
|
|
|
|
slip_begin += n;
|
|
|
|
if(slip_begin == slip_end) {
|
|
|
|
slip_begin = slip_end = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
void
|
|
|
|
write_to_serial(void *inbuf, int len)
|
|
|
|
{
|
|
|
|
u_int8_t *p = inbuf;
|
|
|
|
int i; /*, ecode;*/
|
|
|
|
|
|
|
|
/* printf("Got packet of length %d - write SLIP\n", len);*/
|
|
|
|
/* print_packet(p, len);*/
|
|
|
|
|
|
|
|
/* It would be ``nice'' to send a SLIP_END here but it's not
|
|
|
|
* really necessary.
|
|
|
|
*/
|
|
|
|
/* slip_send(outfd, SLIP_END); */
|
|
|
|
PRINTF("Writing to serial len: %d\n", len);
|
|
|
|
for(i = 0; i < len; i++) {
|
|
|
|
switch(p[i]) {
|
|
|
|
case SLIP_END:
|
|
|
|
slip_send(SLIP_ESC);
|
|
|
|
slip_send(SLIP_ESC_END);
|
|
|
|
break;
|
|
|
|
case SLIP_ESC:
|
|
|
|
slip_send(SLIP_ESC);
|
|
|
|
slip_send(SLIP_ESC_ESC);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
slip_send(p[i]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
slip_send(SLIP_END);
|
|
|
|
PROGRESS("t");
|
|
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
#ifndef BAUDRATE
|
|
|
|
#define BAUDRATE B115200
|
|
|
|
#endif
|
|
|
|
speed_t b_rate = BAUDRATE;
|
|
|
|
|
|
|
|
void
|
|
|
|
stty_telos(int fd)
|
|
|
|
{
|
|
|
|
struct termios tty;
|
|
|
|
speed_t speed = b_rate;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if(tcflush(fd, TCIOFLUSH) == -1) err(1, "tcflush");
|
|
|
|
|
|
|
|
if(tcgetattr(fd, &tty) == -1) err(1, "tcgetattr");
|
|
|
|
|
|
|
|
cfmakeraw(&tty);
|
|
|
|
|
|
|
|
/* Nonblocking read. */
|
|
|
|
tty.c_cc[VTIME] = 0;
|
|
|
|
tty.c_cc[VMIN] = 0;
|
|
|
|
tty.c_cflag &= ~CRTSCTS;
|
|
|
|
tty.c_cflag &= ~HUPCL;
|
|
|
|
tty.c_cflag &= ~CLOCAL;
|
|
|
|
|
|
|
|
cfsetispeed(&tty, speed);
|
|
|
|
cfsetospeed(&tty, speed);
|
|
|
|
|
|
|
|
if(tcsetattr(fd, TCSAFLUSH, &tty) == -1) err(1, "tcsetattr");
|
|
|
|
|
|
|
|
#if 1
|
|
|
|
/* Nonblocking read and write. */
|
|
|
|
/* if(fcntl(fd, F_SETFL, O_NONBLOCK) == -1) err(1, "fcntl"); */
|
|
|
|
|
|
|
|
tty.c_cflag |= CLOCAL;
|
|
|
|
if(tcsetattr(fd, TCSAFLUSH, &tty) == -1) err(1, "tcsetattr");
|
|
|
|
|
|
|
|
i = TIOCM_DTR;
|
|
|
|
if(ioctl(fd, TIOCMBIS, &i) == -1) err(1, "ioctl");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
usleep(10*1000); /* Wait for hardware 10ms. */
|
|
|
|
|
|
|
|
/* Flush input and output buffers. */
|
|
|
|
if(tcflush(fd, TCIOFLUSH) == -1) err(1, "tcflush");
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
devopen(const char *dev, int flags)
|
|
|
|
{
|
|
|
|
char t[32];
|
|
|
|
strcpy(t, "/dev/");
|
|
|
|
strcat(t, dev);
|
|
|
|
return open(t, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef linux
|
|
|
|
#include <linux/if.h>
|
|
|
|
#include <linux/if_tun.h>
|
|
|
|
|
|
|
|
int
|
|
|
|
tun_alloc(char *dev)
|
|
|
|
{
|
|
|
|
struct ifreq ifr;
|
|
|
|
int fd, err;
|
|
|
|
|
|
|
|
if( (fd = open("/dev/net/tun", O_RDWR)) < 0 )
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
memset(&ifr, 0, sizeof(ifr));
|
|
|
|
|
|
|
|
/* Flags: IFF_TUN - TUN device (no Ethernet headers)
|
|
|
|
* IFF_TAP - TAP device
|
|
|
|
*
|
|
|
|
* IFF_NO_PI - Do not provide packet information
|
|
|
|
*/
|
|
|
|
ifr.ifr_flags = IFF_TAP | IFF_NO_PI;
|
|
|
|
if(*dev != 0)
|
|
|
|
strncpy(ifr.ifr_name, dev, IFNAMSIZ);
|
|
|
|
|
|
|
|
if((err = ioctl(fd, TUNSETIFF, (void *) &ifr)) < 0 ){
|
|
|
|
close(fd);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
strcpy(dev, ifr.ifr_name);
|
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
int
|
|
|
|
tun_alloc(char *dev)
|
|
|
|
{
|
|
|
|
return devopen(dev, O_RDWR);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//const char *netmask;
|
|
|
|
|
|
|
|
void
|
|
|
|
cleanup(void)
|
|
|
|
{
|
|
|
|
wpcap_exit();
|
2011-01-17 09:16:55 +00:00
|
|
|
if(clean_addr){
|
|
|
|
delAddress(if_name,local_ipaddr);
|
2010-10-25 10:42:40 +00:00
|
|
|
}
|
2011-01-17 09:16:55 +00:00
|
|
|
if(clean_route){
|
|
|
|
delLoWPANRoute(if_name,br_prefix);
|
|
|
|
}
|
|
|
|
if(clean_neighb){
|
|
|
|
delNeighbor(if_name, rem_ipaddr);
|
|
|
|
}
|
2010-10-25 10:42:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
sigcleanup(int signo)
|
|
|
|
{
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2010-10-25 10:42:40 +00:00
|
|
|
fprintf(stderr, "signal %d\n", signo);
|
|
|
|
exit(0); /* exit(0) will call cleanup() */
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
sigalarm(int signo)
|
|
|
|
{
|
|
|
|
if(!mac_received){
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2011-01-17 09:16:55 +00:00
|
|
|
fprintf(stderr, "Bridge/Router not found!\n");
|
2010-10-25 10:42:40 +00:00
|
|
|
exit(2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void send_commands(void)
|
|
|
|
{
|
|
|
|
char buf[3];
|
|
|
|
|
|
|
|
if (request_mac) {
|
|
|
|
slip_send('?');
|
|
|
|
slip_send('M');
|
|
|
|
slip_send(SLIP_END);
|
|
|
|
|
|
|
|
request_mac = 0;
|
|
|
|
alarm(REQUEST_MAC_TIMEOUT);
|
|
|
|
}
|
|
|
|
/* Send our mac to the device. If it knows our address, it is not needed to change
|
|
|
|
the MAC address of our local interface (this can be also unsupported, especially under
|
|
|
|
Windows).
|
|
|
|
*/
|
|
|
|
else if(send_mac && slip_empty()){
|
|
|
|
short i;
|
|
|
|
PRINTF("Sending our MAC.\n");
|
|
|
|
|
|
|
|
slip_send('!');
|
|
|
|
slip_send('M');
|
|
|
|
|
|
|
|
for(i=0; i < 6; i++){
|
2011-01-17 09:16:55 +00:00
|
|
|
sprintf(buf,"%02X",adapter_eth_addr.addr[i]);
|
2010-10-25 10:42:40 +00:00
|
|
|
slip_send(buf[0]);
|
|
|
|
slip_send(buf[1]);
|
|
|
|
}
|
|
|
|
slip_send(SLIP_END);
|
|
|
|
|
2011-01-17 09:16:55 +00:00
|
|
|
send_mac = false;
|
2010-10-25 10:42:40 +00:00
|
|
|
}
|
|
|
|
else if(set_sniffer_mode && slip_empty()){
|
|
|
|
|
|
|
|
PRINTF("Setting sniffer mode to %d.\n", sniffer_mode);
|
|
|
|
|
|
|
|
slip_send('!');
|
|
|
|
slip_send('O');
|
|
|
|
slip_send('S');
|
|
|
|
|
|
|
|
if(sniffer_mode){
|
|
|
|
slip_send('1');
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
slip_send('0');
|
|
|
|
}
|
|
|
|
slip_send(SLIP_END);
|
|
|
|
|
|
|
|
set_sniffer_mode = 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
else if(set_channel && slip_empty()){
|
|
|
|
|
|
|
|
if(channel != 0){
|
|
|
|
PRINTF("Setting channel %02d.\n", channel);
|
|
|
|
|
|
|
|
slip_send('!');
|
|
|
|
slip_send('O');
|
|
|
|
slip_send('C');
|
|
|
|
|
|
|
|
sprintf(buf,"%02d",channel);
|
|
|
|
slip_send(buf[0]);
|
|
|
|
slip_send(buf[1]);
|
|
|
|
|
|
|
|
slip_send(SLIP_END);
|
|
|
|
}
|
|
|
|
|
|
|
|
set_channel = 0;
|
|
|
|
|
|
|
|
}
|
2011-01-17 09:16:55 +00:00
|
|
|
else if(send_prefix && br_prefix != NULL && slip_empty()){
|
|
|
|
|
|
|
|
struct in6_addr addr;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
inet_pton(AF_INET6, br_prefix, &addr);
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2011-01-17 09:16:55 +00:00
|
|
|
fprintf(stderr,"*** Address:%s => %02x%02x:%02x%02x:%02x%02x:%02x%02x\n",
|
|
|
|
br_prefix,
|
|
|
|
addr.s6_addr[0], addr.s6_addr[1],
|
|
|
|
addr.s6_addr[2], addr.s6_addr[3],
|
|
|
|
addr.s6_addr[4], addr.s6_addr[5],
|
|
|
|
addr.s6_addr[6], addr.s6_addr[7]);
|
|
|
|
slip_send('!');
|
|
|
|
slip_send('P');
|
|
|
|
for(i = 0; i < 8; i++) {
|
|
|
|
/* need to call the slip_send_char for stuffing */
|
|
|
|
slip_send_char(addr.s6_addr[i]);
|
|
|
|
}
|
|
|
|
slip_send(SLIP_END);
|
|
|
|
|
|
|
|
send_prefix = false;
|
|
|
|
|
|
|
|
}
|
2010-10-25 10:42:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void addAddress(const char * ifname, const char * ipaddr)
|
|
|
|
{
|
|
|
|
DWORD exitCode = -1;
|
|
|
|
|
|
|
|
if(osVersionInfo.dwMajorVersion < 6){ // < Windows Vista (i.e., Windows XP; check if this command is ok for Windows Server 2003 too).
|
|
|
|
char * substr;
|
|
|
|
char tmpaddr[44];
|
|
|
|
|
|
|
|
strncpy(tmpaddr,ipaddr,sizeof(tmpaddr));
|
|
|
|
|
|
|
|
strtok(tmpaddr,"/"); // Remove possible prefix length.
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2010-10-25 10:42:40 +00:00
|
|
|
execProcess(&exitCode,"netsh interface ipv6 add address \"%s\" %s",if_name,tmpaddr);
|
|
|
|
substr = strtok(NULL,"/");
|
|
|
|
if(substr == NULL){ // A prefix length is not specified.
|
|
|
|
// Use a 64 bit prefix
|
|
|
|
strcat(tmpaddr,"/64");
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2010-10-25 10:42:40 +00:00
|
|
|
execProcess(NULL,"netsh interface ipv6 add route %s \"%s\"",tmpaddr,if_name);
|
|
|
|
}
|
|
|
|
else {
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2010-10-25 10:42:40 +00:00
|
|
|
execProcess(NULL,"netsh interface ipv6 add route %s \"%s\"",ipaddr,if_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
else{
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2010-10-25 10:42:40 +00:00
|
|
|
execProcess(&exitCode,"netsh interface ipv6 add address \"%s\" %s",if_name,ipaddr);
|
|
|
|
}
|
2011-05-24 13:03:13 +02:00
|
|
|
if(exitCode==0){
|
|
|
|
clean_addr = true;
|
|
|
|
}
|
|
|
|
else {
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2011-05-24 13:03:13 +02:00
|
|
|
fprintf(stderr, "WARNING: subprocess exited with code %ld\n", exitCode);
|
|
|
|
}
|
2010-10-25 10:42:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void delAddress(const char * ifname, const char * ipaddr)
|
|
|
|
{
|
|
|
|
char tmpaddr[42];
|
|
|
|
|
|
|
|
strncpy(tmpaddr,ipaddr,sizeof(tmpaddr));
|
|
|
|
strtok(tmpaddr,"/"); // Remove possible prefix length.
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
|
|
|
|
2010-10-25 10:42:40 +00:00
|
|
|
if(osVersionInfo.dwMajorVersion < 6){ // < Windows Vista (i.e., Windows XP; check if this command is ok for Windows Server 2003 too).
|
|
|
|
char * substr;
|
|
|
|
|
|
|
|
execProcess(NULL,"netsh interface ipv6 delete address \"%s\" %s",if_name,tmpaddr);
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2010-10-25 10:42:40 +00:00
|
|
|
substr = strtok(NULL,"/");
|
|
|
|
if(substr == NULL){ // A prefix length is not specified.
|
|
|
|
// Use a 64 bit prefix
|
2011-08-24 11:05:58 -04:00
|
|
|
strcat(tmpaddr,"/64");
|
2010-10-25 10:42:40 +00:00
|
|
|
execProcess(NULL,"netsh interface ipv6 delete route %s \"%s\"",tmpaddr,if_name);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
execProcess(NULL,"netsh interface ipv6 delete route %s \"%s\"",ipaddr,if_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
else{
|
|
|
|
strtok(tmpaddr,"/"); // Remove possible prefix length.
|
|
|
|
execProcess(NULL,"netsh interface ipv6 delete address \"%s\" %s",if_name,tmpaddr);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-01-17 09:16:55 +00:00
|
|
|
void addLoWPANRoute(const char * ifname, const char * net, const char * gw)
|
|
|
|
{
|
|
|
|
DWORD exitCode = -1;
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2011-01-17 09:16:55 +00:00
|
|
|
execProcess(&exitCode,"netsh interface ipv6 add route %s/64 \"%s\" %s", net, if_name, gw);
|
2011-05-24 13:03:13 +02:00
|
|
|
if(exitCode==0){
|
2011-01-17 09:16:55 +00:00
|
|
|
clean_route = true;
|
2011-05-24 13:03:13 +02:00
|
|
|
}
|
|
|
|
else {
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2011-05-24 13:03:13 +02:00
|
|
|
fprintf(stderr, "WARNING: subprocess exited with code %ld\n", exitCode);
|
|
|
|
}
|
2011-01-17 09:16:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void delLoWPANRoute(const char * ifname, const char * net)
|
|
|
|
{
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2011-01-17 09:16:55 +00:00
|
|
|
execProcess(NULL,"netsh interface ipv6 delete route %s/64 \"%s\"", net, if_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
void addNeighbor(const char * ifname, const char * neighb, const char * neighb_mac)
|
|
|
|
{
|
|
|
|
DWORD exitCode = -1;
|
|
|
|
|
2011-05-24 08:31:05 +02:00
|
|
|
if(osVersionInfo.dwMajorVersion >= 6){
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2011-01-17 09:16:55 +00:00
|
|
|
execProcess(&exitCode,"netsh interface ipv6 add neighbor \"%s\" %s \"%s\"", if_name, neighb, neighb_mac);
|
2011-05-24 13:03:13 +02:00
|
|
|
if(exitCode==0){
|
2011-01-17 09:16:55 +00:00
|
|
|
clean_neighb = true;
|
2011-05-24 13:03:13 +02:00
|
|
|
}
|
|
|
|
else {
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2011-05-24 13:03:13 +02:00
|
|
|
fprintf(stderr, "WARNING: subprocess exited with code %ld\n", exitCode);
|
|
|
|
}
|
2011-01-17 09:16:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void delNeighbor(const char * ifname, const char * neighb)
|
|
|
|
{
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2011-01-17 09:16:55 +00:00
|
|
|
execProcess(NULL,"netsh interface ipv6 delete neighbor \"%s\" %s", if_name, neighb);
|
|
|
|
}
|
|
|
|
|
|
|
|
int IPAddrFromPrefix(char * ipaddr, const char * ipprefix, const char * mac)
|
|
|
|
{
|
|
|
|
struct in6_addr ipv6addr;
|
|
|
|
struct uip_802154_longaddr dev_addr;
|
|
|
|
char tmp_ipprefix[INET6_ADDRSTRLEN];
|
|
|
|
char str_addr[INET6_ADDRSTRLEN] = {0};
|
|
|
|
int addr_bytes[8];
|
|
|
|
int i;
|
|
|
|
|
|
|
|
strncpy(tmp_ipprefix, ipprefix, INET6_ADDRSTRLEN);
|
|
|
|
|
|
|
|
// sscanf requires int instead of 8-bit for hexadecimal variables.
|
|
|
|
|
2011-03-22 12:48:02 +01:00
|
|
|
sscanf(mac, "%2X-%2X-%2X-%2X-%2X-%2X-%2X-%2X",
|
2011-01-17 09:16:55 +00:00
|
|
|
&addr_bytes[0],
|
|
|
|
&addr_bytes[1],
|
|
|
|
&addr_bytes[2],
|
|
|
|
&addr_bytes[3],
|
|
|
|
&addr_bytes[4],
|
|
|
|
&addr_bytes[5],
|
|
|
|
&addr_bytes[6],
|
|
|
|
&addr_bytes[7]);
|
|
|
|
|
|
|
|
for(i=0;i<8;i++){
|
|
|
|
dev_addr.addr[i] = addr_bytes[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
/*int i;
|
|
|
|
PRINTF("MAC:\n");
|
|
|
|
for(i=0; i< 8; i++)
|
|
|
|
PRINTF("%02X ",dev_addr.addr[i]);
|
|
|
|
PRINTF("\n");*/
|
|
|
|
|
2011-05-24 08:31:05 +02:00
|
|
|
dev_addr.addr[0] ^= 0x02;
|
2011-01-17 09:16:55 +00:00
|
|
|
|
|
|
|
strtok(tmp_ipprefix,"/");
|
|
|
|
|
|
|
|
if(inet_pton(AF_INET6, tmp_ipprefix, &ipv6addr)!=1){
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy modified EUI-64 to the last 64 bits of IPv6 address.
|
|
|
|
memcpy(&ipv6addr.s6_addr[8],&dev_addr,8);
|
|
|
|
|
|
|
|
inet_ntop(AF_INET6,&ipv6addr,str_addr,INET6_ADDRSTRLEN); // To string format.
|
|
|
|
|
|
|
|
char * substr = strtok(NULL,"/");
|
|
|
|
if(substr!=NULL){ // Add the prefix length.
|
|
|
|
strcat(str_addr,"/");
|
|
|
|
strcat(str_addr,substr);
|
|
|
|
}
|
|
|
|
strcpy(ipaddr, str_addr);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-01-19 09:22:23 +00:00
|
|
|
/* Check if an IP address is correct and supplied prefix length is as espected. */
|
|
|
|
bool validIPAddr(const char * ip_addr, int prefix_len)
|
|
|
|
{
|
|
|
|
/* Check for a 64 bit prefix. */
|
|
|
|
char tmp_addr[INET6_ADDRSTRLEN], tmp2_addr[INET6_ADDRSTRLEN];
|
|
|
|
char * substr;
|
|
|
|
|
2011-05-10 20:06:48 +02:00
|
|
|
strncpy(tmp_addr, ip_addr, INET6_ADDRSTRLEN);
|
2011-01-19 09:22:23 +00:00
|
|
|
|
|
|
|
strtok(tmp_addr,"/");
|
|
|
|
|
|
|
|
if(inet_pton(AF_INET6, tmp_addr, &tmp2_addr)!=1){
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(prefix_len != 0){
|
|
|
|
|
|
|
|
substr = strtok(NULL,"/");
|
|
|
|
if(substr!=NULL){
|
|
|
|
int i;
|
|
|
|
i = atoi(substr);
|
|
|
|
if(i != prefix_len){
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-01-17 09:16:55 +00:00
|
|
|
|
|
|
|
|
2010-10-25 10:42:40 +00:00
|
|
|
int
|
|
|
|
main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
int c;
|
|
|
|
int slipfd, maxfd;
|
|
|
|
int ret;
|
|
|
|
fd_set rset, wset;
|
|
|
|
FILE *inslip;
|
|
|
|
char siodev[10] = "";
|
|
|
|
int baudrate = -2;
|
|
|
|
|
|
|
|
char buf[4000];
|
2011-01-17 09:16:55 +00:00
|
|
|
|
|
|
|
prog = argv[0];
|
2010-10-25 10:42:40 +00:00
|
|
|
|
|
|
|
setvbuf(stdout, NULL, _IOLBF, 0); /* Line buffered output. */
|
|
|
|
|
|
|
|
memset(&osVersionInfo,0,sizeof(OSVERSIONINFO));
|
|
|
|
osVersionInfo.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
|
|
|
|
GetVersionEx(&osVersionInfo);
|
|
|
|
|
2011-08-24 11:05:58 -04:00
|
|
|
while((c = getopt(argc, argv, "B:D:L:hs:c:ra:p:vtb:")) != -1) {
|
2010-10-25 10:42:40 +00:00
|
|
|
switch (c) {
|
|
|
|
case 'B':
|
|
|
|
baudrate = atoi(optarg);
|
|
|
|
break;
|
|
|
|
|
2011-08-24 11:05:58 -04:00
|
|
|
case 'L':
|
|
|
|
if(strncmp("0", optarg, 1) == 0) {
|
|
|
|
timestamp = 0;
|
|
|
|
} else {
|
|
|
|
timestamp = atoi(optarg);
|
|
|
|
if (timestamp==0) timestamp=1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2010-10-25 10:42:40 +00:00
|
|
|
case 's':
|
|
|
|
if(strncmp("/dev/", optarg, 5) == 0) {
|
|
|
|
strncpy(siodev,optarg + 5,sizeof(siodev)-1);
|
|
|
|
}
|
|
|
|
else if(strncmp("COM", optarg, 3) == 0) {
|
|
|
|
|
|
|
|
int portnum;
|
|
|
|
|
|
|
|
portnum = atoi(optarg+3);
|
|
|
|
|
|
|
|
if(portnum == 0){
|
|
|
|
err(1,"port number is invalid");
|
|
|
|
}
|
|
|
|
sprintf(siodev,"ttyS%d",portnum-1);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
strncpy(siodev,optarg,sizeof(siodev)-1);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'c':
|
|
|
|
channel = atoi(optarg);
|
|
|
|
set_channel = 1;
|
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
sniffer_mode = 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'a':
|
|
|
|
if(autoconf == true){
|
2011-01-17 09:16:55 +00:00
|
|
|
print_help();
|
2010-10-25 10:42:40 +00:00
|
|
|
}
|
2011-01-17 09:16:55 +00:00
|
|
|
local_ipaddr = optarg;
|
2011-01-19 09:22:23 +00:00
|
|
|
if (!validIPAddr(local_ipaddr, 0)){
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2011-01-19 09:22:23 +00:00
|
|
|
fprintf(stderr, "Invalid IPv6 address: %s", local_ipaddr);
|
|
|
|
exit(1);
|
|
|
|
}
|
2010-10-25 10:42:40 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'p':
|
2011-01-17 09:16:55 +00:00
|
|
|
if(local_ipaddr !=NULL){
|
|
|
|
print_help();
|
2010-10-25 10:42:40 +00:00
|
|
|
}
|
|
|
|
autoconf = true;
|
|
|
|
ipprefix = optarg;
|
2011-01-19 09:22:23 +00:00
|
|
|
if (!validIPAddr(ipprefix, 0)){
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2011-01-19 09:22:23 +00:00
|
|
|
fprintf(stderr, "Invalid IPv6 prefix: %s", ipprefix);
|
|
|
|
exit(1);
|
|
|
|
}
|
2010-10-25 10:42:40 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'v':
|
|
|
|
verbose = true;
|
|
|
|
break;
|
|
|
|
|
2011-01-17 09:16:55 +00:00
|
|
|
case 't':
|
|
|
|
tun = true;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'b':
|
|
|
|
br_prefix = optarg;
|
|
|
|
send_prefix = true;
|
|
|
|
send_mac = false;
|
|
|
|
tun = true;
|
2011-01-19 09:22:23 +00:00
|
|
|
|
|
|
|
if (!validIPAddr(br_prefix, 64)){
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2011-01-19 09:22:23 +00:00
|
|
|
fprintf(stderr, "Invalid IPv6 64-bit prefix: %s", br_prefix);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
strtok(br_prefix,"/"); // Remove prefix length if it is present.
|
2011-01-17 09:16:55 +00:00
|
|
|
break;
|
|
|
|
|
2010-10-25 10:42:40 +00:00
|
|
|
case '?':
|
|
|
|
case 'h':
|
|
|
|
default:
|
2011-01-17 09:16:55 +00:00
|
|
|
print_help();
|
2010-10-25 10:42:40 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
argc -= (optind - 1);
|
|
|
|
argv += (optind - 1);
|
|
|
|
|
|
|
|
if(argc != 2 || *siodev == '\0') {
|
2011-01-17 09:16:55 +00:00
|
|
|
print_help();
|
2010-10-25 10:42:40 +00:00
|
|
|
}
|
|
|
|
|
2011-01-17 09:16:55 +00:00
|
|
|
if(autoconf == true && br_prefix != NULL){
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
2011-01-17 09:16:55 +00:00
|
|
|
fprintf(stderr, "-p and -b options cannot be used together.\r\n");
|
|
|
|
print_help();
|
|
|
|
}
|
|
|
|
|
2010-10-25 10:42:40 +00:00
|
|
|
sscanf(argv[1],"%2X-%2X-%2X-%2X-%2X-%2X",
|
2011-01-17 09:16:55 +00:00
|
|
|
(int *)&adapter_eth_addr.addr[0],(int *)&adapter_eth_addr.addr[1],
|
|
|
|
(int *)&adapter_eth_addr.addr[2],(int *)&adapter_eth_addr.addr[3],
|
|
|
|
(int *)&adapter_eth_addr.addr[4],(int *)&adapter_eth_addr.addr[5]);
|
|
|
|
if_name = wpcap_start(&adapter_eth_addr, verbose);
|
2010-10-25 10:42:40 +00:00
|
|
|
|
2011-01-17 09:16:55 +00:00
|
|
|
if(local_ipaddr!=NULL){
|
|
|
|
addAddress(if_name, local_ipaddr);
|
2010-10-25 10:42:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
switch(baudrate) {
|
|
|
|
case -2:
|
|
|
|
break; /* Use default. */
|
|
|
|
case 9600:
|
|
|
|
b_rate = B9600;
|
|
|
|
break;
|
|
|
|
case 19200:
|
|
|
|
b_rate = B19200;
|
|
|
|
break;
|
|
|
|
case 38400:
|
|
|
|
b_rate = B38400;
|
|
|
|
break;
|
|
|
|
case 57600:
|
|
|
|
b_rate = B57600;
|
|
|
|
break;
|
|
|
|
case 115200:
|
|
|
|
b_rate = B115200;
|
|
|
|
break;
|
2011-07-10 10:23:45 -04:00
|
|
|
case 230400:
|
|
|
|
b_rate = B230400;
|
|
|
|
break;
|
|
|
|
case 460800:
|
|
|
|
b_rate = B460800;
|
|
|
|
break;
|
|
|
|
case 921600:
|
|
|
|
b_rate = B921600;
|
|
|
|
break;
|
2010-10-25 10:42:40 +00:00
|
|
|
default:
|
|
|
|
err(1, "unknown baudrate %d", baudrate);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-01-17 09:16:55 +00:00
|
|
|
slipfd = devopen(siodev, O_RDWR | O_NONBLOCK | O_NOCTTY | O_NDELAY | O_DIRECT | O_SYNC );
|
|
|
|
if(slipfd == -1) {
|
|
|
|
err(1, "can't open siodev ``/dev/%s''", siodev);
|
|
|
|
}
|
2011-08-24 11:05:58 -04:00
|
|
|
if (timestamp) stamptime();
|
|
|
|
fprintf(stderr, "wpcapslip6 started on ``/dev/%s''\n", siodev);
|
2010-10-25 10:42:40 +00:00
|
|
|
stty_telos(slipfd);
|
|
|
|
slip_send(SLIP_END);
|
|
|
|
inslip = fdopen(slipfd, "r");
|
|
|
|
if(inslip == NULL) err(1, "main: fdopen");
|
|
|
|
|
|
|
|
|
|
|
|
atexit(cleanup);
|
|
|
|
signal(SIGHUP, sigcleanup);
|
|
|
|
signal(SIGTERM, sigcleanup);
|
|
|
|
signal(SIGINT, sigcleanup);
|
|
|
|
signal(SIGALRM, sigalarm);
|
|
|
|
|
|
|
|
/* Request mac address from gateway. It may be useful for setting the best
|
|
|
|
IPv6 address of the local interface. */
|
|
|
|
|
|
|
|
|
|
|
|
while(1) {
|
|
|
|
maxfd = 0;
|
|
|
|
FD_ZERO(&rset);
|
|
|
|
FD_ZERO(&wset);
|
|
|
|
|
|
|
|
send_commands();
|
|
|
|
|
|
|
|
if(!slip_empty()) { /* Anything to flush? */
|
|
|
|
FD_SET(slipfd, &wset);
|
|
|
|
}
|
|
|
|
|
|
|
|
FD_SET(slipfd, &rset); /* Read from slip ASAP! */
|
|
|
|
if(slipfd > maxfd) maxfd = slipfd;
|
|
|
|
#ifdef WITH_STDIN
|
|
|
|
FD_SET(STDIN_FILENO, &rset); /* Read from stdin too. */
|
|
|
|
if(STDIN_FILENO > maxfd) maxfd = STDIN_FILENO; /* This would not be necessary, since we know STDIN_FILENO is 0. */
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if(slip_empty()) {
|
|
|
|
char *pbuf = buf;
|
|
|
|
|
|
|
|
ret = wpcap_poll(pbuf);
|
|
|
|
if( ret > 0 ){
|
|
|
|
struct uip_eth_hdr * eth_hdr = (struct uip_eth_hdr *)pbuf;
|
|
|
|
|
|
|
|
if(eth_hdr->type == htons(UIP_ETHTYPE_IPV6)){
|
|
|
|
// We forward only IPv6 packet.
|
2011-01-17 09:16:55 +00:00
|
|
|
|
|
|
|
if(tun){
|
|
|
|
// Cut away ethernet header.
|
|
|
|
pbuf += sizeof(struct uip_eth_hdr);
|
|
|
|
ret -= sizeof(struct uip_eth_hdr);
|
|
|
|
}
|
|
|
|
|
2010-10-25 10:42:40 +00:00
|
|
|
write_to_serial(pbuf, ret);
|
|
|
|
/*print_packet(pbuf, ret);*/
|
|
|
|
slip_flushbuf(slipfd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
{
|
|
|
|
struct timeval tv;
|
|
|
|
tv.tv_sec = 0;
|
|
|
|
tv.tv_usec = 10;
|
|
|
|
ret = select(maxfd + 1, &rset, &wset, NULL, &tv);
|
|
|
|
}
|
|
|
|
if(ret == -1 && errno != EINTR) {
|
|
|
|
err(1, "select");
|
|
|
|
}
|
|
|
|
else if(ret > 0) {
|
|
|
|
if(FD_ISSET(slipfd, &rset)) {
|
|
|
|
/* printf("serial_to_wpcap\n"); */
|
|
|
|
serial_to_wpcap(inslip);
|
|
|
|
/* printf("End of serial_to_wpcap\n");*/
|
|
|
|
}
|
|
|
|
|
|
|
|
if(FD_ISSET(slipfd, &wset)) {
|
|
|
|
slip_flushbuf(slipfd);
|
|
|
|
}
|
|
|
|
#ifdef WITH_STDIN
|
|
|
|
if(FD_ISSET(STDIN_FILENO, &rset)) {
|
|
|
|
char inbuf;
|
|
|
|
if(fread(&inbuf,1,1,stdin)){
|
|
|
|
if(inbuf=='q'){
|
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|