strongswan/src/libstrongswan/utils/host.c

366 lines
6.9 KiB
C
Raw Normal View History

2005-11-16 16:11:08 +00:00
/**
* @file host.c
*
2005-11-28 20:29:47 +00:00
* @brief Implementation of host_t.
2005-11-16 16:11:08 +00:00
*
*/
/*
* Copyright (C) 2005 Jan Hutter, Martin Willi
* 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.
*/
#include <string.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
/**
2005-11-24 09:17:51 +00:00
* @brief 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;
/**
* Address family to use, such as AF_INET or AF_INET6
*/
int family;
2005-12-02 16:09:04 +00:00
/**
* string representation of host
*/
char *string;
2005-11-16 16:11:08 +00:00
/**
* low-lewel structure, wich stores the address
*/
union {
struct sockaddr address;
struct sockaddr_in address4;
};
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_default_route.
*/
static bool is_default_route (private_host_t *this)
{
switch (this->family)
{
case AF_INET:
{
2005-12-02 16:09:04 +00:00
static u_int8_t default_route[4] = {0x00,0x00,0x00,0x00};
if (memcmp(default_route,&(this->address4.sin_addr.s_addr),4) == 0)
{
2005-12-02 16:09:04 +00:00
return TRUE;
}
2005-12-02 16:09:04 +00:00
return FALSE;
}
default:
{
/* empty chunk is returned */
return FALSE;
}
}
}
2005-11-16 16:11:08 +00:00
/**
2005-11-21 11:45:04 +00:00
* implements host_t.get_address
*/
static char *get_address(private_host_t *this)
{
switch (this->family)
{
case AF_INET:
{
2005-12-02 16:09:04 +00:00
char *string;
/* we need to clone it, since inet_ntoa overwrites
* internal buffer on subsequent calls
*/
free(this->string);
string = inet_ntoa(this->address4.sin_addr);
this->string = malloc(strlen(string)+1);
2005-12-02 16:09:04 +00:00
strcpy(this->string, string);
return this->string;
2005-11-21 11:45:04 +00:00
}
default:
{
return "(family not supported)";
}
}
}
2005-11-29 15:23:04 +00:00
/**
* Implementation of host_t.get_address_as_chunk.
*/
static chunk_t get_address_as_chunk(private_host_t *this)
{
chunk_t address = CHUNK_INITIALIZER;
switch (this->family)
{
case AF_INET:
{
/* allocate 4 bytes for IPV4 address*/
address.ptr = malloc(4);
2005-11-29 15:23:04 +00:00
address.len = 4;
memcpy(address.ptr,&(this->address4.sin_addr.s_addr),4);
2005-11-29 15:23:04 +00:00
}
default:
{
/* empty chunk is returned */
return address;
}
}
}
static xfrm_address_t get_xfrm_addr(private_host_t *this)
{
switch (this->family)
{
case AF_INET:
{
return (xfrm_address_t)(this->address4.sin_addr.s_addr);
}
default:
{
/* todo */
return (xfrm_address_t)(this->address4.sin_addr.s_addr);
}
}
}
static int get_family(private_host_t *this)
{
return this->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->family)
{
case AF_INET:
{
return ntohs(this->address4.sin_port);
2005-11-21 11:45:04 +00:00
}
default:
{
return 0;
}
}
}
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
*/
2005-11-28 20:29:47 +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-12-02 16:09:04 +00:00
if (this->string)
{
new->string = malloc(strlen(this->string)+1);
2005-12-02 16:09:04 +00:00
strcpy(new->string, this->string);
}
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)
{
switch (this->family)
{
/* IPv4 */
case AF_INET:
{
if ((this->address4.sin_family == other->address4.sin_family) &&
(this->address4.sin_addr.s_addr == other->address4.sin_addr.s_addr))
{
return TRUE;
}
}
}
return FALSE;
}
2005-11-16 16:11:08 +00:00
/**
* Impelements host_t.equals
*/
static bool equals(private_host_t *this, private_host_t *other)
{
switch (this->family)
{
/* IPv4 */
case AF_INET:
{
if ((this->address4.sin_family == other->address4.sin_family) &&
(this->address4.sin_addr.s_addr == other->address4.sin_addr.s_addr) &&
(this->address4.sin_port == other->address4.sin_port))
{
return TRUE;
}
}
}
return FALSE;
}
2005-12-02 16:09:04 +00:00
/**
* Implements host_t.destroy
*/
static void destroy(private_host_t *this)
{
free(this->string);
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;
2005-11-28 20:29:47 +00:00
this->public.clone = (host_t* (*) (host_t*))clone;
this->public.get_family = (int (*) (host_t*))get_family;
this->public.get_xfrm_addr = (xfrm_address_t (*) (host_t *))get_xfrm_addr;
2005-11-21 11:45:04 +00:00
this->public.get_address = (char* (*) (host_t *))get_address;
2005-11-29 15:23:04 +00:00
this->public.get_address_as_chunk = (chunk_t (*) (host_t *)) get_address_as_chunk;
2005-11-21 11:45:04 +00:00
this->public.get_port = (u_int16_t (*) (host_t *))get_port;
this->public.ip_equals = (bool (*) (host_t *,host_t *)) ip_equals;
this->public.equals = (bool (*) (host_t *,host_t *)) equals;
this->public.is_default_route = (bool (*) (host_t *)) is_default_route;
2005-11-28 20:29:47 +00:00
this->public.destroy = (void (*) (host_t*))destroy;
2005-11-16 16:11:08 +00:00
2005-12-02 16:09:04 +00:00
this->string = NULL;
return this;
}
/*
* Described in header.
*/
host_t *host_create(int family, char *address, u_int16_t port)
{
private_host_t *this = host_create_empty();
2005-11-16 16:11:08 +00:00
this->family = family;
switch (family)
{
/* IPv4 */
case AF_INET:
{
this->address4.sin_family = AF_INET;
this->address4.sin_addr.s_addr = inet_addr(address);
this->address4.sin_port = htons(port);
2005-11-16 16:11:08 +00:00
this->socklen = sizeof(struct sockaddr_in);
return &(this->public);
}
default:
{
free(this);
return NULL;
2005-11-16 16:11:08 +00:00
}
}
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->family = family;
switch (family)
2005-11-29 15:23:04 +00:00
{
/* IPv4 */
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
}
this->address4.sin_family = AF_INET;
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
}
}
free(this);
2005-11-29 15:23:04 +00:00
return NULL;
}
/*
* Described in header.
*/
host_t *host_create_from_sockaddr(sockaddr_t *sockaddr)
{
chunk_t address;
switch (sockaddr->sa_family)
{
/* IPv4 */
case AF_INET:
{
struct sockaddr_in *sin = (struct sockaddr_in *)sockaddr;
address.ptr = (void*)&(sin->sin_addr.s_addr);
address.len = 4;
return host_create_from_chunk(AF_INET, address, ntohs(sin->sin_port));
}
default:
return NULL;
}
}