strongswan/src/libstrongswan/utils/host.c

538 lines
9.9 KiB
C
Raw Normal View History

2005-11-16 16:11:08 +00:00
/*
2007-02-28 14:04:36 +00:00
* Copyright (C) 2006-2007 Tobias Brunner
* Copyright (C) 2006 Daniel Roethlisberger
2006-07-07 08:49:06 +00:00
* Copyright (C) 2005-2006 Martin Willi
* Copyright (C) 2005 Jan Hutter
2005-11-16 16:11:08 +00:00
* Hochschule fuer Technik Rapperswil
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation; either version 2 of the License, or (at your
* option) any later version. See <http://www.fsf.org/copyleft/gpl.txt>.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* for more details.
*
* $Id$
2005-11-16 16:11:08 +00:00
*/
#include <string.h>
2006-09-27 14:15:49 +00:00
#include <printf.h>
2005-11-16 16:11:08 +00:00
#include "host.h"
2005-11-16 16:11:08 +00:00
2005-11-24 09:17:51 +00:00
typedef struct private_host_t private_host_t;
2005-11-16 16:11:08 +00:00
/**
* Private Data of a host object.
2005-11-16 16:11:08 +00:00
*/
2005-11-24 09:17:51 +00:00
struct private_host_t {
2005-11-16 16:11:08 +00:00
/**
* Public data
*/
host_t public;
/**
* low-lewel structure, wich stores the address
*/
union {
/** generic type */
struct sockaddr address;
2007-02-28 14:04:36 +00:00
/** maximum sockaddr size */
struct sockaddr_storage address_max;
/** IPv4 address */
struct sockaddr_in address4;
/** IPv6 address */
struct sockaddr_in6 address6;
};
2005-11-16 16:11:08 +00:00
/**
* length of address structure
*/
socklen_t socklen;
};
2005-11-21 11:45:04 +00:00
/**
* implements host_t.get_sockaddr
*/
static sockaddr_t *get_sockaddr(private_host_t *this)
2005-11-16 16:11:08 +00:00
{
return &(this->address);
}
2005-11-21 11:45:04 +00:00
/**
* implements host_t.get_sockaddr_len
*/
static socklen_t *get_sockaddr_len(private_host_t *this)
2005-11-16 16:11:08 +00:00
{
return &(this->socklen);
}
/**
* Implementation of host_t.is_anyaddr.
*/
static bool is_anyaddr(private_host_t *this)
{
switch (this->address.sa_family)
{
case AF_INET:
{
u_int8_t default_route[4];
memset(default_route, 0, sizeof(default_route));
return memeq(default_route, &(this->address4.sin_addr.s_addr),
sizeof(default_route));
}
case AF_INET6:
{
u_int8_t default_route[16];
memset(default_route, 0, sizeof(default_route));
return memeq(default_route, &(this->address6.sin6_addr.s6_addr),
sizeof(default_route));
}
default:
{
return FALSE;
}
}
}
2005-11-16 16:11:08 +00:00
/**
2006-09-27 14:15:49 +00:00
* output handler in printf()
*/
2006-09-27 14:15:49 +00:00
static int print(FILE *stream, const struct printf_info *info,
const void *const *args)
{
2006-09-27 14:15:49 +00:00
private_host_t *this = *((private_host_t**)(args[0]));
char buffer[INET6_ADDRSTRLEN + 16];
int len;
2006-09-27 14:15:49 +00:00
if (this == NULL)
{
len = sprintf(buffer, "(null)");
2006-09-27 14:15:49 +00:00
}
else if (is_anyaddr(this))
2006-09-27 14:15:49 +00:00
{
len = sprintf(buffer, "%%any");
2006-09-27 14:15:49 +00:00
}
else
{
void *address;
u_int16_t port;
address = &this->address6.sin6_addr;
port = this->address6.sin6_port;
switch (this->address.sa_family)
{
case AF_INET:
address = &this->address4.sin_addr;
port = this->address4.sin_port;
/* fall */
case AF_INET6:
if (inet_ntop(this->address.sa_family, address,
buffer, sizeof(buffer)) == NULL)
{
len = sprintf(buffer, "(address conversion failed)");
}
else if (info->alt)
{
len = sprintf(buffer, "%s[%d]", buffer, ntohs(port));
}
break;
default:
len = sprintf(buffer, "(family not supported)");
break;
}
2005-11-21 11:45:04 +00:00
}
return fprintf(stream, "%*s", info->width, buffer);
2005-11-21 11:45:04 +00:00
}
2006-09-27 14:15:49 +00:00
/**
* arginfo handler for printf() hosts
2006-09-27 14:15:49 +00:00
*/
int arginfo(const struct printf_info *info, size_t n, int *argtypes)
2006-09-27 14:15:49 +00:00
{
if (n > 0)
{
argtypes[0] = PA_POINTER;
}
return 1;
}
/**
* return printf hook functions for a host
*/
printf_hook_functions_t host_get_printf_hooks()
{
printf_hook_functions_t hooks = {print, arginfo};
return hooks;
2006-09-27 14:15:49 +00:00
}
2005-11-29 15:23:04 +00:00
/**
* Implementation of host_t.get_address.
2005-11-29 15:23:04 +00:00
*/
static chunk_t get_address(private_host_t *this)
2005-11-29 15:23:04 +00:00
{
chunk_t address = chunk_empty;
2005-11-29 15:23:04 +00:00
switch (this->address.sa_family)
2005-11-29 15:23:04 +00:00
{
case AF_INET:
2005-11-29 15:23:04 +00:00
{
address.ptr = (char*)&(this->address4.sin_addr.s_addr);
2005-11-29 15:23:04 +00:00
address.len = 4;
return address;
}
case AF_INET6:
{
address.ptr = (char*)&(this->address6.sin6_addr.s6_addr);
address.len = 16;
return address;
2005-11-29 15:23:04 +00:00
}
default:
{
/* return empty chunk */
2005-11-29 15:23:04 +00:00
return address;
}
}
}
/**
* implements host_t.get_family
*/
static int get_family(private_host_t *this)
{
return this->address.sa_family;
2005-11-29 15:23:04 +00:00
}
2005-11-21 11:45:04 +00:00
/**
* implements host_t.get_port
*/
static u_int16_t get_port(private_host_t *this)
{
switch (this->address.sa_family)
2005-11-21 11:45:04 +00:00
{
case AF_INET:
2005-11-21 11:45:04 +00:00
{
return ntohs(this->address4.sin_port);
2005-11-21 11:45:04 +00:00
}
case AF_INET6:
{
return ntohs(this->address6.sin6_port);
}
2005-11-21 11:45:04 +00:00
default:
{
return 0;
}
}
}
2006-06-22 06:36:28 +00:00
/**
* implements host_t.set_port
*/
static void set_port(private_host_t *this, u_int16_t port)
{
switch (this->address.sa_family)
2006-06-22 06:36:28 +00:00
{
case AF_INET:
{
this->address4.sin_port = htons(port);
break;
}
case AF_INET6:
{
this->address6.sin6_port = htons(port);
break;
2006-06-22 06:36:28 +00:00
}
default:
{
break;
2006-06-22 06:36:28 +00:00
}
}
}
2005-11-16 16:11:08 +00:00
/**
2005-11-21 11:45:04 +00:00
* Implements host_t.clone.
2005-11-16 16:11:08 +00:00
*/
static private_host_t *clone_(private_host_t *this)
2005-11-16 16:11:08 +00:00
{
private_host_t *new = malloc_thing(private_host_t);
2005-12-02 16:09:04 +00:00
2005-11-16 16:11:08 +00:00
memcpy(new, this, sizeof(private_host_t));
2005-11-28 20:29:47 +00:00
return new;
2005-11-16 16:11:08 +00:00
}
/**
* Impelements host_t.ip_equals
*/
static bool ip_equals(private_host_t *this, private_host_t *other)
{
if (this->address.sa_family != other->address.sa_family)
{
/* 0.0.0.0 and ::0 are equal */
if (is_anyaddr(this) && is_anyaddr(other))
{
return TRUE;
}
return FALSE;
}
switch (this->address.sa_family)
{
case AF_INET:
{
if (memeq(&this->address4.sin_addr, &other->address4.sin_addr,
sizeof(this->address4.sin_addr)))
{
return TRUE;
}
break;
}
case AF_INET6:
{
if (memeq(&this->address6.sin6_addr, &other->address6.sin6_addr,
sizeof(this->address6.sin6_addr)))
{
return TRUE;
}
}
default:
break;
}
return FALSE;
}
2005-11-16 16:11:08 +00:00
2006-06-22 06:36:28 +00:00
/**
* Implements host_t.get_differences
*/
static host_diff_t get_differences(host_t *this, host_t *other)
2006-06-22 06:36:28 +00:00
{
host_diff_t ret = HOST_DIFF_NONE;
2006-06-22 06:36:28 +00:00
if (!this->ip_equals(this, other))
2006-06-22 06:36:28 +00:00
{
ret |= HOST_DIFF_ADDR;
}
if (this->get_port(this) != other->get_port(other))
2006-06-22 06:36:28 +00:00
{
ret |= HOST_DIFF_PORT;
}
2006-06-22 06:36:28 +00:00
return ret;
}
/**
* Impelements host_t.equals
*/
static bool equals(private_host_t *this, private_host_t *other)
{
if (!ip_equals(this, other))
{
return FALSE;
}
switch (this->address.sa_family)
{
case AF_INET:
{
if (this->address4.sin_port == other->address4.sin_port)
{
return TRUE;
}
break;
}
case AF_INET6:
{
if (this->address6.sin6_port == other->address6.sin6_port)
{
return TRUE;
}
break;
}
default:
break;
}
return FALSE;
}
2005-12-02 16:09:04 +00:00
/**
* Implements host_t.destroy
*/
static void destroy(private_host_t *this)
{
free(this);
2005-12-02 16:09:04 +00:00
}
/**
* Creates an empty host_t object
2005-11-16 16:11:08 +00:00
*/
static private_host_t *host_create_empty(void)
2005-11-16 16:11:08 +00:00
{
private_host_t *this = malloc_thing(private_host_t);
2005-11-16 16:11:08 +00:00
this->public.get_sockaddr = (sockaddr_t* (*) (host_t*))get_sockaddr;
this->public.get_sockaddr_len = (socklen_t*(*) (host_t*))get_sockaddr_len;
this->public.clone = (host_t* (*) (host_t*))clone_;
this->public.get_family = (int (*) (host_t*))get_family;
this->public.get_address = (chunk_t (*) (host_t *)) get_address;
2005-11-21 11:45:04 +00:00
this->public.get_port = (u_int16_t (*) (host_t *))get_port;
2006-06-22 06:36:28 +00:00
this->public.set_port = (void (*) (host_t *,u_int16_t))set_port;
this->public.get_differences = get_differences;
this->public.ip_equals = (bool (*) (host_t *,host_t *)) ip_equals;
this->public.equals = (bool (*) (host_t *,host_t *)) equals;
this->public.is_anyaddr = (bool (*) (host_t *)) is_anyaddr;
2005-11-28 20:29:47 +00:00
this->public.destroy = (void (*) (host_t*))destroy;
2005-11-16 16:11:08 +00:00
return this;
}
/*
* Described in header.
*/
host_t *host_create_from_string(char *string, u_int16_t port)
{
private_host_t *this = host_create_empty();
if (strchr(string, '.'))
{
this->address.sa_family = AF_INET;
}
else
{
this->address.sa_family = AF_INET6;
}
switch (this->address.sa_family)
{
case AF_INET:
{
if (inet_pton(AF_INET, string, &this->address4.sin_addr) <=0)
{
break;
}
this->address4.sin_port = htons(port);
this->socklen = sizeof(struct sockaddr_in);
return &this->public;
}
case AF_INET6:
{
if (inet_pton(AF_INET6, string, &this->address6.sin6_addr) <=0)
{
break;
}
this->address6.sin6_port = htons(port);
this->socklen = sizeof(struct sockaddr_in6);
return &this->public;
}
default:
{
break;
}
}
free(this);
return NULL;
2005-11-16 16:11:08 +00:00
}
2005-11-29 15:23:04 +00:00
/*
* Described in header.
*/
host_t *host_create_from_chunk(int family, chunk_t address, u_int16_t port)
{
private_host_t *this = host_create_empty();
2005-11-29 15:23:04 +00:00
this->address.sa_family = family;
switch (family)
2005-11-29 15:23:04 +00:00
{
case AF_INET:
2005-11-29 15:23:04 +00:00
{
if (address.len != 4)
2005-11-29 15:23:04 +00:00
{
break;
2005-11-29 15:23:04 +00:00
}
memcpy(&(this->address4.sin_addr.s_addr), address.ptr,4);
this->address4.sin_port = htons(port);
this->socklen = sizeof(struct sockaddr_in);
return &(this->public);
2005-11-29 15:23:04 +00:00
}
case AF_INET6:
{
if (address.len != 16)
{
break;
}
memcpy(&(this->address6.sin6_addr.s6_addr), address.ptr, 16);
this->address6.sin6_port = htons(port);
this->socklen = sizeof(struct sockaddr_in6);
return &this->public;
}
default:
break;
2005-11-29 15:23:04 +00:00
}
free(this);
2005-11-29 15:23:04 +00:00
return NULL;
}
/*
* Described in header.
*/
host_t *host_create_from_sockaddr(sockaddr_t *sockaddr)
{
private_host_t *this = host_create_empty();
switch (sockaddr->sa_family)
{
case AF_INET:
2006-08-31 06:16:52 +00:00
{
memcpy(&this->address4, sockaddr, sizeof(struct sockaddr_in));
this->socklen = sizeof(struct sockaddr_in);
return &this->public;
2006-08-31 06:16:52 +00:00
}
case AF_INET6:
{
memcpy(&this->address6, sockaddr, sizeof(struct sockaddr_in6));
this->socklen = sizeof(struct sockaddr_in6);
return &this->public;
}
default:
break;
}
free(this);
return NULL;
}
2007-02-28 14:04:36 +00:00
/*
* Described in header.
*/
host_t *host_create_any(int family)
{
private_host_t *this = host_create_empty();
memset(&this->address_max, 0, sizeof(struct sockaddr_storage));
this->address.sa_family = family;
switch (family)
{
case AF_INET:
{
this->socklen = sizeof(struct sockaddr_in);
return &(this->public);
}
case AF_INET6:
{
this->socklen = sizeof(struct sockaddr_in6);
return &this->public;
}
default:
break;
}
return NULL;
}