aboutsummaryrefslogtreecommitdiffstats
path: root/Source/charon
diff options
context:
space:
mode:
Diffstat (limited to 'Source/charon')
-rw-r--r--Source/charon/encoding/message.c9
-rw-r--r--Source/charon/encoding/message.h8
-rw-r--r--Source/charon/queues/jobs/Makefile.jobs4
-rw-r--r--Source/charon/queues/jobs/job.h3
-rw-r--r--Source/charon/queues/jobs/retransmit_request_job.c109
-rw-r--r--Source/charon/queues/jobs/retransmit_request_job.h83
-rw-r--r--Source/charon/sa/ike_sa.c100
-rw-r--r--Source/charon/sa/ike_sa.h29
-rw-r--r--Source/charon/sa/states/ike_sa_init_requested.c29
-rw-r--r--Source/charon/sa/states/ike_sa_init_responded.c24
-rw-r--r--Source/charon/sa/states/initiator_init.c29
-rw-r--r--Source/charon/sa/states/responder_init.c39
-rw-r--r--Source/charon/threads/thread_pool.c103
13 files changed, 435 insertions, 134 deletions
diff --git a/Source/charon/encoding/message.c b/Source/charon/encoding/message.c
index bcc34ca96..84f174365 100644
--- a/Source/charon/encoding/message.c
+++ b/Source/charon/encoding/message.c
@@ -631,6 +631,14 @@ static status_t generate(private_message_t *this, crypter_t *crypter, signer_t*
}
/**
+ * Implementation of message_t.get_packet.
+ */
+static packet_t *get_packet (private_message_t *this)
+{
+ return this->packet->clone(this->packet);
+}
+
+/**
* Implementation of message_t.parse_header.
*/
static status_t parse_header(private_message_t *this)
@@ -1154,6 +1162,7 @@ message_t *message_create_from_packet(packet_t *packet)
this->public.parse_header = (status_t (*) (message_t *)) parse_header;
this->public.parse_body = (status_t (*) (message_t *,crypter_t*,signer_t*)) parse_body;
this->public.verify = (status_t (*) (message_t*)) verify;
+ this->public.get_packet = (packet_t * (*) (message_t*)) get_packet;
this->public.destroy = (void(*)(message_t*))destroy;
/* private values */
diff --git a/Source/charon/encoding/message.h b/Source/charon/encoding/message.h
index 8d420fbe2..dfbb10da9 100644
--- a/Source/charon/encoding/message.h
+++ b/Source/charon/encoding/message.h
@@ -282,6 +282,14 @@ struct message_t {
iterator_t * (*get_payload_iterator) (message_t *this);
/**
+ * Returns a clone of the internal stored packet_t object.
+ *
+ * @param this message_t object
+ * @return packet_t object as clone of internal one
+ */
+ packet_t * (*get_packet) (message_t *this);
+
+ /**
* @brief Destroys a message and all including objects.
*
* @param this message_t object
diff --git a/Source/charon/queues/jobs/Makefile.jobs b/Source/charon/queues/jobs/Makefile.jobs
index b0482f6e3..d4cbf75cc 100644
--- a/Source/charon/queues/jobs/Makefile.jobs
+++ b/Source/charon/queues/jobs/Makefile.jobs
@@ -26,6 +26,10 @@ OBJS+= $(BUILD_DIR)initiate_ike_sa_job.o
$(BUILD_DIR)initiate_ike_sa_job.o : $(JOBS_DIR)initiate_ike_sa_job.c $(JOBS_DIR)initiate_ike_sa_job.h
$(CC) $(CFLAGS) -c -o $@ $<
+OBJS+= $(BUILD_DIR)retransmit_request_job.o
+$(BUILD_DIR)retransmit_request_job.o : $(JOBS_DIR)retransmit_request_job.c $(JOBS_DIR)retransmit_request_job.h
+ $(CC) $(CFLAGS) -c -o $@ $<
+
OBJS+= $(BUILD_DIR)job.o
$(BUILD_DIR)job.o : $(JOBS_DIR)job.c $(JOBS_DIR)job.h
$(CC) $(CFLAGS) -c -o $@ $<
diff --git a/Source/charon/queues/jobs/job.h b/Source/charon/queues/jobs/job.h
index ee1675fcd..753bea358 100644
--- a/Source/charon/queues/jobs/job.h
+++ b/Source/charon/queues/jobs/job.h
@@ -41,15 +41,18 @@ enum job_type_t {
* Job is implemented in class type incoming_packet_job_t
*/
INCOMING_PACKET,
+
/**
* Retransmit an IKEv2-Message
*/
RETRANSMIT_REQUEST,
+
/**
* Establish an ike sa as initiator
*
* Job is implemented in class type initiate_ike_sa_job_t
*/
+
INITIATE_IKE_SA,
/**
* Delete an ike sa
diff --git a/Source/charon/queues/jobs/retransmit_request_job.c b/Source/charon/queues/jobs/retransmit_request_job.c
new file mode 100644
index 000000000..f54de361d
--- /dev/null
+++ b/Source/charon/queues/jobs/retransmit_request_job.c
@@ -0,0 +1,109 @@
+/**
+ * @file retransmit_request_job.c
+ *
+ * @brief Implementation of retransmit_request_job_t.
+ *
+ */
+
+/*
+ * 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 "retransmit_request_job.h"
+
+
+#include <utils/allocator.h>
+
+
+typedef struct private_retransmit_request_job_t private_retransmit_request_job_t;
+
+/**
+ * Private data of an retransmit_request_job_t Object.
+ */
+struct private_retransmit_request_job_t {
+ /**
+ * Public retransmit_request_job_t interface.
+ */
+ retransmit_request_job_t public;
+
+ /**
+ * Message ID of the request to resend.
+ */
+ u_int32_t message_id;
+
+ /**
+ * ID of the IKE_SA which the message belongs to.
+ */
+ ike_sa_id_t *ike_sa_id;
+};
+
+
+/**
+ * Implements job_t.get_type.
+ */
+static job_type_t get_type(private_retransmit_request_job_t *this)
+{
+ return RETRANSMIT_REQUEST;
+}
+
+/**
+ * Implements retransmit_request_job_t.get_ike_sa_id.
+ */
+static ike_sa_id_t *get_ike_sa_id(private_retransmit_request_job_t *this)
+{
+ return this->ike_sa_id;
+}
+
+/**
+ * Implements retransmit_request_job_t.get_message_id.
+ */
+static u_int32_t get_message_id(private_retransmit_request_job_t *this)
+{
+ return this->message_id;
+}
+
+
+/**
+ * Implements job_t.destroy.
+ */
+static void destroy(private_retransmit_request_job_t *this)
+{
+ this->ike_sa_id->destroy(this->ike_sa_id);
+ allocator_free(this);
+}
+
+/*
+ * Described in header.
+ */
+retransmit_request_job_t *retransmit_request_job_create(u_int32_t message_id,ike_sa_id_t *ike_sa_id)
+{
+ private_retransmit_request_job_t *this = allocator_alloc_thing(private_retransmit_request_job_t);
+
+ /* interface functions */
+ this->public.job_interface.get_type = (job_type_t (*) (job_t *)) get_type;
+ /* same as destroy */
+ this->public.job_interface.destroy_all = (void (*) (job_t *)) destroy;
+ this->public.job_interface.destroy = (void (*) (job_t *)) destroy;
+
+ /* public functions */
+ this->public.get_ike_sa_id = (ike_sa_id_t * (*)(retransmit_request_job_t *)) get_ike_sa_id;
+ this->public.get_message_id = (u_int32_t (*)(retransmit_request_job_t *)) get_message_id;
+ this->public.destroy = (void (*)(retransmit_request_job_t *)) destroy;
+
+ /* private variables */
+ this->message_id = message_id;
+ this->ike_sa_id = ike_sa_id->clone(ike_sa_id);
+
+ return &(this->public);
+}
diff --git a/Source/charon/queues/jobs/retransmit_request_job.h b/Source/charon/queues/jobs/retransmit_request_job.h
new file mode 100644
index 000000000..591c79864
--- /dev/null
+++ b/Source/charon/queues/jobs/retransmit_request_job.h
@@ -0,0 +1,83 @@
+/**
+ * @file retransmit_request_job.c
+ *
+ * @brief Interface of retransmit_request_job_t.
+ *
+ */
+
+/*
+ * 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.
+ */
+
+#ifndef _RESEND_MESSAGE_JOB_H_
+#define _RESEND_MESSAGE_JOB_H_
+
+#include <types.h>
+#include <queues/jobs/job.h>
+#include <sa/ike_sa_id.h>
+
+
+typedef struct retransmit_request_job_t retransmit_request_job_t;
+
+/**
+ * Object representing an RETRANSMIT_REQUEST Job.
+ *
+ * @ingroup jobs
+ */
+struct retransmit_request_job_t {
+ /**
+ * The job_t interface.
+ */
+ job_t job_interface;
+
+ /**
+ * @brief Returns the message_id of the request to be resent
+ *
+ * @param this calling retransmit_request_job_t object
+ * @return message id of the request to resend
+ */
+ u_int32_t (*get_message_id) (retransmit_request_job_t *this);
+
+ /**
+ * @brief Returns the ike_sa_id_t object of the IKE_SA
+ * which the request belongs to
+ *
+ * @warning returned ike_sa_id_t object is getting destroyed in
+ * retransmit_request_job_t.destroy.
+ *
+ * @param this calling retransmit_request_job_t object
+ * @return ike_sa_id_t object to identify IKE_SA (gets NOT cloned)
+ */
+ ike_sa_id_t *(*get_ike_sa_id) (retransmit_request_job_t *this);
+
+ /**
+ * @brief Destroys an retransmit_request_job_t object.
+ *
+ * @param this retransmit_request_job_t object to destroy
+ */
+ void (*destroy) (retransmit_request_job_t *this);
+};
+
+/**
+ * @brief Creates a job of type RETRANSMIT_REQUEST.
+ *
+ * @param message_id message_id of the request to resend
+ * @param ike_sa_id identification of the ike_sa as ike_sa_id_t object (gets cloned)
+ * @return retransmit_request_job_t object
+ *
+ * @ingroup jobs
+ */
+retransmit_request_job_t *retransmit_request_job_create(u_int32_t message_id,ike_sa_id_t *ike_sa_id);
+
+#endif //_RESEND_MESSAGE_JOB_H_
diff --git a/Source/charon/sa/ike_sa.c b/Source/charon/sa/ike_sa.c
index 36ed8ee9a..46807de51 100644
--- a/Source/charon/sa/ike_sa.c
+++ b/Source/charon/sa/ike_sa.c
@@ -407,23 +407,36 @@ static void compute_secrets(private_ike_sa_t *this,chunk_t dh_shared_secret,chun
}
/**
- * Implements protected_ike_sa_t.resend_last_reply.
+ * Implementation of private_ike_sa_t.resend_last_reply.
*/
static status_t resend_last_reply(private_ike_sa_t *this)
{
packet_t *packet;
- status_t status;
- status = this->last_responded_message->generate(this->last_responded_message, NULL, NULL, &packet);
- if (status != SUCCESS)
+ packet = this->last_responded_message->get_packet(this->last_responded_message);
+ charon->send_queue->add(charon->send_queue, packet);
+
+ return SUCCESS;
+}
+
+/**
+ * Implementation of ike_sa_t.retransmit_request.
+ */
+status_t retransmit_request (private_ike_sa_t *this, u_int32_t message_id)
+{
+ packet_t *packet;
+
+ if ((this->message_id_out -1) != message_id)
{
- this->logger->log(this->logger, ERROR, "Could not generate message to resent");
- return status;
+ return NOT_FOUND;
}
+ packet = this->last_responded_message->get_packet(this->last_responded_message);
charon->send_queue->add(charon->send_queue, packet);
+
return SUCCESS;
}
+
/**
* Implements protected_ike_sa_t.resend_last_reply.
@@ -615,6 +628,9 @@ static signer_t *get_signer_initiator (private_ike_sa_t *this)
}
/**
+<<<<<<< .mine
+ * Implementation of protected_ike_sa_t.send_request.
+=======
* Implementation of protected_ike_sa_t.get_crypter_responder.
*/
static crypter_t *get_crypter_responder(private_ike_sa_t *this)
@@ -633,50 +649,82 @@ static signer_t *get_signer_responder (private_ike_sa_t *this)
/**
* Implementation of protected_ike_sa_t.set_last_requested_message.
+>>>>>>> .r660
*/
-static status_t set_last_requested_message (private_ike_sa_t *this,message_t * message)
+static status_t send_request (private_ike_sa_t *this,message_t * message)
{
- if (this->last_requested_message != NULL)
+ packet_t *packet;
+ status_t status;
+
+ if (message->get_message_id(message) != this->message_id_out)
{
- /* destroy message */
- this->last_requested_message->destroy(this->last_requested_message);
+ this->logger->log(this->logger, ERROR, "Message could not be sent cause id was not as expected");
+ return FAILED;
}
- if (message->get_message_id(message) != this->message_id_out)
+ /* generate packet */
+ this->logger->log(this->logger, CONTROL|MOST, "Generate packet from message");
+
+ status = message->generate(message, this->crypter_initiator,this->signer_initiator, &packet);
+ if (status != SUCCESS)
{
- this->logger->log(this->logger, CONTROL|MOST, "last requested message could not be set cause id was not as expected");
+ this->logger->log(this->logger, ERROR, "Could not generate packet from message");
return FAILED;
}
+
+ this->logger->log(this->logger, CONTROL|MOST, "Add packet to global send queue");
+ charon->send_queue->add(charon->send_queue, packet);
+
+ if (this->last_requested_message != NULL)
+ {
+ /* destroy message */
+ this->last_requested_message->destroy(this->last_requested_message);
+ }
+
this->logger->log(this->logger, CONTROL|MOST, "replace last requested message with new one");
this->last_requested_message = message;
/* message counter can now be increased */
- this->logger->log(this->logger, CONTROL|MOST, "Increate message counter for outgoing messages");
+ this->logger->log(this->logger, CONTROL|MOST, "Increase message counter for outgoing messages");
this->message_id_out++;
return SUCCESS;
}
/**
- * Implementation of protected_ike_sa_t.set_last_responded_message.
+ * Implementation of protected_ike_sa_t.send_response.
*/
-static status_t set_last_responded_message (private_ike_sa_t *this,message_t * message)
+static status_t send_response (private_ike_sa_t *this,message_t * message)
{
- if (this->last_responded_message != NULL)
+ packet_t *packet;
+ status_t status;
+
+ if (message->get_message_id(message) != this->message_id_in)
{
- /* destroy message */
- this->last_responded_message->destroy(this->last_responded_message);
+ this->logger->log(this->logger, CONTROL|MOST, "Message could not be sent cause id was not as expected");
+ return FAILED;
}
- if (message->get_message_id(message) != this->message_id_in)
+
+ status = message->generate(message, this->crypter_initiator,this->signer_initiator, &packet);
+ if (status != SUCCESS)
{
- this->logger->log(this->logger, CONTROL|MOST, "last responded message could not be set cause id was not as expected");
+ this->logger->log(this->logger, ERROR, "Could not generate packet from message");
return FAILED;
-
}
+
+ this->logger->log(this->logger, CONTROL|MOST, "Add packet to global send queue");
+ charon->send_queue->add(charon->send_queue, packet);
+
+ if (this->last_responded_message != NULL)
+ {
+ /* destroy message */
+ this->last_responded_message->destroy(this->last_responded_message);
+ }
+
this->logger->log(this->logger, CONTROL|MOST, "replace last responded message with new one");
this->last_responded_message = message;
/* message counter can now be increased */
- this->logger->log(this->logger, CONTROL|MOST, "Increate message counter for incoming messages");
+ this->logger->log(this->logger, CONTROL|MOST, "Increase message counter for incoming messages");
this->message_id_in++;
return SUCCESS;
@@ -811,6 +859,7 @@ ike_sa_t * ike_sa_create(ike_sa_id_t *ike_sa_id)
this->protected.public.process_message = (status_t(*)(ike_sa_t*, message_t*)) process_message;
this->protected.public.initialize_connection = (status_t(*)(ike_sa_t*, char*)) initialize_connection;
this->protected.public.get_id = (ike_sa_id_t*(*)(ike_sa_t*)) get_id;
+ this->protected.public.retransmit_request = (status_t (*) (ike_sa_t *, u_int32_t)) retransmit_request;
this->protected.public.destroy = (void(*)(ike_sa_t*))destroy;
/* protected functions */
@@ -826,8 +875,8 @@ ike_sa_t * ike_sa_create(ike_sa_id_t *ike_sa_id)
this->protected.set_my_host = (void(*) (protected_ike_sa_t *,host_t *)) set_my_host;
this->protected.set_other_host = (void(*) (protected_ike_sa_t *, host_t *)) set_other_host;
this->protected.get_randomizer = (randomizer_t *(*) (protected_ike_sa_t *)) get_randomizer;
- this->protected.set_last_requested_message = (status_t (*) (protected_ike_sa_t *,message_t *)) set_last_requested_message;
- this->protected.set_last_responded_message = (status_t (*) (protected_ike_sa_t *,message_t *)) set_last_responded_message;
+ this->protected.send_request = (status_t (*) (protected_ike_sa_t *,message_t *)) send_request;
+ this->protected.send_response = (status_t (*) (protected_ike_sa_t *,message_t *)) send_response;
this->protected.create_transforms_from_proposal = (status_t (*) (protected_ike_sa_t *,ike_proposal_t *)) create_transforms_from_proposal;
this->protected.set_new_state = (void (*) (protected_ike_sa_t *,state_t *)) set_new_state;
this->protected.get_crypter_initiator = (crypter_t *(*) (protected_ike_sa_t *)) get_crypter_initiator;
@@ -840,9 +889,6 @@ ike_sa_t * ike_sa_create(ike_sa_id_t *ike_sa_id)
this->resend_last_reply = resend_last_reply;
this->create_delete_job = create_delete_job;
-
-
-
/* initialize private fields */
this->logger = charon->logger_manager->create_logger(charon->logger_manager, IKE_SA, NULL);
diff --git a/Source/charon/sa/ike_sa.h b/Source/charon/sa/ike_sa.h
index 806fd6581..6a12aaf8d 100644
--- a/Source/charon/sa/ike_sa.h
+++ b/Source/charon/sa/ike_sa.h
@@ -71,6 +71,17 @@ struct ike_sa_t {
* @return TODO
*/
status_t (*initialize_connection) (ike_sa_t *this, char *name);
+
+ /**
+ * @brief Retransmits a request.
+ *
+ * @param this calling object
+ * @param message_id ID of the request to retransmit
+ * @return
+ * - SUCCESS
+ * - NOT_FOUND if request doesn't have to be retransmited
+ */
+ status_t (*retransmit_request) (ike_sa_t *this, u_int32_t message_id);
/**
* @brief Get the id of the SA.
@@ -223,30 +234,32 @@ struct protected_ike_sa_t {
status_t (*create_transforms_from_proposal) (protected_ike_sa_t *this,ike_proposal_t * proposal);
/**
- * Sets the last requested message.
+ * Sends the next request message.
+ *
+ * Also the first retransmit job is created.
*
- * Allready set last requested message gets destroyed. object gets not cloned!
+ * Stored requested message gets destroyed. object gets not cloned!
*
* @param this calling object
- * @param message pointer to the new last requested message
+ * @param message pointer to the message which should be sent
* @return
* - SUCCESS
* - FAILED if message id is not next expected one
*/
- status_t (*set_last_requested_message) (protected_ike_sa_t *this,message_t * message);
+ status_t (*send_request) (protected_ike_sa_t *this,message_t * message);
/**
- * Sets the last responded message.
+ * Sends the next response message.
*
- * Allready set last requested message gets destroyed. object gets not cloned!
+ * Stored responded message gets destroyed. object gets not cloned!
*
* @param this calling object
- * @param message pointer to the new last responded message
+ * @param message pointer to the message which should be sent
* return
* - SUCCESS
* - FAILED if message id is not next expected one
*/
- status_t (*set_last_responded_message) (protected_ike_sa_t *this,message_t * message);
+ status_t (*send_response) (protected_ike_sa_t *this,message_t * message);
/**
* Gets the internal stored randomizer_t object.
diff --git a/Source/charon/sa/states/ike_sa_init_requested.c b/Source/charon/sa/states/ike_sa_init_requested.c
index 424f75950..62fa172a3 100644
--- a/Source/charon/sa/states/ike_sa_init_requested.c
+++ b/Source/charon/sa/states/ike_sa_init_requested.c
@@ -162,7 +162,6 @@ static status_t process_message(private_ike_sa_init_requested_t *this, message_t
ike_sa_id_t *ike_sa_id;
iterator_t *payloads;
message_t *request;
- packet_t *packet;
status_t status;
/*
@@ -384,34 +383,18 @@ static status_t process_message(private_ike_sa_init_requested_t *this, message_t
/* build the complete IKE_AUTH request */
this->build_ike_auth_request (this,&request);
- /* generate packet */
- this->logger->log(this->logger, CONTROL|MOST, "Generate packet from message");
-
- status = request->generate(request, this->ike_sa->get_crypter_initiator(this->ike_sa), this->ike_sa->get_signer_initiator(this->ike_sa), &packet);
+ /* message can now be sent (must not be destroyed) */
+ status = this->ike_sa->send_request(this->ike_sa, request);
if (status != SUCCESS)
{
- this->logger->log(this->logger, ERROR, "Could not generate packet from message");
+ this->logger->log(this->logger, ERROR, "Could not send request message");
request->destroy(request);
return DELETE_ME;
}
-
- this->logger->log(this->logger, CONTROL|MOST, "Add packet to global send queue");
- charon->send_queue->add(charon->send_queue, packet);
-
+
/* state can now be changed */
this->logger->log(this->logger, CONTROL|MOST, "Create next state object");
next_state = ike_auth_requested_create(this->ike_sa,this->sent_nonce,this->received_nonce);
-
- /* last messages can now be set */
- status = this->ike_sa->set_last_requested_message(this->ike_sa, request);
-
- if (status != SUCCESS)
- {
- this->logger->log(this->logger, ERROR, "Could not set last requested message");
- (next_state->state_interface).destroy(&(next_state->state_interface));
- request->destroy(request);
- return DELETE_ME;
- }
/* state can now be changed */
this->ike_sa->set_new_state(this->ike_sa,(state_t *) next_state);
@@ -579,7 +562,7 @@ static void destroy_after_state_change (private_ike_sa_init_requested_t *this)
this->logger->log(this->logger, CONTROL | MOST, "Destroy diffie hellman object");
this->diffie_hellman->destroy(this->diffie_hellman);
this->logger->log(this->logger, CONTROL | MOST, "Destroy shared secret (secrets allready derived)");
- allocator_free(this->shared_secret.ptr);
+ allocator_free_chunk(&(this->shared_secret));
this->logger->log(this->logger, CONTROL | MOST, "Destroy object itself");
allocator_free(this);
}
@@ -598,7 +581,7 @@ static void destroy(private_ike_sa_init_requested_t *this)
this->logger->log(this->logger, CONTROL | MOST, "Destroy received nonce");
allocator_free(this->received_nonce.ptr);
this->logger->log(this->logger, CONTROL | MOST, "Destroy shared secret (secrets allready derived)");
- allocator_free(this->shared_secret.ptr);
+ allocator_free_chunk(&(this->shared_secret));
this->logger->log(this->logger, CONTROL | MOST, "Destroy object itself");
allocator_free(this);
}
diff --git a/Source/charon/sa/states/ike_sa_init_responded.c b/Source/charon/sa/states/ike_sa_init_responded.c
index 58d5ed506..f97bae87e 100644
--- a/Source/charon/sa/states/ike_sa_init_responded.c
+++ b/Source/charon/sa/states/ike_sa_init_responded.c
@@ -83,7 +83,6 @@ static status_t process_message(private_ike_sa_init_responded_t *this, message_t
sa_payload_t *sa_request;
ts_payload_t *tsi_request, *tsr_request;
message_t *response;
- packet_t *response_packet;
exchange_type = request->get_exchange_type(request);
if (exchange_type != IKE_AUTH)
@@ -210,25 +209,18 @@ static status_t process_message(private_ike_sa_init_responded_t *this, message_t
this->logger->log(this->logger, ERROR, "Building tsr payload failed");
response->destroy(response);
return status;
- }
-
- /* generate response, get transfroms first */
- signer = this->ike_sa->get_signer_responder(this->ike_sa);
- crypter = this->ike_sa->get_crypter_responder(this->ike_sa);
- status = response->generate(response, crypter, signer, &response_packet);
+ }
+
+ this->logger->log(this->logger, CONTROL | MORE, "IKE_AUTH request successfully handled. Sending reply.");
+ status = this->ike_sa->send_response(this->ike_sa, response);
+
+ /* message can now be sent (must not be destroyed) */
if (status != SUCCESS)
{
- this->logger->log(this->logger, ERROR, "Error in message generation");
+ this->logger->log(this->logger, ERROR, "Could not send response message");
response->destroy(response);
- return status;
+ return DELETE_ME;
}
-
-
- /* send it out */
- this->logger->log(this->logger, CONTROL | MORE, "IKE_AUTH request successfully handled. Sending reply.");
- charon->send_queue->add(charon->send_queue, response_packet);
- /* store for timeout reply */
- this->ike_sa->set_last_responded_message(this->ike_sa, response);
/* create new state */
this->ike_sa->set_new_state(this->ike_sa, (state_t*)ike_sa_established_create(this->ike_sa));
diff --git a/Source/charon/sa/states/initiator_init.c b/Source/charon/sa/states/initiator_init.c
index 7c63a4587..59a6a2e2a 100644
--- a/Source/charon/sa/states/initiator_init.c
+++ b/Source/charon/sa/states/initiator_init.c
@@ -27,6 +27,7 @@
#include <sa/states/state.h>
#include <sa/states/ike_sa_init_requested.h>
#include <utils/allocator.h>
+#include <queues/jobs/retransmit_request_job.h>
#include <transforms/diffie_hellman.h>
#include <encoding/payloads/sa_payload.h>
#include <encoding/payloads/ke_payload.h>
@@ -179,7 +180,6 @@ status_t retry_initiate_connection (private_initiator_init_t *this, int dh_group
init_config_t *init_config;
randomizer_t *randomizer;
message_t *message;
- packet_t *packet;
status_t status;
ike_sa_id_t *ike_sa_id;
@@ -201,49 +201,36 @@ status_t retry_initiate_connection (private_initiator_init_t *this, int dh_group
this->logger->log(this->logger, CONTROL|MOST, "Get pseudo random bytes for nonce");
randomizer = this->ike_sa->get_randomizer(this->ike_sa);
+
+ allocator_free_chunk(&(this->sent_nonce));
+
randomizer->allocate_pseudo_random_bytes(randomizer, NONCE_SIZE, &(this->sent_nonce));
this->logger->log(this->logger, RAW|MOST, "Nonce",&(this->sent_nonce));
this->build_ike_sa_init_request (this,&message);
- /* generate packet */
- this->logger->log(this->logger, CONTROL|MOST, "generate packet from message");
- status = message->generate(message, NULL, NULL, &packet);
+ /* message can now be sent (must not be destroyed) */
+ status = this->ike_sa->send_request(this->ike_sa, message);
if (status != SUCCESS)
{
- this->logger->log(this->logger, ERROR, "could not generate packet from message");
+ this->logger->log(this->logger, ERROR, "Could not send request message");
message->destroy(message);
return DELETE_ME;
}
-
- this->logger->log(this->logger, CONTROL|MOST, "Add packet to global send queue");
- charon->send_queue->add(charon->send_queue, packet);
/* state can now be changed */
this->logger->log(this->logger, CONTROL|MOST, "Create next state object");
next_state = ike_sa_init_requested_create(this->ike_sa, this->dh_group_priority, this->diffie_hellman, this->sent_nonce);
- /* last message can now be set */
- status = this->ike_sa->set_last_requested_message(this->ike_sa, message);
-
- if (status != SUCCESS)
- {
- this->logger->log(this->logger, ERROR, "Could not set last requested message");
- (next_state->state_interface).destroy(&(next_state->state_interface));
- message->destroy(message);
- return DELETE_ME;
- }
-
/* state can now be changed */
this->ike_sa->set_new_state(this->ike_sa,(state_t *) next_state);
/* state has NOW changed :-) */
this->logger->log(this->logger, CONTROL|MORE, "Changed state of IKE_SA from %s to %s", mapping_find(ike_sa_state_m,INITIATOR_INIT),mapping_find(ike_sa_state_m,IKE_SA_INIT_REQUESTED) );
-
+
this->logger->log(this->logger, CONTROL|MOST, "Destroy old sate object");
this->destroy_after_state_change(this);
-
return SUCCESS;
}
diff --git a/Source/charon/sa/states/responder_init.c b/Source/charon/sa/states/responder_init.c
index 3c03adf41..9409d57df 100644
--- a/Source/charon/sa/states/responder_init.c
+++ b/Source/charon/sa/states/responder_init.c
@@ -163,7 +163,6 @@ static status_t process_message(private_responder_init_t *this, message_t *messa
iterator_t *payloads;
message_t *response;
host_t *other_host;
- packet_t *packet;
host_t *my_host;
status_t status;
@@ -344,36 +343,27 @@ static status_t process_message(private_responder_init_t *this, message_t *messa
this->ike_sa->compute_secrets(this->ike_sa,shared_secret,this->received_nonce, this->sent_nonce);
+ /* not used anymore */
+ allocator_free_chunk(&shared_secret);
+
this->build_ike_sa_init_reply(this,&response);
-
- /* generate packet */
- this->logger->log(this->logger, CONTROL|MOST, "generate packet from message");
- status = response->generate(response, NULL, NULL, &packet);
+
+ /* message can now be sent (must not be destroyed) */
+ status = this->ike_sa->send_response(this->ike_sa, response);
if (status != SUCCESS)
{
- this->logger->log(this->logger, ERROR, "could not generate packet from message");
+ this->logger->log(this->logger, ERROR, "Could not send response message");
+ response->destroy(response);
return DELETE_ME;
}
-
- this->logger->log(this->logger, CONTROL|MOST, "Add packet to global send queue");
- charon->send_queue->add(charon->send_queue, packet);
+
+
/* state can now be changed */
this->logger->log(this->logger, CONTROL|MOST, "Create next state object");
next_state = ike_sa_init_responded_create(this->ike_sa);
- /* last message can now be set */
- status = this->ike_sa->set_last_responded_message(this->ike_sa, response);
-
- if (status != SUCCESS)
- {
- this->logger->log(this->logger, ERROR, "Could not set last responded message");
- response->destroy(response);
- (next_state->state_interface).destroy(&(next_state->state_interface));
- return DELETE_ME;
- }
-
/* state can now be changed */
this->ike_sa->set_new_state(this->ike_sa, (state_t *) next_state);
/* state has NOW changed :-) */
@@ -517,9 +507,9 @@ static void destroy(private_responder_init_t *this)
this->logger->log(this->logger, CONTROL | MORE, "Going to destroy responder init state object");
this->logger->log(this->logger, CONTROL | MOST, "Destroy sent nonce");
- allocator_free(this->sent_nonce.ptr);
+ allocator_free_chunk(&(this->sent_nonce));
this->logger->log(this->logger, CONTROL | MOST, "Destroy received nonce");
- allocator_free(this->received_nonce.ptr);
+ allocator_free_chunk(&(this->received_nonce));
if (this->diffie_hellman != NULL)
{
@@ -543,6 +533,11 @@ static void destroy_after_state_change (private_responder_init_t *this)
this->logger->log(this->logger, CONTROL | MOST, "Destroy diffie_hellman_t object");
this->diffie_hellman->destroy(this->diffie_hellman);
}
+
+ this->logger->log(this->logger, CONTROL | MOST, "Destroy sent nonce");
+ allocator_free_chunk(&(this->sent_nonce));
+ this->logger->log(this->logger, CONTROL | MOST, "Destroy received nonce");
+ allocator_free_chunk(&(this->received_nonce));
this->logger->log(this->logger, CONTROL | MOST, "Destroy object");
allocator_free(this);
diff --git a/Source/charon/threads/thread_pool.c b/Source/charon/threads/thread_pool.c
index f6f5278f5..649679fc3 100644
--- a/Source/charon/threads/thread_pool.c
+++ b/Source/charon/threads/thread_pool.c
@@ -33,6 +33,7 @@
#include <queues/jobs/delete_ike_sa_job.h>
#include <queues/jobs/incoming_packet_job.h>
#include <queues/jobs/initiate_ike_sa_job.h>
+#include <queues/jobs/retransmit_request_job.h>
#include <utils/allocator.h>
#include <utils/logger.h>
@@ -58,27 +59,38 @@ struct private_thread_pool_t {
void (*process_jobs) (private_thread_pool_t *this);
/**
- * @brief Process a INCOMING_PACKET_JOB.
+ * @brief Process a INCOMING_PACKET job.
*
- * @param this private_thread_pool_t-Object
+ * @param this private_thread_pool_t object
+ * @param job incoming_packet_job_t object
*/
void (*process_incoming_packet_job) (private_thread_pool_t *this, incoming_packet_job_t *job);
/**
- * @brief Process a INITIATE_IKE_SA_JOB.
+ * @brief Process a INITIATE_IKE_SA job.
*
- * @param this private_thread_pool_t-Object
+ * @param this private_thread_pool_t object
+ * @param job initiate_ike_sa_job_t object
*/
void (*process_initiate_ike_sa_job) (private_thread_pool_t *this, initiate_ike_sa_job_t *job);
/**
- * @brief Process a DELETE_IKE_SA_JOB.
+ * @brief Process a DELETE_IKE_SA job.
*
- * @param this private_thread_pool_t-Object
+ * @param this private_thread_pool_t object
+ * @param job delete_ike_sa_job_t object
*/
void (*process_delete_ike_sa_job) (private_thread_pool_t *this, delete_ike_sa_job_t *job);
/**
+ * @brief Process a RETRANSMIT_REQUEST job.
+ *
+ * @param this private_thread_pool_t object
+ * @param job retransmit_request_job_t object
+ */
+ void (*process_retransmit_request_job) (private_thread_pool_t *this, retransmit_request_job_t *job);
+
+ /**
* number of running threads
*/
size_t pool_size;
@@ -99,10 +111,8 @@ struct private_thread_pool_t {
logger_t *worker_logger;
} ;
-
-
/**
- * implements private_thread_pool_t.function
+ * Implementation of private_thread_pool_t.process_jobs.
*/
static void process_jobs(private_thread_pool_t *this)
{
@@ -117,7 +127,7 @@ static void process_jobs(private_thread_pool_t *this)
job = charon->job_queue->get(charon->job_queue);
job_type = job->get_type(job);
- this->worker_logger->log(this->worker_logger, CONTROL|MORE, "got a job of type %s",
+ this->worker_logger->log(this->worker_logger, CONTROL|MORE, "Process job of type %s",
mapping_find(job_type_m,job_type));
switch (job_type)
@@ -125,31 +135,44 @@ static void process_jobs(private_thread_pool_t *this)
case INCOMING_PACKET:
{
this->process_incoming_packet_job(this, (incoming_packet_job_t*)job);
+ job->destroy(job);
break;
}
case INITIATE_IKE_SA:
{
this->process_initiate_ike_sa_job(this, (initiate_ike_sa_job_t*)job);
+ job->destroy(job);
break;
}
case DELETE_IKE_SA:
{
this->process_delete_ike_sa_job(this, (delete_ike_sa_job_t*)job);
+ job->destroy(job);
+ break;
+ }
+ case RETRANSMIT_REQUEST:
+ {
+ this->process_retransmit_request_job(this, (retransmit_request_job_t*)job);
+ job->destroy(job);
break;
}
default:
{
this->worker_logger->log(this->worker_logger, ERROR, "job of type %s not supported!",
mapping_find(job_type_m,job_type));
+ job->destroy(job);
break;
}
}
- job->destroy(job);
+
+ this->worker_logger->log(this->worker_logger, CONTROL|MORE, "Processing of job finished");
+
+
}
}
/**
- * implementation of private_thread_pool_t.process_incoming_packet_job
+ * Implementation of private_thread_pool_t.process_incoming_packet_job.
*/
static void process_incoming_packet_job(private_thread_pool_t *this, incoming_packet_job_t *job)
{
@@ -240,7 +263,7 @@ static void process_incoming_packet_job(private_thread_pool_t *this, incoming_pa
}
/**
- * implementation of private_thread_pool_t.process_initiate_ike_sa_job
+ * Implementation of private_thread_pool_t.process_initiate_ike_sa_job.
*/
static void process_initiate_ike_sa_job(private_thread_pool_t *this, initiate_ike_sa_job_t *job)
{
@@ -279,7 +302,7 @@ static void process_initiate_ike_sa_job(private_thread_pool_t *this, initiate_ik
}
/**
- * implementation of private_thread_pool_t.process_delete_ike_sa_job
+ * Implementation of private_thread_pool_t.process_delete_ike_sa_job.
*/
static void process_delete_ike_sa_job(private_thread_pool_t *this, delete_ike_sa_job_t *job)
{
@@ -299,9 +322,54 @@ static void process_delete_ike_sa_job(private_thread_pool_t *this, delete_ike_sa
}
}
+/**
+ * Implementation of private_thread_pool_t.process_retransmit_request_job.
+ */
+static void process_retransmit_request_job(private_thread_pool_t *this, retransmit_request_job_t *job)
+{
+ status_t status;
+ ike_sa_id_t *ike_sa_id = job->get_ike_sa_id(job);
+ u_int32_t message_id = job->get_message_id(job);
+ ike_sa_t *ike_sa;
+
+ this->worker_logger->log(this->worker_logger, CONTROL|MOST, "checking out IKE SA %lld:%lld, role %s",
+ ike_sa_id->get_initiator_spi(ike_sa_id),
+ ike_sa_id->get_responder_spi(ike_sa_id),
+ ike_sa_id->is_initiator(ike_sa_id) ? "initiator" : "responder");
+
+ status = charon->ike_sa_manager->checkout(charon->ike_sa_manager,ike_sa_id, &ike_sa);
+ if (status != SUCCESS)
+ {
+ this->worker_logger->log(this->worker_logger, ERROR, "IKE SA could not be checked out. Allready deleted?");
+ return;
+ }
+
+ status = ike_sa->retransmit_request(ike_sa, message_id);
+
+ if (status != SUCCESS)
+ {
+ this->worker_logger->log(this->worker_logger, CONTROL | MOST, "Message does'nt have to be retransmitted");
+ }
+
+ this->worker_logger->log(this->worker_logger, CONTROL|MOST, "Checkin IKE SA %lld:%lld, role %s",
+ ike_sa_id->get_initiator_spi(ike_sa_id),
+ ike_sa_id->get_responder_spi(ike_sa_id),
+ ike_sa_id->is_initiator(ike_sa_id) ? "initiator" : "responder");
+
+ status = charon->ike_sa_manager->checkin(charon->ike_sa_manager, ike_sa);
+ if (status != SUCCESS)
+ {
+ this->worker_logger->log(this->worker_logger, ERROR, "Checkin of IKE SA failed!");
+ }
+/*
+ u_int32_t message_id = message->get_message_id(message);
+ retransmit_request_job_t *new_job = retransmit_request_job_create(message_id,ike_sa_id);
+ charon->event_queue->add_relative(charon->event_queue,(job_t *) new_job,5000);*/
+
+}
/**
- * implementation of thread_pool_t.get_pool_size
+ * Implementation of thread_pool_t.get_pool_size.
*/
static size_t get_pool_size(private_thread_pool_t *this)
{
@@ -309,7 +377,7 @@ static size_t get_pool_size(private_thread_pool_t *this)
}
/**
- * Implementation of thread_pool_t.destroy
+ * Implementation of thread_pool_t.destroy.
*/
static void destroy(private_thread_pool_t *this)
{
@@ -334,7 +402,7 @@ static void destroy(private_thread_pool_t *this)
}
/*
- * see header
+ * Described in header.
*/
thread_pool_t *thread_pool_create(size_t pool_size)
{
@@ -350,6 +418,7 @@ thread_pool_t *thread_pool_create(size_t pool_size)
this->process_initiate_ike_sa_job = process_initiate_ike_sa_job;
this->process_delete_ike_sa_job = process_delete_ike_sa_job;
this->process_incoming_packet_job = process_incoming_packet_job;
+ this->process_retransmit_request_job = process_retransmit_request_job;
this->pool_size = pool_size;
this->threads = allocator_alloc(sizeof(pthread_t) * pool_size);