|
|
|
@ -62,13 +62,18 @@ mapping_t extended_sequence_numbers_m[] = {
|
|
|
|
|
{MAPPING_END, NULL}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct protocol_proposal_t protocol_proposal_t;
|
|
|
|
|
typedef struct private_proposal_t private_proposal_t;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* substructure which holds all data algos for a specific protocol
|
|
|
|
|
* Private data of an proposal_t object
|
|
|
|
|
*/
|
|
|
|
|
struct protocol_proposal_t {
|
|
|
|
|
struct private_proposal_t {
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Public part
|
|
|
|
|
*/
|
|
|
|
|
proposal_t public;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* protocol (ESP or AH)
|
|
|
|
|
*/
|
|
|
|
@ -96,85 +101,15 @@ struct protocol_proposal_t {
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* priority ordered list of extended sequence number flags
|
|
|
|
|
*/
|
|
|
|
|
*/
|
|
|
|
|
linked_list_t *esns;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* senders SPI
|
|
|
|
|
*/
|
|
|
|
|
chunk_t spi;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct private_proposal_t private_proposal_t;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Private data of an proposal_t object
|
|
|
|
|
*/
|
|
|
|
|
struct private_proposal_t {
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Public part
|
|
|
|
|
*/
|
|
|
|
|
proposal_t public;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* number of this proposal, as used in the payload
|
|
|
|
|
*/
|
|
|
|
|
u_int8_t number;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* list of protocol_proposal_t's
|
|
|
|
|
*/
|
|
|
|
|
linked_list_t *protocol_proposals;
|
|
|
|
|
u_int64_t spi;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Look up a protocol_proposal, or create one if necessary...
|
|
|
|
|
*/
|
|
|
|
|
static protocol_proposal_t *get_protocol_proposal(private_proposal_t *this, protocol_id_t proto, bool create)
|
|
|
|
|
{
|
|
|
|
|
protocol_proposal_t *proto_proposal = NULL, *current_proto_proposal;;
|
|
|
|
|
iterator_t *iterator;
|
|
|
|
|
|
|
|
|
|
/* find our protocol in the proposals */
|
|
|
|
|
iterator = this->protocol_proposals->create_iterator(this->protocol_proposals, TRUE);
|
|
|
|
|
while (iterator->has_next(iterator))
|
|
|
|
|
{
|
|
|
|
|
iterator->current(iterator, (void**)¤t_proto_proposal);
|
|
|
|
|
if (current_proto_proposal->protocol == proto)
|
|
|
|
|
{
|
|
|
|
|
proto_proposal = current_proto_proposal;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
iterator->destroy(iterator);
|
|
|
|
|
|
|
|
|
|
if (!proto_proposal && create)
|
|
|
|
|
{
|
|
|
|
|
/* nope, create a new one */
|
|
|
|
|
proto_proposal = malloc_thing(protocol_proposal_t);
|
|
|
|
|
proto_proposal->protocol = proto;
|
|
|
|
|
proto_proposal->encryption_algos = linked_list_create();
|
|
|
|
|
proto_proposal->integrity_algos = linked_list_create();
|
|
|
|
|
proto_proposal->prf_algos = linked_list_create();
|
|
|
|
|
proto_proposal->dh_groups = linked_list_create();
|
|
|
|
|
proto_proposal->esns = linked_list_create();
|
|
|
|
|
if (proto == PROTO_IKE)
|
|
|
|
|
{
|
|
|
|
|
proto_proposal->spi.len = 8;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
proto_proposal->spi.len = 4;
|
|
|
|
|
}
|
|
|
|
|
proto_proposal->spi.ptr = malloc(proto_proposal->spi.len);
|
|
|
|
|
/* add to the list */
|
|
|
|
|
this->protocol_proposals->insert_last(this->protocol_proposals, (void*)proto_proposal);
|
|
|
|
|
}
|
|
|
|
|
return proto_proposal;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Add algorithm/keysize to a algorithm list
|
|
|
|
|
*/
|
|
|
|
@ -190,26 +125,24 @@ static void add_algo(linked_list_t *list, u_int8_t algo, size_t key_size)
|
|
|
|
|
/**
|
|
|
|
|
* Implements proposal_t.add_algorithm
|
|
|
|
|
*/
|
|
|
|
|
static void add_algorithm(private_proposal_t *this, protocol_id_t proto, transform_type_t type, u_int16_t algo, size_t key_size)
|
|
|
|
|
static void add_algorithm(private_proposal_t *this, transform_type_t type, u_int16_t algo, size_t key_size)
|
|
|
|
|
{
|
|
|
|
|
protocol_proposal_t *proto_proposal = get_protocol_proposal(this, proto, TRUE);
|
|
|
|
|
|
|
|
|
|
switch (type)
|
|
|
|
|
{
|
|
|
|
|
case ENCRYPTION_ALGORITHM:
|
|
|
|
|
add_algo(proto_proposal->encryption_algos, algo, key_size);
|
|
|
|
|
add_algo(this->encryption_algos, algo, key_size);
|
|
|
|
|
break;
|
|
|
|
|
case INTEGRITY_ALGORITHM:
|
|
|
|
|
add_algo(proto_proposal->integrity_algos, algo, key_size);
|
|
|
|
|
add_algo(this->integrity_algos, algo, key_size);
|
|
|
|
|
break;
|
|
|
|
|
case PSEUDO_RANDOM_FUNCTION:
|
|
|
|
|
add_algo(proto_proposal->prf_algos, algo, key_size);
|
|
|
|
|
add_algo(this->prf_algos, algo, key_size);
|
|
|
|
|
break;
|
|
|
|
|
case DIFFIE_HELLMAN_GROUP:
|
|
|
|
|
add_algo(proto_proposal->dh_groups, algo, 0);
|
|
|
|
|
add_algo(this->dh_groups, algo, 0);
|
|
|
|
|
break;
|
|
|
|
|
case EXTENDED_SEQUENCE_NUMBERS:
|
|
|
|
|
add_algo(proto_proposal->esns, algo, 0);
|
|
|
|
|
add_algo(this->esns, algo, 0);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
@ -219,31 +152,25 @@ static void add_algorithm(private_proposal_t *this, protocol_id_t proto, transfo
|
|
|
|
|
/**
|
|
|
|
|
* Implements proposal_t.get_algorithm.
|
|
|
|
|
*/
|
|
|
|
|
static bool get_algorithm(private_proposal_t *this, protocol_id_t proto, transform_type_t type, algorithm_t** algo)
|
|
|
|
|
static bool get_algorithm(private_proposal_t *this, transform_type_t type, algorithm_t** algo)
|
|
|
|
|
{
|
|
|
|
|
linked_list_t * list;
|
|
|
|
|
protocol_proposal_t *proto_proposal = get_protocol_proposal(this, proto, FALSE);
|
|
|
|
|
|
|
|
|
|
if (proto_proposal == NULL)
|
|
|
|
|
{
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
linked_list_t *list;
|
|
|
|
|
switch (type)
|
|
|
|
|
{
|
|
|
|
|
case ENCRYPTION_ALGORITHM:
|
|
|
|
|
list = proto_proposal->encryption_algos;
|
|
|
|
|
list = this->encryption_algos;
|
|
|
|
|
break;
|
|
|
|
|
case INTEGRITY_ALGORITHM:
|
|
|
|
|
list = proto_proposal->integrity_algos;
|
|
|
|
|
list = this->integrity_algos;
|
|
|
|
|
break;
|
|
|
|
|
case PSEUDO_RANDOM_FUNCTION:
|
|
|
|
|
list = proto_proposal->prf_algos;
|
|
|
|
|
list = this->prf_algos;
|
|
|
|
|
break;
|
|
|
|
|
case DIFFIE_HELLMAN_GROUP:
|
|
|
|
|
list = proto_proposal->dh_groups;
|
|
|
|
|
list = this->dh_groups;
|
|
|
|
|
break;
|
|
|
|
|
case EXTENDED_SEQUENCE_NUMBERS:
|
|
|
|
|
list = proto_proposal->esns;
|
|
|
|
|
list = this->esns;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
return FALSE;
|
|
|
|
@ -258,26 +185,20 @@ static bool get_algorithm(private_proposal_t *this, protocol_id_t proto, transfo
|
|
|
|
|
/**
|
|
|
|
|
* Implements proposal_t.create_algorithm_iterator.
|
|
|
|
|
*/
|
|
|
|
|
static iterator_t *create_algorithm_iterator(private_proposal_t *this, protocol_id_t proto, transform_type_t type)
|
|
|
|
|
static iterator_t *create_algorithm_iterator(private_proposal_t *this, transform_type_t type)
|
|
|
|
|
{
|
|
|
|
|
protocol_proposal_t *proto_proposal = get_protocol_proposal(this, proto, FALSE);
|
|
|
|
|
if (proto_proposal == NULL)
|
|
|
|
|
{
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (type)
|
|
|
|
|
{
|
|
|
|
|
case ENCRYPTION_ALGORITHM:
|
|
|
|
|
return proto_proposal->encryption_algos->create_iterator(proto_proposal->encryption_algos, TRUE);
|
|
|
|
|
return this->encryption_algos->create_iterator(this->encryption_algos, TRUE);
|
|
|
|
|
case INTEGRITY_ALGORITHM:
|
|
|
|
|
return proto_proposal->integrity_algos->create_iterator(proto_proposal->integrity_algos, TRUE);
|
|
|
|
|
return this->integrity_algos->create_iterator(this->integrity_algos, TRUE);
|
|
|
|
|
case PSEUDO_RANDOM_FUNCTION:
|
|
|
|
|
return proto_proposal->prf_algos->create_iterator(proto_proposal->prf_algos, TRUE);
|
|
|
|
|
return this->prf_algos->create_iterator(this->prf_algos, TRUE);
|
|
|
|
|
case DIFFIE_HELLMAN_GROUP:
|
|
|
|
|
return proto_proposal->dh_groups->create_iterator(proto_proposal->dh_groups, TRUE);
|
|
|
|
|
return this->dh_groups->create_iterator(this->dh_groups, TRUE);
|
|
|
|
|
case EXTENDED_SEQUENCE_NUMBERS:
|
|
|
|
|
return proto_proposal->esns->create_iterator(proto_proposal->esns, TRUE);
|
|
|
|
|
return this->esns->create_iterator(this->esns, TRUE);
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
@ -336,201 +257,112 @@ static proposal_t *select_proposal(private_proposal_t *this, private_proposal_t
|
|
|
|
|
proposal_t *selected;
|
|
|
|
|
u_int16_t algo;
|
|
|
|
|
size_t key_size;
|
|
|
|
|
iterator_t *iterator;
|
|
|
|
|
protocol_proposal_t *this_prop, *other_prop;
|
|
|
|
|
protocol_id_t proto;
|
|
|
|
|
bool add;
|
|
|
|
|
u_int64_t spi;
|
|
|
|
|
|
|
|
|
|
/* empty proposal? no match */
|
|
|
|
|
if (this->protocol_proposals->get_count(this->protocol_proposals) == 0 ||
|
|
|
|
|
other->protocol_proposals->get_count(other->protocol_proposals) == 0)
|
|
|
|
|
{
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
/* they MUST have the same amount of protocols */
|
|
|
|
|
if (this->protocol_proposals->get_count(this->protocol_proposals) !=
|
|
|
|
|
other->protocol_proposals->get_count(other->protocol_proposals))
|
|
|
|
|
/* check protocol */
|
|
|
|
|
if (this->protocol != other->protocol)
|
|
|
|
|
{
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
selected = proposal_create(this->number);
|
|
|
|
|
selected = proposal_create(this->protocol);
|
|
|
|
|
|
|
|
|
|
/* iterate over supplied proposals */
|
|
|
|
|
iterator = other->protocol_proposals->create_iterator(other->protocol_proposals, TRUE);
|
|
|
|
|
while (iterator->has_next(iterator))
|
|
|
|
|
/* select encryption algorithm */
|
|
|
|
|
if (select_algo(this->encryption_algos, other->encryption_algos, &add, &algo, &key_size))
|
|
|
|
|
{
|
|
|
|
|
iterator->current(iterator, (void**)&other_prop);
|
|
|
|
|
/* get the proposal with the same protocol */
|
|
|
|
|
proto = other_prop->protocol;
|
|
|
|
|
this_prop = get_protocol_proposal(this, proto, FALSE);
|
|
|
|
|
|
|
|
|
|
if (this_prop == NULL)
|
|
|
|
|
{
|
|
|
|
|
iterator->destroy(iterator);
|
|
|
|
|
selected->destroy(selected);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* select encryption algorithm */
|
|
|
|
|
if (select_algo(this_prop->encryption_algos, other_prop->encryption_algos, &add, &algo, &key_size))
|
|
|
|
|
{
|
|
|
|
|
if (add)
|
|
|
|
|
{
|
|
|
|
|
selected->add_algorithm(selected, proto, ENCRYPTION_ALGORITHM, algo, key_size);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
iterator->destroy(iterator);
|
|
|
|
|
selected->destroy(selected);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
/* select integrity algorithm */
|
|
|
|
|
if (select_algo(this_prop->integrity_algos, other_prop->integrity_algos, &add, &algo, &key_size))
|
|
|
|
|
{
|
|
|
|
|
if (add)
|
|
|
|
|
{
|
|
|
|
|
selected->add_algorithm(selected, proto, INTEGRITY_ALGORITHM, algo, key_size);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
iterator->destroy(iterator);
|
|
|
|
|
selected->destroy(selected);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
/* select prf algorithm */
|
|
|
|
|
if (select_algo(this_prop->prf_algos, other_prop->prf_algos, &add, &algo, &key_size))
|
|
|
|
|
if (add)
|
|
|
|
|
{
|
|
|
|
|
if (add)
|
|
|
|
|
{
|
|
|
|
|
selected->add_algorithm(selected, proto, PSEUDO_RANDOM_FUNCTION, algo, key_size);
|
|
|
|
|
}
|
|
|
|
|
selected->add_algorithm(selected, ENCRYPTION_ALGORITHM, algo, key_size);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
iterator->destroy(iterator);
|
|
|
|
|
selected->destroy(selected);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
/* select a DH-group */
|
|
|
|
|
if (select_algo(this_prop->dh_groups, other_prop->dh_groups, &add, &algo, &key_size))
|
|
|
|
|
{
|
|
|
|
|
if (add)
|
|
|
|
|
{
|
|
|
|
|
selected->add_algorithm(selected, proto, DIFFIE_HELLMAN_GROUP, algo, 0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
selected->destroy(selected);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
/* select integrity algorithm */
|
|
|
|
|
if (select_algo(this->integrity_algos, other->integrity_algos, &add, &algo, &key_size))
|
|
|
|
|
{
|
|
|
|
|
if (add)
|
|
|
|
|
{
|
|
|
|
|
iterator->destroy(iterator);
|
|
|
|
|
selected->destroy(selected);
|
|
|
|
|
return NULL;
|
|
|
|
|
selected->add_algorithm(selected, INTEGRITY_ALGORITHM, algo, key_size);
|
|
|
|
|
}
|
|
|
|
|
/* select if we use ESNs */
|
|
|
|
|
if (select_algo(this_prop->esns, other_prop->esns, &add, &algo, &key_size))
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
selected->destroy(selected);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
/* select prf algorithm */
|
|
|
|
|
if (select_algo(this->prf_algos, other->prf_algos, &add, &algo, &key_size))
|
|
|
|
|
{
|
|
|
|
|
if (add)
|
|
|
|
|
{
|
|
|
|
|
if (add)
|
|
|
|
|
{
|
|
|
|
|
selected->add_algorithm(selected, proto, EXTENDED_SEQUENCE_NUMBERS, algo, 0);
|
|
|
|
|
}
|
|
|
|
|
selected->add_algorithm(selected, PSEUDO_RANDOM_FUNCTION, algo, key_size);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
selected->destroy(selected);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
/* select a DH-group */
|
|
|
|
|
if (select_algo(this->dh_groups, other->dh_groups, &add, &algo, &key_size))
|
|
|
|
|
{
|
|
|
|
|
if (add)
|
|
|
|
|
{
|
|
|
|
|
iterator->destroy(iterator);
|
|
|
|
|
selected->destroy(selected);
|
|
|
|
|
return NULL;
|
|
|
|
|
selected->add_algorithm(selected, DIFFIE_HELLMAN_GROUP, algo, 0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
iterator->destroy(iterator);
|
|
|
|
|
|
|
|
|
|
/* apply spis from "other" */
|
|
|
|
|
spi = other->public.get_spi(&(other->public), PROTO_AH);
|
|
|
|
|
if (spi)
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
selected->set_spi(selected, PROTO_AH, spi);
|
|
|
|
|
selected->destroy(selected);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
spi = other->public.get_spi(&(other->public), PROTO_ESP);
|
|
|
|
|
if (spi)
|
|
|
|
|
/* select if we use ESNs */
|
|
|
|
|
if (select_algo(this->esns, other->esns, &add, &algo, &key_size))
|
|
|
|
|
{
|
|
|
|
|
selected->set_spi(selected, PROTO_ESP, spi);
|
|
|
|
|
if (add)
|
|
|
|
|
{
|
|
|
|
|
selected->add_algorithm(selected, EXTENDED_SEQUENCE_NUMBERS, algo, 0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
selected->destroy(selected);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* apply SPI from "other" */
|
|
|
|
|
selected->set_spi(selected, other->spi);
|
|
|
|
|
|
|
|
|
|
/* everything matched, return new proposal */
|
|
|
|
|
return selected;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Implements proposal_t.get_number.
|
|
|
|
|
*/
|
|
|
|
|
static u_int8_t get_number(private_proposal_t *this)
|
|
|
|
|
{
|
|
|
|
|
return this->number;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Implements proposal_t.get_protocols.
|
|
|
|
|
*/
|
|
|
|
|
static void get_protocols(private_proposal_t *this, protocol_id_t ids[2])
|
|
|
|
|
static protocol_id_t get_protocol(private_proposal_t *this)
|
|
|
|
|
{
|
|
|
|
|
iterator_t *iterator = this->protocol_proposals->create_iterator(this->protocol_proposals, TRUE);
|
|
|
|
|
u_int i = 0;
|
|
|
|
|
|
|
|
|
|
ids[0] = PROTO_NONE;
|
|
|
|
|
ids[1] = PROTO_NONE;
|
|
|
|
|
while (iterator->has_next(iterator))
|
|
|
|
|
{
|
|
|
|
|
protocol_proposal_t *proto_prop;
|
|
|
|
|
iterator->current(iterator, (void**)&proto_prop);
|
|
|
|
|
ids[i++] = proto_prop->protocol;
|
|
|
|
|
if (i>1)
|
|
|
|
|
{
|
|
|
|
|
/* should not happen, but who knows */
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
iterator->destroy(iterator);
|
|
|
|
|
return this->protocol;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Implements proposal_t.set_spi.
|
|
|
|
|
*/
|
|
|
|
|
static void set_spi(private_proposal_t *this, protocol_id_t proto, u_int64_t spi)
|
|
|
|
|
static void set_spi(private_proposal_t *this, u_int64_t spi)
|
|
|
|
|
{
|
|
|
|
|
protocol_proposal_t *proto_proposal = get_protocol_proposal(this, proto, FALSE);
|
|
|
|
|
if (proto_proposal)
|
|
|
|
|
{
|
|
|
|
|
if (proto == PROTO_AH || proto == PROTO_ESP)
|
|
|
|
|
{
|
|
|
|
|
*((u_int32_t*)proto_proposal->spi.ptr) = (u_int32_t)spi;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
*((u_int64_t*)proto_proposal->spi.ptr) = spi;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
this->spi = spi;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Implements proposal_t.get_spi.
|
|
|
|
|
*/
|
|
|
|
|
static u_int64_t get_spi(private_proposal_t *this, protocol_id_t proto)
|
|
|
|
|
static u_int64_t get_spi(private_proposal_t *this)
|
|
|
|
|
{
|
|
|
|
|
protocol_proposal_t *proto_proposal = get_protocol_proposal(this, proto, FALSE);
|
|
|
|
|
if (proto_proposal)
|
|
|
|
|
{
|
|
|
|
|
if (proto == PROTO_AH || proto == PROTO_ESP)
|
|
|
|
|
{
|
|
|
|
|
return (u_int64_t)*((u_int32_t*)proto_proposal->spi.ptr);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
return *((u_int64_t*)proto_proposal->spi.ptr);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
return this->spi;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
@ -555,24 +387,15 @@ static void clone_algo_list(linked_list_t *list, linked_list_t *clone_list)
|
|
|
|
|
*/
|
|
|
|
|
static proposal_t *clone(private_proposal_t *this)
|
|
|
|
|
{
|
|
|
|
|
private_proposal_t *clone = (private_proposal_t*)proposal_create(this->number);
|
|
|
|
|
private_proposal_t *clone = (private_proposal_t*)proposal_create(this->protocol);
|
|
|
|
|
|
|
|
|
|
iterator_t *iterator = this->protocol_proposals->create_iterator(this->protocol_proposals, TRUE);
|
|
|
|
|
while (iterator->has_next(iterator))
|
|
|
|
|
{
|
|
|
|
|
protocol_proposal_t *proto_prop, *clone_proto_prop;
|
|
|
|
|
iterator->current(iterator, (void**)&proto_prop);
|
|
|
|
|
|
|
|
|
|
clone_proto_prop = get_protocol_proposal(clone, proto_prop->protocol, TRUE);
|
|
|
|
|
memcpy(clone_proto_prop->spi.ptr, proto_prop->spi.ptr, clone_proto_prop->spi.len);
|
|
|
|
|
|
|
|
|
|
clone_algo_list(proto_prop->encryption_algos, clone_proto_prop->encryption_algos);
|
|
|
|
|
clone_algo_list(proto_prop->integrity_algos, clone_proto_prop->integrity_algos);
|
|
|
|
|
clone_algo_list(proto_prop->prf_algos, clone_proto_prop->prf_algos);
|
|
|
|
|
clone_algo_list(proto_prop->dh_groups, clone_proto_prop->dh_groups);
|
|
|
|
|
clone_algo_list(proto_prop->esns, clone_proto_prop->esns);
|
|
|
|
|
}
|
|
|
|
|
iterator->destroy(iterator);
|
|
|
|
|
clone_algo_list(this->encryption_algos, clone->encryption_algos);
|
|
|
|
|
clone_algo_list(this->integrity_algos, clone->integrity_algos);
|
|
|
|
|
clone_algo_list(this->prf_algos, clone->prf_algos);
|
|
|
|
|
clone_algo_list(this->dh_groups, clone->dh_groups);
|
|
|
|
|
clone_algo_list(this->esns, clone->esns);
|
|
|
|
|
|
|
|
|
|
clone->spi = this->spi;
|
|
|
|
|
|
|
|
|
|
return &clone->public;
|
|
|
|
|
}
|
|
|
|
@ -597,46 +420,39 @@ static void free_algo_list(linked_list_t *list)
|
|
|
|
|
*/
|
|
|
|
|
static void destroy(private_proposal_t *this)
|
|
|
|
|
{
|
|
|
|
|
while(this->protocol_proposals->get_count(this->protocol_proposals) > 0)
|
|
|
|
|
{
|
|
|
|
|
protocol_proposal_t *proto_prop;
|
|
|
|
|
this->protocol_proposals->remove_last(this->protocol_proposals, (void**)&proto_prop);
|
|
|
|
|
|
|
|
|
|
free_algo_list(proto_prop->encryption_algos);
|
|
|
|
|
free_algo_list(proto_prop->integrity_algos);
|
|
|
|
|
free_algo_list(proto_prop->prf_algos);
|
|
|
|
|
free_algo_list(proto_prop->dh_groups);
|
|
|
|
|
free_algo_list(proto_prop->esns);
|
|
|
|
|
|
|
|
|
|
free(proto_prop->spi.ptr);
|
|
|
|
|
free(proto_prop);
|
|
|
|
|
}
|
|
|
|
|
this->protocol_proposals->destroy(this->protocol_proposals);
|
|
|
|
|
|
|
|
|
|
free_algo_list(this->encryption_algos);
|
|
|
|
|
free_algo_list(this->integrity_algos);
|
|
|
|
|
free_algo_list(this->prf_algos);
|
|
|
|
|
free_algo_list(this->dh_groups);
|
|
|
|
|
free_algo_list(this->esns);
|
|
|
|
|
free(this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Describtion in header-file
|
|
|
|
|
*/
|
|
|
|
|
proposal_t *proposal_create(u_int8_t number)
|
|
|
|
|
proposal_t *proposal_create(protocol_id_t protocol)
|
|
|
|
|
{
|
|
|
|
|
private_proposal_t *this = malloc_thing(private_proposal_t);
|
|
|
|
|
|
|
|
|
|
this->public.add_algorithm = (void (*)(proposal_t*,protocol_id_t,transform_type_t,u_int16_t,size_t))add_algorithm;
|
|
|
|
|
this->public.create_algorithm_iterator = (iterator_t* (*)(proposal_t*,protocol_id_t,transform_type_t))create_algorithm_iterator;
|
|
|
|
|
this->public.get_algorithm = (bool (*)(proposal_t*,protocol_id_t,transform_type_t,algorithm_t**))get_algorithm;
|
|
|
|
|
this->public.add_algorithm = (void (*)(proposal_t*,transform_type_t,u_int16_t,size_t))add_algorithm;
|
|
|
|
|
this->public.create_algorithm_iterator = (iterator_t* (*)(proposal_t*,transform_type_t))create_algorithm_iterator;
|
|
|
|
|
this->public.get_algorithm = (bool (*)(proposal_t*,transform_type_t,algorithm_t**))get_algorithm;
|
|
|
|
|
this->public.select = (proposal_t* (*)(proposal_t*,proposal_t*))select_proposal;
|
|
|
|
|
this->public.get_number = (u_int8_t (*)(proposal_t*))get_number;
|
|
|
|
|
this->public.get_protocols = (void(*)(proposal_t *this, protocol_id_t ids[2]))get_protocols;
|
|
|
|
|
this->public.set_spi = (void(*)(proposal_t*,protocol_id_t,u_int64_t spi))set_spi;
|
|
|
|
|
this->public.get_spi = (u_int64_t(*)(proposal_t*,protocol_id_t))get_spi;
|
|
|
|
|
this->public.get_protocol = (protocol_id_t(*)(proposal_t*))get_protocol;
|
|
|
|
|
this->public.set_spi = (void(*)(proposal_t*,u_int64_t))set_spi;
|
|
|
|
|
this->public.get_spi = (u_int64_t(*)(proposal_t*))get_spi;
|
|
|
|
|
this->public.clone = (proposal_t*(*)(proposal_t*))clone;
|
|
|
|
|
this->public.destroy = (void(*)(proposal_t*))destroy;
|
|
|
|
|
|
|
|
|
|
/* init private members*/
|
|
|
|
|
this->number = number;
|
|
|
|
|
this->protocol_proposals = linked_list_create();
|
|
|
|
|
this->spi = 0;
|
|
|
|
|
this->protocol = protocol;
|
|
|
|
|
|
|
|
|
|
this->encryption_algos = linked_list_create();
|
|
|
|
|
this->integrity_algos = linked_list_create();
|
|
|
|
|
this->prf_algos = linked_list_create();
|
|
|
|
|
this->dh_groups = linked_list_create();
|
|
|
|
|
this->esns = linked_list_create();
|
|
|
|
|
|
|
|
|
|
return (&this->public);
|
|
|
|
|
return &this->public;
|
|
|
|
|
}
|
|
|
|
|