- route.c is now used to determine destination
- flags are removed, since they were not used at all. Use options instead. - indirectdata works now, tcponly almost... - made functions that don't return useful information void
This commit is contained in:
parent
d2a54597e0
commit
34f9e6cf2d
8 changed files with 187 additions and 198 deletions
|
@ -1,15 +1,15 @@
|
||||||
## Produce this file with automake to get Makefile.in
|
## Produce this file with automake to get Makefile.in
|
||||||
# $Id: Makefile.am,v 1.4.4.12 2001/02/11 11:50:09 guus Exp $
|
# $Id: Makefile.am,v 1.4.4.13 2001/03/04 13:59:25 guus Exp $
|
||||||
|
|
||||||
sbin_PROGRAMS = tincd
|
sbin_PROGRAMS = tincd
|
||||||
|
|
||||||
tincd_SOURCES = conf.c connection.c meta.c net.c netutl.c process.c \
|
tincd_SOURCES = conf.c connection.c meta.c net.c netutl.c process.c \
|
||||||
protocol.c subnet.c tincd.c
|
protocol.c route.c subnet.c tincd.c
|
||||||
|
|
||||||
INCLUDES = @INCLUDES@ -I$(top_builddir) -I$(top_srcdir)/lib -I$(top_srcdir)/intl
|
INCLUDES = @INCLUDES@ -I$(top_builddir) -I$(top_srcdir)/lib -I$(top_srcdir)/intl
|
||||||
|
|
||||||
noinst_HEADERS = conf.h connection.h meta.h net.h netutl.h process.h \
|
noinst_HEADERS = conf.h connection.h meta.h net.h netutl.h process.h \
|
||||||
protocol.h subnet.h
|
protocol.h route.h subnet.h
|
||||||
|
|
||||||
LIBS = @LIBS@ @INTLLIBS@
|
LIBS = @LIBS@ @INTLLIBS@
|
||||||
|
|
||||||
|
|
|
@ -17,7 +17,7 @@
|
||||||
along with this program; if not, write to the Free Software
|
along with this program; if not, write to the Free Software
|
||||||
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||||
|
|
||||||
$Id: connection.c,v 1.1.2.9 2001/01/07 17:08:56 guus Exp $
|
$Id: connection.c,v 1.1.2.10 2001/03/04 13:59:25 guus Exp $
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
|
@ -190,15 +190,15 @@ void dump_connection_list(void)
|
||||||
cp
|
cp
|
||||||
syslog(LOG_DEBUG, _("Connection list:"));
|
syslog(LOG_DEBUG, _("Connection list:"));
|
||||||
|
|
||||||
syslog(LOG_DEBUG, _(" %s at %s port %hd flags %d sockets %d, %d status %04x"),
|
syslog(LOG_DEBUG, _(" %s at %s port %hd options %ld sockets %d, %d status %04x"),
|
||||||
myself->name, myself->hostname, myself->port, myself->flags,
|
myself->name, myself->hostname, myself->port, myself->options,
|
||||||
myself->socket, myself->meta_socket, myself->status);
|
myself->socket, myself->meta_socket, myself->status);
|
||||||
|
|
||||||
for(node = connection_tree->head; node; node = node->next)
|
for(node = connection_tree->head; node; node = node->next)
|
||||||
{
|
{
|
||||||
cl = (connection_t *)node->data;
|
cl = (connection_t *)node->data;
|
||||||
syslog(LOG_DEBUG, _(" %s at %s port %hd flags %d sockets %d, %d status %04x"),
|
syslog(LOG_DEBUG, _(" %s at %s port %hd options %ld sockets %d, %d status %04x"),
|
||||||
cl->name, cl->hostname, cl->port, cl->flags,
|
cl->name, cl->hostname, cl->port, cl->options,
|
||||||
cl->socket, cl->meta_socket, cl->status);
|
cl->socket, cl->meta_socket, cl->status);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -17,7 +17,7 @@
|
||||||
along with this program; if not, write to the Free Software
|
along with this program; if not, write to the Free Software
|
||||||
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||||
|
|
||||||
$Id: connection.h,v 1.1.2.7 2001/01/07 20:19:29 guus Exp $
|
$Id: connection.h,v 1.1.2.8 2001/03/04 13:59:25 guus Exp $
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef __TINC_CONNECTION_H__
|
#ifndef __TINC_CONNECTION_H__
|
||||||
|
@ -70,7 +70,6 @@ typedef struct connection_t {
|
||||||
short unsigned int port; /* port number for UDP traffic */
|
short unsigned int port; /* port number for UDP traffic */
|
||||||
long int options; /* options turned on for this connection */
|
long int options; /* options turned on for this connection */
|
||||||
|
|
||||||
int flags; /* his flags */
|
|
||||||
int socket; /* our udp vpn socket */
|
int socket; /* our udp vpn socket */
|
||||||
int meta_socket; /* our tcp meta socket */
|
int meta_socket; /* our tcp meta socket */
|
||||||
status_bits_t status; /* status info */
|
status_bits_t status; /* status info */
|
||||||
|
|
240
src/net.c
240
src/net.c
|
@ -17,7 +17,7 @@
|
||||||
along with this program; if not, write to the Free Software
|
along with this program; if not, write to the Free Software
|
||||||
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||||
|
|
||||||
$Id: net.c,v 1.35.4.101 2001/03/01 21:32:01 guus Exp $
|
$Id: net.c,v 1.35.4.102 2001/03/04 13:59:25 guus Exp $
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
|
@ -85,6 +85,7 @@
|
||||||
#include "protocol.h"
|
#include "protocol.h"
|
||||||
#include "subnet.h"
|
#include "subnet.h"
|
||||||
#include "process.h"
|
#include "process.h"
|
||||||
|
#include "route.h"
|
||||||
|
|
||||||
#include "system.h"
|
#include "system.h"
|
||||||
|
|
||||||
|
@ -103,34 +104,43 @@ int keyexpires = 0;
|
||||||
|
|
||||||
char *unknown = NULL;
|
char *unknown = NULL;
|
||||||
|
|
||||||
subnet_t mymac;
|
void send_udppacket(connection_t *cl, vpn_packet_t *inpkt)
|
||||||
|
|
||||||
int xsend(connection_t *cl, vpn_packet_t *inpkt)
|
|
||||||
{
|
{
|
||||||
vpn_packet_t outpkt;
|
vpn_packet_t outpkt;
|
||||||
int outlen, outpad;
|
int outlen, outpad;
|
||||||
EVP_CIPHER_CTX ctx;
|
EVP_CIPHER_CTX ctx;
|
||||||
struct sockaddr_in to;
|
struct sockaddr_in to;
|
||||||
socklen_t tolen = sizeof(to);
|
socklen_t tolen = sizeof(to);
|
||||||
|
vpn_packet_t *copy;
|
||||||
cp
|
cp
|
||||||
outpkt.len = inpkt->len;
|
if(!cl->status.validkey)
|
||||||
|
{
|
||||||
|
if(debug_lvl >= DEBUG_TRAFFIC)
|
||||||
|
syslog(LOG_INFO, _("No valid key known yet for %s (%s), queueing packet"),
|
||||||
|
cl->name, cl->hostname);
|
||||||
|
|
||||||
|
/* Since packet is on the stack of handle_tap_input(),
|
||||||
|
we have to make a copy of it first. */
|
||||||
|
|
||||||
|
copy = xmalloc(sizeof(vpn_packet_t));
|
||||||
|
memcpy(copy, inpkt, sizeof(vpn_packet_t));
|
||||||
|
|
||||||
|
list_insert_tail(cl->queue, copy);
|
||||||
|
|
||||||
|
if(!cl->status.waitingforkey)
|
||||||
|
send_req_key(myself, cl);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
/* Encrypt the packet. */
|
/* Encrypt the packet. */
|
||||||
|
|
||||||
|
outpkt.len = inpkt->len;
|
||||||
|
|
||||||
EVP_EncryptInit(&ctx, cl->cipher_pkttype, cl->cipher_pktkey, cl->cipher_pktkey + cl->cipher_pkttype->key_len);
|
EVP_EncryptInit(&ctx, cl->cipher_pkttype, cl->cipher_pktkey, cl->cipher_pktkey + cl->cipher_pkttype->key_len);
|
||||||
EVP_EncryptUpdate(&ctx, outpkt.data, &outlen, inpkt->data, inpkt->len);
|
EVP_EncryptUpdate(&ctx, outpkt.data, &outlen, inpkt->data, inpkt->len);
|
||||||
EVP_EncryptFinal(&ctx, outpkt.data + outlen, &outpad);
|
EVP_EncryptFinal(&ctx, outpkt.data + outlen, &outpad);
|
||||||
outlen += outpad + 2;
|
outlen += outpad + 2;
|
||||||
|
|
||||||
/* Bypass
|
|
||||||
outlen = outpkt.len + 2;
|
|
||||||
memcpy(&outpkt, inpkt, outlen);
|
|
||||||
*/
|
|
||||||
|
|
||||||
if(debug_lvl >= DEBUG_TRAFFIC)
|
|
||||||
syslog(LOG_ERR, _("Sending packet of %d bytes to %s (%s)"),
|
|
||||||
outlen, cl->name, cl->hostname);
|
|
||||||
|
|
||||||
total_socket_out += outlen;
|
total_socket_out += outlen;
|
||||||
|
|
||||||
to.sin_family = AF_INET;
|
to.sin_family = AF_INET;
|
||||||
|
@ -141,13 +151,22 @@ cp
|
||||||
{
|
{
|
||||||
syslog(LOG_ERR, _("Error sending packet to %s (%s): %m"),
|
syslog(LOG_ERR, _("Error sending packet to %s (%s): %m"),
|
||||||
cl->name, cl->hostname);
|
cl->name, cl->hostname);
|
||||||
return -1;
|
return;
|
||||||
}
|
}
|
||||||
cp
|
cp
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int xrecv(connection_t *cl, vpn_packet_t *inpkt)
|
void receive_packet(connection_t *cl, vpn_packet_t *packet)
|
||||||
|
{
|
||||||
|
cp
|
||||||
|
if(debug_lvl >= DEBUG_TRAFFIC)
|
||||||
|
syslog(LOG_DEBUG, _("Received packet of %d bytes from %s (%s)"), packet->len, cl->name, cl->hostname);
|
||||||
|
|
||||||
|
route_incoming(cl, packet);
|
||||||
|
cp
|
||||||
|
}
|
||||||
|
|
||||||
|
void receive_udppacket(connection_t *cl, vpn_packet_t *inpkt)
|
||||||
{
|
{
|
||||||
vpn_packet_t outpkt;
|
vpn_packet_t outpkt;
|
||||||
int outlen, outpad;
|
int outlen, outpad;
|
||||||
|
@ -162,24 +181,17 @@ cp
|
||||||
EVP_DecryptFinal(&ctx, outpkt.data + outlen, &outpad);
|
EVP_DecryptFinal(&ctx, outpkt.data + outlen, &outpad);
|
||||||
outlen += outpad;
|
outlen += outpad;
|
||||||
|
|
||||||
/* Bypass
|
receive_packet(cl, &outpkt);
|
||||||
outlen = outpkt.len+2;
|
|
||||||
memcpy(&outpkt, inpkt, outlen);
|
|
||||||
*/
|
|
||||||
cp
|
cp
|
||||||
return receive_packet(cl, &outpkt);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int receive_packet(connection_t *cl, vpn_packet_t *packet)
|
void accept_packet(vpn_packet_t *packet)
|
||||||
{
|
{
|
||||||
|
cp
|
||||||
if(debug_lvl >= DEBUG_TRAFFIC)
|
if(debug_lvl >= DEBUG_TRAFFIC)
|
||||||
syslog(LOG_ERR, _("Writing packet of %d bytes to tap device"),
|
syslog(LOG_DEBUG, _("Writing packet of %d bytes to tap device"),
|
||||||
packet->len);
|
packet->len);
|
||||||
|
|
||||||
/* Fix mac address */
|
|
||||||
|
|
||||||
memcpy(packet->data, mymac.net.mac.address.x, 6);
|
|
||||||
|
|
||||||
if(taptype == TAP_TYPE_TUNTAP)
|
if(taptype == TAP_TYPE_TUNTAP)
|
||||||
{
|
{
|
||||||
if(write(tap_fd, packet->data, packet->len) < 0)
|
if(write(tap_fd, packet->data, packet->len) < 0)
|
||||||
|
@ -195,40 +207,26 @@ int receive_packet(connection_t *cl, vpn_packet_t *packet)
|
||||||
total_tap_out += packet->len + 2;
|
total_tap_out += packet->len + 2;
|
||||||
}
|
}
|
||||||
cp
|
cp
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
send a packet to the given vpn ip.
|
send a packet to the given vpn ip.
|
||||||
*/
|
*/
|
||||||
int send_packet(ip_t to, vpn_packet_t *packet)
|
void send_packet(connection_t *cl, vpn_packet_t *packet)
|
||||||
{
|
{
|
||||||
connection_t *cl;
|
|
||||||
subnet_t *subnet;
|
|
||||||
vpn_packet_t *copy;
|
|
||||||
cp
|
cp
|
||||||
if((subnet = lookup_subnet_ipv4(&to)) == NULL)
|
if(debug_lvl >= DEBUG_TRAFFIC)
|
||||||
{
|
syslog(LOG_ERR, _("Sending packet of %d bytes to %s (%s)"),
|
||||||
if(debug_lvl >= DEBUG_TRAFFIC)
|
packet->len, cl->name, cl->hostname);
|
||||||
{
|
|
||||||
syslog(LOG_NOTICE, _("Trying to look up %d.%d.%d.%d in connection list failed!"),
|
|
||||||
IP_ADDR_V(to));
|
|
||||||
}
|
|
||||||
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
cl = subnet->owner;
|
|
||||||
|
|
||||||
if(cl == myself)
|
if(cl == myself)
|
||||||
{
|
{
|
||||||
if(debug_lvl >= DEBUG_TRAFFIC)
|
if(debug_lvl >= DEBUG_TRAFFIC)
|
||||||
{
|
{
|
||||||
syslog(LOG_NOTICE, _("Packet with destination %d.%d.%d.%d is looping back to us!"),
|
syslog(LOG_NOTICE, _("Packet is looping back to us!"));
|
||||||
IP_ADDR_V(to));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return -1;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!cl->status.active)
|
if(!cl->status.active)
|
||||||
|
@ -237,34 +235,18 @@ cp
|
||||||
syslog(LOG_INFO, _("%s (%s) is not active, dropping packet"),
|
syslog(LOG_INFO, _("%s (%s) is not active, dropping packet"),
|
||||||
cl->name, cl->hostname);
|
cl->name, cl->hostname);
|
||||||
|
|
||||||
return 0;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!cl->status.validkey)
|
/* Check if it has to go via TCP or UDP... */
|
||||||
{
|
|
||||||
if(debug_lvl >= DEBUG_TRAFFIC)
|
|
||||||
syslog(LOG_INFO, _("No valid key known yet for %s (%s), queueing packet"),
|
|
||||||
cl->name, cl->hostname);
|
|
||||||
|
|
||||||
/* Since packet is on the stack of handle_tap_input(),
|
|
||||||
we have to make a copy of it first. */
|
|
||||||
|
|
||||||
copy = xmalloc(sizeof(vpn_packet_t));
|
|
||||||
memcpy(copy, packet, sizeof(vpn_packet_t));
|
|
||||||
|
|
||||||
list_insert_tail(cl->queue, copy);
|
|
||||||
|
|
||||||
if(!cl->status.waitingforkey)
|
|
||||||
send_req_key(myself, cl); /* Keys should be sent to the host running the tincd */
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Check if it has to go via UDP or TCP... */
|
|
||||||
cp
|
cp
|
||||||
if(cl->options & OPTION_TCPONLY)
|
if((cl->options | myself->options) & OPTION_TCPONLY)
|
||||||
return send_tcppacket(cl, packet);
|
{
|
||||||
|
if(send_tcppacket(cl, packet))
|
||||||
|
terminate_connection(cl);
|
||||||
|
}
|
||||||
else
|
else
|
||||||
return xsend(cl, packet);
|
send_udppacket(cl, packet);
|
||||||
}
|
}
|
||||||
|
|
||||||
void flush_queue(connection_t *cl)
|
void flush_queue(connection_t *cl)
|
||||||
|
@ -273,11 +255,11 @@ void flush_queue(connection_t *cl)
|
||||||
cp
|
cp
|
||||||
if(debug_lvl >= DEBUG_TRAFFIC)
|
if(debug_lvl >= DEBUG_TRAFFIC)
|
||||||
syslog(LOG_INFO, _("Flushing queue for %s (%s)"), cl->name, cl->hostname);
|
syslog(LOG_INFO, _("Flushing queue for %s (%s)"), cl->name, cl->hostname);
|
||||||
|
|
||||||
for(node = cl->queue->head; node; node = next)
|
for(node = cl->queue->head; node; node = next)
|
||||||
{
|
{
|
||||||
next = node->next;
|
next = node->next;
|
||||||
xsend(cl, (vpn_packet_t *)node->data);
|
send_udppacket(cl, (vpn_packet_t *)node->data);
|
||||||
list_delete_node(cl->queue, node);
|
list_delete_node(cl->queue, node);
|
||||||
}
|
}
|
||||||
cp
|
cp
|
||||||
|
@ -297,7 +279,7 @@ int setup_tap_fd(void)
|
||||||
# endif
|
# endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
cp
|
cp
|
||||||
if((cfg = get_config_val(config, config_tapdevice)))
|
if((cfg = get_config_val(config, config_tapdevice)))
|
||||||
tapfname = cfg->data.ptr;
|
tapfname = cfg->data.ptr;
|
||||||
else
|
else
|
||||||
|
@ -328,7 +310,7 @@ cp
|
||||||
taptype = TAP_TYPE_ETHERTAP;
|
taptype = TAP_TYPE_ETHERTAP;
|
||||||
|
|
||||||
/* Set default MAC address for ethertap devices */
|
/* Set default MAC address for ethertap devices */
|
||||||
|
|
||||||
mymac.type = SUBNET_MAC;
|
mymac.type = SUBNET_MAC;
|
||||||
mymac.net.mac.address.x[0] = 0xfe;
|
mymac.net.mac.address.x[0] = 0xfe;
|
||||||
mymac.net.mac.address.x[1] = 0xfd;
|
mymac.net.mac.address.x[1] = 0xfd;
|
||||||
|
@ -387,7 +369,7 @@ cp
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Optimize TCP settings */
|
/* Optimize TCP settings */
|
||||||
|
|
||||||
option = 1;
|
option = 1;
|
||||||
setsockopt(nfd, SOL_SOCKET, SO_REUSEADDR, &option, sizeof(option));
|
setsockopt(nfd, SOL_SOCKET, SO_REUSEADDR, &option, sizeof(option));
|
||||||
setsockopt(nfd, SOL_SOCKET, SO_KEEPALIVE, &option, sizeof(option));
|
setsockopt(nfd, SOL_SOCKET, SO_KEEPALIVE, &option, sizeof(option));
|
||||||
|
@ -395,7 +377,7 @@ cp
|
||||||
|
|
||||||
option = IPTOS_LOWDELAY;
|
option = IPTOS_LOWDELAY;
|
||||||
setsockopt(nfd, SOL_IP, IP_TOS, &option, sizeof(option));
|
setsockopt(nfd, SOL_IP, IP_TOS, &option, sizeof(option));
|
||||||
|
|
||||||
if((cfg = get_config_val(config, config_interface)))
|
if((cfg = get_config_val(config, config_interface)))
|
||||||
{
|
{
|
||||||
if(setsockopt(nfd, SOL_SOCKET, SO_BINDTODEVICE, cfg->data.ptr, strlen(cfg->data.ptr)))
|
if(setsockopt(nfd, SOL_SOCKET, SO_BINDTODEVICE, cfg->data.ptr, strlen(cfg->data.ptr)))
|
||||||
|
@ -409,7 +391,7 @@ cp
|
||||||
memset(&a, 0, sizeof(a));
|
memset(&a, 0, sizeof(a));
|
||||||
a.sin_family = AF_INET;
|
a.sin_family = AF_INET;
|
||||||
a.sin_port = htons(port);
|
a.sin_port = htons(port);
|
||||||
|
|
||||||
if((cfg = get_config_val(config, config_interfaceip)))
|
if((cfg = get_config_val(config, config_interfaceip)))
|
||||||
a.sin_addr.s_addr = htonl(cfg->data.ip->address);
|
a.sin_addr.s_addr = htonl(cfg->data.ip->address);
|
||||||
else
|
else
|
||||||
|
@ -516,16 +498,16 @@ cp
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Optimize TCP settings */
|
/* Optimize TCP settings */
|
||||||
|
|
||||||
option = 1;
|
option = 1;
|
||||||
setsockopt(cl->meta_socket, SOL_SOCKET, SO_KEEPALIVE, &option, sizeof(option));
|
setsockopt(cl->meta_socket, SOL_SOCKET, SO_KEEPALIVE, &option, sizeof(option));
|
||||||
setsockopt(cl->meta_socket, SOL_TCP, TCP_NODELAY, &option, sizeof(option));
|
setsockopt(cl->meta_socket, SOL_TCP, TCP_NODELAY, &option, sizeof(option));
|
||||||
|
|
||||||
option = IPTOS_LOWDELAY;
|
option = IPTOS_LOWDELAY;
|
||||||
setsockopt(cl->meta_socket, SOL_IP, IP_TOS, &option, sizeof(option));
|
setsockopt(cl->meta_socket, SOL_IP, IP_TOS, &option, sizeof(option));
|
||||||
|
|
||||||
/* Connect */
|
/* Connect */
|
||||||
|
|
||||||
a.sin_family = AF_INET;
|
a.sin_family = AF_INET;
|
||||||
a.sin_port = htons(cl->port);
|
a.sin_port = htons(cl->port);
|
||||||
a.sin_addr.s_addr = htonl(cl->address);
|
a.sin_addr.s_addr = htonl(cl->address);
|
||||||
|
@ -572,21 +554,21 @@ cp
|
||||||
|
|
||||||
ncn = new_connection();
|
ncn = new_connection();
|
||||||
asprintf(&ncn->name, "%s", name);
|
asprintf(&ncn->name, "%s", name);
|
||||||
|
|
||||||
if(read_host_config(ncn))
|
if(read_host_config(ncn))
|
||||||
{
|
{
|
||||||
syslog(LOG_ERR, _("Error reading host configuration file for %s"), ncn->name);
|
syslog(LOG_ERR, _("Error reading host configuration file for %s"), ncn->name);
|
||||||
free_connection(ncn);
|
free_connection(ncn);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!(cfg = get_config_val(ncn->config, config_address)))
|
if(!(cfg = get_config_val(ncn->config, config_address)))
|
||||||
{
|
{
|
||||||
syslog(LOG_ERR, _("No address specified for %s"), ncn->name);
|
syslog(LOG_ERR, _("No address specified for %s"), ncn->name);
|
||||||
free_connection(ncn);
|
free_connection(ncn);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!(h = gethostbyname(cfg->data.ptr)))
|
if(!(h = gethostbyname(cfg->data.ptr)))
|
||||||
{
|
{
|
||||||
syslog(LOG_ERR, _("Error looking up `%s': %m"), cfg->data.ptr);
|
syslog(LOG_ERR, _("Error looking up `%s': %m"), cfg->data.ptr);
|
||||||
|
@ -596,7 +578,7 @@ cp
|
||||||
|
|
||||||
ncn->address = ntohl(*((ip_t*)(h->h_addr_list[0])));
|
ncn->address = ntohl(*((ip_t*)(h->h_addr_list[0])));
|
||||||
ncn->hostname = hostlookup(htonl(ncn->address));
|
ncn->hostname = hostlookup(htonl(ncn->address));
|
||||||
|
|
||||||
if(setup_outgoing_meta_socket(ncn) < 0)
|
if(setup_outgoing_meta_socket(ncn) < 0)
|
||||||
{
|
{
|
||||||
syslog(LOG_ERR, _("Could not set up a meta connection to %s"),
|
syslog(LOG_ERR, _("Could not set up a meta connection to %s"),
|
||||||
|
@ -660,10 +642,10 @@ cp
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Else, check if a harnessed public key is in the config file */
|
/* Else, check if a harnessed public key is in the config file */
|
||||||
|
|
||||||
asprintf(&fname, "%s/hosts/%s", confbase, cl->name);
|
asprintf(&fname, "%s/hosts/%s", confbase, cl->name);
|
||||||
if((fp = fopen(fname, "r")))
|
if((fp = fopen(fname, "r")))
|
||||||
{
|
{
|
||||||
|
@ -713,7 +695,7 @@ cp
|
||||||
cfg->data.ptr);
|
cfg->data.ptr);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
syslog(LOG_ERR, _("No private key for tinc daemon specified!"));
|
syslog(LOG_ERR, _("No private key for tinc daemon specified!"));
|
||||||
|
@ -735,7 +717,7 @@ cp
|
||||||
myself = new_connection();
|
myself = new_connection();
|
||||||
|
|
||||||
asprintf(&myself->hostname, "MYSELF");
|
asprintf(&myself->hostname, "MYSELF");
|
||||||
myself->flags = 0;
|
myself->options = 0;
|
||||||
myself->protocol_version = PROT_CURRENT;
|
myself->protocol_version = PROT_CURRENT;
|
||||||
|
|
||||||
if(!(cfg = get_config_val(config, config_name))) /* Not acceptable */
|
if(!(cfg = get_config_val(config, config_name))) /* Not acceptable */
|
||||||
|
@ -763,7 +745,7 @@ cp
|
||||||
|
|
||||||
if(read_rsa_public_key(myself))
|
if(read_rsa_public_key(myself))
|
||||||
return -1;
|
return -1;
|
||||||
cp
|
cp
|
||||||
|
|
||||||
/*
|
/*
|
||||||
if(RSA_check_key(myself->rsa_key) != 1)
|
if(RSA_check_key(myself->rsa_key) != 1)
|
||||||
|
@ -779,11 +761,11 @@ cp
|
||||||
|
|
||||||
if((cfg = get_config_val(myself->config, config_indirectdata)))
|
if((cfg = get_config_val(myself->config, config_indirectdata)))
|
||||||
if(cfg->data.val == stupid_true)
|
if(cfg->data.val == stupid_true)
|
||||||
myself->flags |= EXPORTINDIRECTDATA;
|
myself->options |= OPTION_INDIRECT;
|
||||||
|
|
||||||
if((cfg = get_config_val(myself->config, config_tcponly)))
|
if((cfg = get_config_val(myself->config, config_tcponly)))
|
||||||
if(cfg->data.val == stupid_true)
|
if(cfg->data.val == stupid_true)
|
||||||
myself->flags |= TCPONLY;
|
myself->options |= OPTION_TCPONLY;
|
||||||
|
|
||||||
/* Read in all the subnets specified in the host configuration file */
|
/* Read in all the subnets specified in the host configuration file */
|
||||||
|
|
||||||
|
@ -793,18 +775,18 @@ cp
|
||||||
net->type = SUBNET_IPV4;
|
net->type = SUBNET_IPV4;
|
||||||
net->net.ipv4.address = cfg->data.ip->address;
|
net->net.ipv4.address = cfg->data.ip->address;
|
||||||
net->net.ipv4.mask = cfg->data.ip->mask;
|
net->net.ipv4.mask = cfg->data.ip->mask;
|
||||||
|
|
||||||
/* Teach newbies what subnets are... */
|
/* Teach newbies what subnets are... */
|
||||||
|
|
||||||
if((net->net.ipv4.address & net->net.ipv4.mask) != net->net.ipv4.address)
|
if((net->net.ipv4.address & net->net.ipv4.mask) != net->net.ipv4.address)
|
||||||
{
|
{
|
||||||
syslog(LOG_ERR, _("Network address and subnet mask do not match!"));
|
syslog(LOG_ERR, _("Network address and subnet mask do not match!"));
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
subnet_add(myself, net);
|
subnet_add(myself, net);
|
||||||
}
|
}
|
||||||
|
|
||||||
if((myself->meta_socket = setup_listen_meta_socket(myself->port)) < 0)
|
if((myself->meta_socket = setup_listen_meta_socket(myself->port)) < 0)
|
||||||
{
|
{
|
||||||
syslog(LOG_ERR, _("Unable to set up a listening TCP socket!"));
|
syslog(LOG_ERR, _("Unable to set up a listening TCP socket!"));
|
||||||
|
@ -830,11 +812,11 @@ cp
|
||||||
keylifetime = 3600;
|
keylifetime = 3600;
|
||||||
else
|
else
|
||||||
keylifetime = cfg->data.val;
|
keylifetime = cfg->data.val;
|
||||||
|
|
||||||
keyexpires = time(NULL) + keylifetime;
|
keyexpires = time(NULL) + keylifetime;
|
||||||
cp
|
cp
|
||||||
/* Check some options */
|
/* Check some options */
|
||||||
|
|
||||||
if((cfg = get_config_val(config, config_indirectdata)))
|
if((cfg = get_config_val(config, config_indirectdata)))
|
||||||
{
|
{
|
||||||
if(cfg->data.val == stupid_true)
|
if(cfg->data.val == stupid_true)
|
||||||
|
@ -846,6 +828,10 @@ cp
|
||||||
if(cfg->data.val == stupid_true)
|
if(cfg->data.val == stupid_true)
|
||||||
myself->options |= OPTION_TCPONLY;
|
myself->options |= OPTION_TCPONLY;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(myself->options & OPTION_TCPONLY)
|
||||||
|
myself->options |= OPTION_INDIRECT;
|
||||||
|
|
||||||
/* Activate ourselves */
|
/* Activate ourselves */
|
||||||
|
|
||||||
myself->status.active = 1;
|
myself->status.active = 1;
|
||||||
|
@ -914,7 +900,7 @@ cp
|
||||||
|
|
||||||
/* Run tinc-up script to further initialize the tap interface */
|
/* Run tinc-up script to further initialize the tap interface */
|
||||||
execute_script("tinc-up");
|
execute_script("tinc-up");
|
||||||
|
|
||||||
if(setup_myself() < 0)
|
if(setup_myself() < 0)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
|
@ -929,7 +915,7 @@ cp
|
||||||
return 0;
|
return 0;
|
||||||
cfg = get_config_val(upstreamcfg, config_connectto); /* Or else we try the next ConnectTo line */
|
cfg = get_config_val(upstreamcfg, config_connectto); /* Or else we try the next ConnectTo line */
|
||||||
}
|
}
|
||||||
|
|
||||||
if(do_detach)
|
if(do_detach)
|
||||||
{
|
{
|
||||||
signal(SIGALRM, sigalrm_handler);
|
signal(SIGALRM, sigalrm_handler);
|
||||||
|
@ -1008,7 +994,7 @@ cp
|
||||||
p->buffer = xmalloc(MAXBUFSIZE);
|
p->buffer = xmalloc(MAXBUFSIZE);
|
||||||
p->buflen = 0;
|
p->buflen = 0;
|
||||||
p->last_ping_time = time(NULL);
|
p->last_ping_time = time(NULL);
|
||||||
|
|
||||||
if(debug_lvl >= DEBUG_CONNECTIONS)
|
if(debug_lvl >= DEBUG_CONNECTIONS)
|
||||||
syslog(LOG_NOTICE, _("Connection from %s port %d"),
|
syslog(LOG_NOTICE, _("Connection from %s port %d"),
|
||||||
p->hostname, htons(ci.sin_port));
|
p->hostname, htons(ci.sin_port));
|
||||||
|
@ -1047,7 +1033,7 @@ cp
|
||||||
udp socket and write it to the ethertap
|
udp socket and write it to the ethertap
|
||||||
device after being decrypted
|
device after being decrypted
|
||||||
*/
|
*/
|
||||||
int handle_incoming_vpn_data(void)
|
void handle_incoming_vpn_data(void)
|
||||||
{
|
{
|
||||||
vpn_packet_t pkt;
|
vpn_packet_t pkt;
|
||||||
int x, l = sizeof(x);
|
int x, l = sizeof(x);
|
||||||
|
@ -1060,36 +1046,30 @@ cp
|
||||||
{
|
{
|
||||||
syslog(LOG_ERR, _("This is a bug: %s:%d: %d:%m"),
|
syslog(LOG_ERR, _("This is a bug: %s:%d: %d:%m"),
|
||||||
__FILE__, __LINE__, myself->socket);
|
__FILE__, __LINE__, myself->socket);
|
||||||
return -1;
|
return;
|
||||||
}
|
}
|
||||||
if(x)
|
if(x)
|
||||||
{
|
{
|
||||||
syslog(LOG_ERR, _("Incoming data socket error: %s"), strerror(x));
|
syslog(LOG_ERR, _("Incoming data socket error: %s"), strerror(x));
|
||||||
return -1;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if((lenin = recvfrom(myself->socket, (char *) &(pkt.len), MTU, 0, (struct sockaddr *)&from, &fromlen)) <= 0)
|
if((lenin = recvfrom(myself->socket, (char *) &(pkt.len), MTU, 0, (struct sockaddr *)&from, &fromlen)) <= 0)
|
||||||
{
|
{
|
||||||
syslog(LOG_ERR, _("Receiving packet failed: %m"));
|
syslog(LOG_ERR, _("Receiving packet failed: %m"));
|
||||||
return -1;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
cl = lookup_connection(ntohl(from.sin_addr.s_addr), ntohs(from.sin_port));
|
cl = lookup_connection(ntohl(from.sin_addr.s_addr), ntohs(from.sin_port));
|
||||||
|
|
||||||
if(!cl)
|
if(!cl)
|
||||||
{
|
{
|
||||||
syslog(LOG_WARNING, _("Received UDP packets on port %hd from unknown source %x:%hd"), myself->port, ntohl(from.sin_addr.s_addr), ntohs(from.sin_port));
|
syslog(LOG_WARNING, _("Received UDP packets on port %hd from unknown source %x:%hd"), myself->port, ntohl(from.sin_addr.s_addr), ntohs(from.sin_port));
|
||||||
return 0;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
if(debug_lvl >= DEBUG_TRAFFIC)
|
|
||||||
{
|
|
||||||
syslog(LOG_DEBUG, _("Received packet of %d bytes from %s (%s)"), lenin,
|
|
||||||
cl->name, cl->hostname);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
receive_udppacket(cl, &pkt);
|
||||||
cp
|
cp
|
||||||
return xrecv(cl, &pkt);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -1108,9 +1088,9 @@ cp
|
||||||
if(debug_lvl >= DEBUG_CONNECTIONS)
|
if(debug_lvl >= DEBUG_CONNECTIONS)
|
||||||
syslog(LOG_NOTICE, _("Closing connection with %s (%s)"),
|
syslog(LOG_NOTICE, _("Closing connection with %s (%s)"),
|
||||||
cl->name, cl->hostname);
|
cl->name, cl->hostname);
|
||||||
|
|
||||||
cl->status.remove = 1;
|
cl->status.remove = 1;
|
||||||
|
|
||||||
if(cl->socket)
|
if(cl->socket)
|
||||||
close(cl->socket);
|
close(cl->socket);
|
||||||
if(cl->status.meta)
|
if(cl->status.meta)
|
||||||
|
@ -1118,7 +1098,7 @@ cp
|
||||||
|
|
||||||
if(cl->status.meta)
|
if(cl->status.meta)
|
||||||
{
|
{
|
||||||
|
|
||||||
/* Find all connections that were lost because they were behind cl
|
/* Find all connections that were lost because they were behind cl
|
||||||
(the connection that was dropped). */
|
(the connection that was dropped). */
|
||||||
|
|
||||||
|
@ -1150,7 +1130,7 @@ cp
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check if this was our outgoing connection */
|
/* Check if this was our outgoing connection */
|
||||||
|
|
||||||
if(cl->status.outgoing)
|
if(cl->status.outgoing)
|
||||||
{
|
{
|
||||||
cl->status.outgoing = 0;
|
cl->status.outgoing = 0;
|
||||||
|
@ -1232,7 +1212,7 @@ cp
|
||||||
}
|
}
|
||||||
|
|
||||||
connection_add(ncn);
|
connection_add(ncn);
|
||||||
|
|
||||||
send_id(ncn);
|
send_id(ncn);
|
||||||
cp
|
cp
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -1263,9 +1243,9 @@ cp
|
||||||
{
|
{
|
||||||
terminate_connection(p);
|
terminate_connection(p);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if(FD_ISSET(myself->meta_socket, f))
|
if(FD_ISSET(myself->meta_socket, f))
|
||||||
handle_new_meta_connection();
|
handle_new_meta_connection();
|
||||||
cp
|
cp
|
||||||
|
@ -1279,7 +1259,7 @@ void handle_tap_input(void)
|
||||||
{
|
{
|
||||||
vpn_packet_t vp;
|
vpn_packet_t vp;
|
||||||
int lenin;
|
int lenin;
|
||||||
cp
|
cp
|
||||||
if(taptype == TAP_TYPE_TUNTAP)
|
if(taptype == TAP_TYPE_TUNTAP)
|
||||||
{
|
{
|
||||||
if((lenin = read(tap_fd, vp.data, MTU)) <= 0)
|
if((lenin = read(tap_fd, vp.data, MTU)) <= 0)
|
||||||
|
@ -1313,7 +1293,7 @@ cp
|
||||||
syslog(LOG_DEBUG, _("Read packet of length %d from tap device"), vp.len);
|
syslog(LOG_DEBUG, _("Read packet of length %d from tap device"), vp.len);
|
||||||
}
|
}
|
||||||
|
|
||||||
send_packet(ntohl(*((unsigned long*)(&vp.data[30]))), &vp);
|
route_outgoing(&vp);
|
||||||
cp
|
cp
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1361,10 +1341,10 @@ cp
|
||||||
}
|
}
|
||||||
|
|
||||||
sleep(5);
|
sleep(5);
|
||||||
|
|
||||||
if(setup_network_connections())
|
if(setup_network_connections())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1383,7 +1363,7 @@ cp
|
||||||
{
|
{
|
||||||
if(debug_lvl >= DEBUG_STATUS)
|
if(debug_lvl >= DEBUG_STATUS)
|
||||||
syslog(LOG_INFO, _("Regenerating symmetric key"));
|
syslog(LOG_INFO, _("Regenerating symmetric key"));
|
||||||
|
|
||||||
RAND_bytes(myself->cipher_pktkey, myself->cipher_pktkeylength);
|
RAND_bytes(myself->cipher_pktkey, myself->cipher_pktkeylength);
|
||||||
send_key_changed(myself, NULL);
|
send_key_changed(myself, NULL);
|
||||||
keyexpires = time(NULL) + keylifetime;
|
keyexpires = time(NULL) + keylifetime;
|
||||||
|
|
12
src/net.h
12
src/net.h
|
@ -17,7 +17,7 @@
|
||||||
along with this program; if not, write to the Free Software
|
along with this program; if not, write to the Free Software
|
||||||
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||||
|
|
||||||
$Id: net.h,v 1.9.4.28 2001/02/27 16:37:28 guus Exp $
|
$Id: net.h,v 1.9.4.29 2001/03/04 13:59:28 guus Exp $
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef __TINC_NET_H__
|
#ifndef __TINC_NET_H__
|
||||||
|
@ -47,11 +47,6 @@
|
||||||
|
|
||||||
#define MAXBUFSIZE 4096 /* Probably way too much, but it must fit every possible request. */
|
#define MAXBUFSIZE 4096 /* Probably way too much, but it must fit every possible request. */
|
||||||
|
|
||||||
/* flags */
|
|
||||||
#define INDIRECTDATA 0x0001 /* Used to indicate that this host has to be reached indirect */
|
|
||||||
#define EXPORTINDIRECTDATA 0x0002 /* Used to indicate uplink that it has to tell others to do INDIRECTDATA */
|
|
||||||
#define TCPONLY 0x0004 /* Tells sender to send packets over TCP instead of UDP (for firewalls) */
|
|
||||||
|
|
||||||
/* tap types */
|
/* tap types */
|
||||||
#define TAP_TYPE_ETHERTAP 0
|
#define TAP_TYPE_ETHERTAP 0
|
||||||
#define TAP_TYPE_TUNTAP 1
|
#define TAP_TYPE_TUNTAP 1
|
||||||
|
@ -112,8 +107,9 @@ extern char *status_text[10];
|
||||||
|
|
||||||
extern int str2opt(const char *);
|
extern int str2opt(const char *);
|
||||||
extern char *opt2str(int);
|
extern char *opt2str(int);
|
||||||
extern int send_packet(ip_t, vpn_packet_t *);
|
extern void send_packet(connection_t *, vpn_packet_t *);
|
||||||
extern int receive_packet(connection_t *, vpn_packet_t *);
|
extern void receive_packet(connection_t *, vpn_packet_t *);
|
||||||
|
extern void accept_packet(vpn_packet_t *);
|
||||||
extern int setup_network_connections(void);
|
extern int setup_network_connections(void);
|
||||||
extern void close_network_connections(void);
|
extern void close_network_connections(void);
|
||||||
extern void main_loop(void);
|
extern void main_loop(void);
|
||||||
|
|
|
@ -17,7 +17,7 @@
|
||||||
along with this program; if not, write to the Free Software
|
along with this program; if not, write to the Free Software
|
||||||
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||||
|
|
||||||
$Id: protocol.c,v 1.28.4.84 2001/03/02 11:25:56 guus Exp $
|
$Id: protocol.c,v 1.28.4.85 2001/03/04 13:59:28 guus Exp $
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
|
@ -119,7 +119,7 @@ int receive_request(connection_t *cl)
|
||||||
cp
|
cp
|
||||||
if(sscanf(cl->buffer, "%d", &request) == 1)
|
if(sscanf(cl->buffer, "%d", &request) == 1)
|
||||||
{
|
{
|
||||||
if((request < 0) || (request > 255) || (request_handlers[request] == NULL))
|
if((request < 0) || (request >= LAST) || (request_handlers[request] == NULL))
|
||||||
{
|
{
|
||||||
syslog(LOG_ERR, _("Unknown request from %s (%s)"),
|
syslog(LOG_ERR, _("Unknown request from %s (%s)"),
|
||||||
cl->name, cl->hostname);
|
cl->name, cl->hostname);
|
||||||
|
@ -1264,7 +1264,7 @@ cp
|
||||||
int send_tcppacket(connection_t *cl, vpn_packet_t *packet)
|
int send_tcppacket(connection_t *cl, vpn_packet_t *packet)
|
||||||
{
|
{
|
||||||
int x;
|
int x;
|
||||||
|
cp
|
||||||
x = send_request(cl->nexthop, "%d %hd", PACKET, packet->len);
|
x = send_request(cl->nexthop, "%d %hd", PACKET, packet->len);
|
||||||
|
|
||||||
if(x)
|
if(x)
|
||||||
|
@ -1278,8 +1278,8 @@ int tcppacket_h(connection_t *cl)
|
||||||
vpn_packet_t packet;
|
vpn_packet_t packet;
|
||||||
char *p;
|
char *p;
|
||||||
int todo, x;
|
int todo, x;
|
||||||
|
cp
|
||||||
if(sscanf(cl->buffer, "%*d %hd", packet.len) != 1)
|
if(sscanf(cl->buffer, "%*d %hd", &packet.len) != 1)
|
||||||
{
|
{
|
||||||
syslog(LOG_ERR, _("Got bad PACKET from %s (%s)"), cl->name, cl->hostname);
|
syslog(LOG_ERR, _("Got bad PACKET from %s (%s)"), cl->name, cl->hostname);
|
||||||
return -1;
|
return -1;
|
||||||
|
@ -1289,7 +1289,7 @@ int tcppacket_h(connection_t *cl)
|
||||||
|
|
||||||
p = packet.data;
|
p = packet.data;
|
||||||
todo = packet.len;
|
todo = packet.len;
|
||||||
|
|
||||||
while(todo)
|
while(todo)
|
||||||
{
|
{
|
||||||
x = read(cl->meta_socket, p, todo);
|
x = read(cl->meta_socket, p, todo);
|
||||||
|
@ -1299,7 +1299,7 @@ int tcppacket_h(connection_t *cl)
|
||||||
if(x==0)
|
if(x==0)
|
||||||
syslog(LOG_NOTICE, _("Connection closed by %s (%s)"), cl->name, cl->hostname);
|
syslog(LOG_NOTICE, _("Connection closed by %s (%s)"), cl->name, cl->hostname);
|
||||||
else
|
else
|
||||||
if(errno==EINTR)
|
if(errno==EINTR || errno==EAGAIN) /* FIXME: select() or poll() or reimplement this evil hack */
|
||||||
continue;
|
continue;
|
||||||
else
|
else
|
||||||
syslog(LOG_ERR, _("Error during reception of PACKET from %s (%s): %m"), cl->name, cl->hostname);
|
syslog(LOG_ERR, _("Error during reception of PACKET from %s (%s): %m"), cl->name, cl->hostname);
|
||||||
|
@ -1311,7 +1311,9 @@ int tcppacket_h(connection_t *cl)
|
||||||
p += x;
|
p += x;
|
||||||
}
|
}
|
||||||
|
|
||||||
return receive_packet(cl, &packet);
|
receive_packet(cl, &packet);
|
||||||
|
cp
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Jumptable for the request handlers */
|
/* Jumptable for the request handlers */
|
||||||
|
|
89
src/route.c
89
src/route.c
|
@ -17,11 +17,12 @@
|
||||||
along with this program; if not, write to the Free Software
|
along with this program; if not, write to the Free Software
|
||||||
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||||
|
|
||||||
$Id: route.c,v 1.1.2.6 2001/01/07 17:09:06 guus Exp $
|
$Id: route.c,v 1.1.2.7 2001/03/04 13:59:32 guus Exp $
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
|
|
||||||
|
#include <netinet/in.h>
|
||||||
#include <utils.h>
|
#include <utils.h>
|
||||||
#include <xalloc.h>
|
#include <xalloc.h>
|
||||||
#include <syslog.h>
|
#include <syslog.h>
|
||||||
|
@ -34,15 +35,15 @@
|
||||||
#include "system.h"
|
#include "system.h"
|
||||||
|
|
||||||
int routing_mode = RMODE_ROUTER;
|
int routing_mode = RMODE_ROUTER;
|
||||||
|
subnet_t mymac;
|
||||||
|
|
||||||
void learn_mac(connection_t *source, mac_t *address)
|
void learn_mac(connection_t *source, mac_t *address)
|
||||||
{
|
{
|
||||||
connection_t *old;
|
|
||||||
subnet_t *subnet;
|
subnet_t *subnet;
|
||||||
cp
|
cp
|
||||||
old = lookup_subnet_mac(address)->owner;
|
subnet = lookup_subnet_mac(address);
|
||||||
|
|
||||||
if(!old)
|
if(!subnet)
|
||||||
{
|
{
|
||||||
subnet = new_subnet();
|
subnet = new_subnet();
|
||||||
subnet->type = SUBNET_MAC;
|
subnet->type = SUBNET_MAC;
|
||||||
|
@ -50,23 +51,22 @@ cp
|
||||||
memcpy(&subnet->net.mac.address, address, sizeof(mac_t));
|
memcpy(&subnet->net.mac.address, address, sizeof(mac_t));
|
||||||
subnet_add(source, subnet);
|
subnet_add(source, subnet);
|
||||||
|
|
||||||
if(DEBUG_LVL >= DEBUG_TRAFFIC)
|
if(debug_lvl >= DEBUG_TRAFFIC)
|
||||||
{
|
{
|
||||||
syslog(LOG_DEBUG, _("Learned new MAC address %x:%x:%x:%x:%x:%x from %s (%s)"),
|
syslog(LOG_DEBUG, _("Learned new MAC address %x:%x:%x:%x:%x:%x from %s (%s)"),
|
||||||
address->address.x[0],
|
address->x[0],
|
||||||
address->address.x[1],
|
address->x[1],
|
||||||
address->address.x[2],
|
address->x[2],
|
||||||
address->address.x[3],
|
address->x[3],
|
||||||
address->address.x[4],
|
address->x[4],
|
||||||
address->address.x[5],
|
address->x[5],
|
||||||
cl->name, cl->hostname);
|
source->name, source->hostname);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
connection_t *route_mac(connection_t *source, vpn_packet_t *packet)
|
connection_t *route_mac(connection_t *source, vpn_packet_t *packet)
|
||||||
{
|
{
|
||||||
connection_t *oldsrc, *dst;
|
|
||||||
subnet_t *subnet;
|
subnet_t *subnet;
|
||||||
cp
|
cp
|
||||||
/* Learn source address */
|
/* Learn source address */
|
||||||
|
@ -75,39 +75,47 @@ cp
|
||||||
|
|
||||||
/* Lookup destination address */
|
/* Lookup destination address */
|
||||||
|
|
||||||
dst = lookup_subnet_mac((mac_t *)(&packet->data[6]))->owner;
|
subnet = lookup_subnet_mac((mac_t *)(&packet->data[6]));
|
||||||
|
|
||||||
if(!dst)
|
if(!subnet)
|
||||||
if(debug_lvl >= DEBUG_TRAFFIC)
|
{
|
||||||
{
|
if(debug_lvl >= DEBUG_TRAFFIC)
|
||||||
syslog(LOG_WARNING, _("Cannot route packet: unknown destination address %x:%x:%x:%x:%x:%x"),
|
{
|
||||||
packet->data[6],
|
syslog(LOG_WARNING, _("Cannot route packet: unknown destination address %x:%x:%x:%x:%x:%x"),
|
||||||
packet->data[7],
|
packet->data[6],
|
||||||
packet->data[8],
|
packet->data[7],
|
||||||
packet->data[9],
|
packet->data[8],
|
||||||
packet->data[10],
|
packet->data[9],
|
||||||
packet->data[11]);
|
packet->data[10],
|
||||||
}
|
packet->data[11]);
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
cp
|
cp
|
||||||
return dst;
|
return subnet->owner;
|
||||||
}
|
}
|
||||||
|
|
||||||
connection_t *route_ipv4(vpn_packet_t *packet)
|
connection_t *route_ipv4(vpn_packet_t *packet)
|
||||||
{
|
{
|
||||||
ipv4_t dest;
|
ipv4_t dest;
|
||||||
connection_t *cl;
|
subnet_t *subnet;
|
||||||
cp
|
cp
|
||||||
dest = ntohl(*((unsigned long*)(&packet->data[30])));
|
dest = ntohl(*((unsigned long*)(&packet->data[30])));
|
||||||
|
|
||||||
cl = lookup_subnet_ipv4(&dest)->owner;
|
subnet = lookup_subnet_ipv4(&dest);
|
||||||
if(!cl)
|
|
||||||
if(debug_lvl >= DEBUG_TRAFFIC)
|
if(!subnet)
|
||||||
{
|
{
|
||||||
syslog(LOG_WARNING, _("Cannot route packet: unknown destination address %d.%d.%d.%d"),
|
if(debug_lvl >= DEBUG_TRAFFIC)
|
||||||
packet->data[30], packet->data[31], packet->data[32], packet->data[33]);
|
{
|
||||||
}
|
syslog(LOG_WARNING, _("Cannot route packet: unknown destination address %d.%d.%d.%d"),
|
||||||
|
packet->data[30], packet->data[31], packet->data[32], packet->data[33]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
cp
|
cp
|
||||||
return cl;
|
return subnet->owner;
|
||||||
}
|
}
|
||||||
|
|
||||||
connection_t *route_ipv6(vpn_packet_t *packet)
|
connection_t *route_ipv6(vpn_packet_t *packet)
|
||||||
|
@ -124,7 +132,7 @@ cp
|
||||||
void route_outgoing(vpn_packet_t *packet)
|
void route_outgoing(vpn_packet_t *packet)
|
||||||
{
|
{
|
||||||
unsigned short int type;
|
unsigned short int type;
|
||||||
avl_tree_t *node;
|
avl_node_t *node;
|
||||||
connection_t *cl;
|
connection_t *cl;
|
||||||
cp
|
cp
|
||||||
/* FIXME: multicast? */
|
/* FIXME: multicast? */
|
||||||
|
@ -148,11 +156,12 @@ cp
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
send_packet(cl, packet);
|
if(cl)
|
||||||
|
send_packet(cl, packet);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case RMODE_SWITCH:
|
case RMODE_SWITCH:
|
||||||
cl = route_mac(packet);
|
cl = route_mac(myself, packet);
|
||||||
if(cl)
|
if(cl)
|
||||||
send_packet(cl, packet);
|
send_packet(cl, packet);
|
||||||
break;
|
break;
|
||||||
|
@ -173,8 +182,10 @@ void route_incoming(connection_t *source, vpn_packet_t *packet)
|
||||||
switch(routing_mode)
|
switch(routing_mode)
|
||||||
{
|
{
|
||||||
case RMODE_SWITCH:
|
case RMODE_SWITCH:
|
||||||
learn_mac(source, &packet->data[0]);
|
learn_mac(source, (mac_t *)(&packet->data[0]));
|
||||||
break;
|
break;
|
||||||
|
case RMODE_ROUTER:
|
||||||
|
memcpy(packet->data, mymac.net.mac.address.x, 6);
|
||||||
}
|
}
|
||||||
|
|
||||||
accept_packet(packet);
|
accept_packet(packet);
|
||||||
|
|
|
@ -17,7 +17,7 @@
|
||||||
along with this program; if not, write to the Free Software
|
along with this program; if not, write to the Free Software
|
||||||
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||||
|
|
||||||
$Id: route.h,v 1.1.2.2 2001/01/07 17:09:06 guus Exp $
|
$Id: route.h,v 1.1.2.3 2001/03/04 13:59:32 guus Exp $
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef __TINC_ROUTE_H__
|
#ifndef __TINC_ROUTE_H__
|
||||||
|
@ -31,8 +31,9 @@ enum
|
||||||
};
|
};
|
||||||
|
|
||||||
extern int routing_mode;
|
extern int routing_mode;
|
||||||
|
extern subnet_t mymac;
|
||||||
|
|
||||||
extern connection_t *route_incoming(connection_t *, vpn_packet_t *);
|
extern void route_incoming(connection_t *, vpn_packet_t *);
|
||||||
extern connection_t *route_outgoing(connection_t *, vpn_packet_t *);
|
extern void route_outgoing(vpn_packet_t *);
|
||||||
|
|
||||||
#endif /* __TINC_ROUTE_H__ */
|
#endif /* __TINC_ROUTE_H__ */
|
||||||
|
|
Loading…
Reference in a new issue