diff options
Diffstat (limited to 'Source/charon/sa/ike_sa.c')
-rw-r--r-- | Source/charon/sa/ike_sa.c | 574 |
1 files changed, 574 insertions, 0 deletions
diff --git a/Source/charon/sa/ike_sa.c b/Source/charon/sa/ike_sa.c new file mode 100644 index 000000000..9747391f4 --- /dev/null +++ b/Source/charon/sa/ike_sa.c @@ -0,0 +1,574 @@ +/** + * @file ike_sa.c + * + * @brief Class ike_sa_t. An object of this type is managed by an + * ike_sa_manager_t object and represents an IKE_SA + * + */ + +/* + * 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 "ike_sa.h" + +#include <types.h> +#include <globals.h> +#include <definitions.h> +#include <utils/allocator.h> +#include <utils/linked_list.h> +#include <utils/logger_manager.h> +#include <utils/randomizer.h> +#include <transforms/diffie_hellman.h> +#include <transforms/prf_plus.h> +#include <encoding/payloads/sa_payload.h> +#include <encoding/payloads/nonce_payload.h> +#include <encoding/payloads/ke_payload.h> +#include <encoding/payloads/transform_substructure.h> +#include <encoding/payloads/transform_attribute.h> +#include <sa/states/initiator_init.h> +#include <sa/states/responder_init.h> + + + + +/** + * @brief implements function process_message of protected_ike_sa_t + */ +static status_t process_message (protected_ike_sa_t *this, message_t *message) +{ + u_int32_t message_id; + exchange_type_t exchange_type; + bool is_request; + status_t status; + state_t *new_state; + + /* we must process each request or response from remote host */ + + /* find out type of message (request or response) */ + is_request = message->get_request(message); + exchange_type = message->get_exchange_type(message); + + this->logger->log(this->logger, CONTROL|MORE, "Process %s message of exchange type %s",(is_request) ? "REQUEST" : "RESPONSE",mapping_find(exchange_type_m,exchange_type)); + + message_id = message->get_message_id(message); + + /* + * It has to be checked, if the message has to be resent cause of lost packets! + */ + if (is_request && ( message_id == (this->message_id_in - 1))) + { + /* message can be resent ! */ + this->logger->log(this->logger, CONTROL|MORE, "Resent message detected. Send stored reply"); + return (this->resend_last_reply(this)); + } + + /* Now, the message id is checked for request AND reply */ + if (is_request) + { + /* In a request, the message has to be this->message_id_in (other case is already handled) */ + if (message_id != this->message_id_in) + { + this->logger->log(this->logger, ERROR | MORE, "Message request with message id %d received, but %d expected",message_id,this->message_id_in); + return FAILED; + } + } + else + { + /* In a reply, the message has to be this->message_id_out -1 cause it is the reply to the last sent message*/ + if (message_id != (this->message_id_out - 1)) + { + this->logger->log(this->logger, ERROR | MORE, "Message reply with message id %d received, but %d expected",message_id,this->message_id_in); + return FAILED; + } + } + + /* now the message is processed by the current state object */ + status = this->current_state->process_message(this->current_state,message,&new_state); + + if (status == SUCCESS) + { + this->current_state = new_state; + } + return status; +} + +/** + * @brief Implements function build_message of protected_ike_sa_t. + */ +static status_t build_message(protected_ike_sa_t *this, exchange_type_t type, bool request, message_t **message) +{ + status_t status; + message_t *new_message; + host_t *source, *destination; + + this ->logger->log(this->logger, CONTROL|MORE, "build empty message"); + new_message = message_create(); + if (new_message == NULL) + { + this->logger->log(this->logger, ERROR, "Fatal error: could not create empty message object"); + return OUT_OF_RES; + } + + status = this->me.host->clone(this->me.host, &source); + if (status != SUCCESS) + { + this->logger->log(this->logger, ERROR, "Fatal error: could not clone my host information"); + new_message->destroy(new_message); + return status; + } + status = this->other.host->clone(this->other.host, &destination); + if (status != SUCCESS) + { + this->logger->log(this->logger, ERROR, "Fatal error: could not clone other host information"); + source->destroy(source); + new_message->destroy(new_message); + return status; + } + + new_message->set_source(new_message, source); + new_message->set_destination(new_message, destination); + + new_message->set_exchange_type(new_message, type); + new_message->set_request(new_message, request); + + new_message->set_message_id(new_message, (request) ? this->message_id_out : this->message_id_in); + + status = new_message->set_ike_sa_id(new_message, this->ike_sa_id); + if (status != SUCCESS) + { + this->logger->log(this->logger, ERROR, "Fatal error: could not set ike_sa_id of message"); + new_message->destroy(new_message); + return status; + } + + *message = new_message; + + return SUCCESS; +} + +/** + * @brief implements function process_configuration of protected_ike_sa_t + */ +static status_t initialize_connection(protected_ike_sa_t *this, char *name) +{ + /* work is done in state object of type INITIATOR_INIT */ + initiator_init_t *current_state; + status_t status; + state_t *new_state; + + if (this->current_state->get_state(this->current_state) != INITIATOR_INIT) + { + return FAILED; + } + + current_state = (initiator_init_t *) this->current_state; + + status = current_state->initiate_connection(current_state,name,&new_state); + + if (status == SUCCESS) + { + this->current_state = new_state; + } + else + { + this->create_delete_job(this); + } + return status; +} + +/** + * @brief implements function protected_ike_sa_t.get_id + */ +static ike_sa_id_t* get_id(protected_ike_sa_t *this) +{ + return this->ike_sa_id; +} + +static status_t compute_secrets (protected_ike_sa_t *this,chunk_t dh_shared_secret,chunk_t initiator_nonce, chunk_t responder_nonce) +{ + chunk_t concatenated_nonces; + chunk_t skeyseed; + chunk_t prf_plus_seed; + status_t status; + u_int64_t initiator_spi; + u_int64_t responder_spi; + prf_plus_t *prf_plus; + chunk_t secrets_raw; + + /* + * TODO check length for specific prf's + */ + concatenated_nonces.len = (initiator_nonce.len + responder_nonce.len); + concatenated_nonces.ptr = allocator_alloc(concatenated_nonces.len); + if (concatenated_nonces.ptr == NULL) + { + this->logger->log(this->logger, ERROR, "Fatal errror: Could not allocate memory for concatenated nonces"); + return FAILED; + } + /* first is initiator */ + memcpy(concatenated_nonces.ptr,initiator_nonce.ptr,initiator_nonce.len); + /* second is responder */ + memcpy(concatenated_nonces.ptr + initiator_nonce.len,responder_nonce.ptr,responder_nonce.len); + + this->logger->log_chunk(this->logger, RAW, "Nonce data", &concatenated_nonces); + + + /* status of set_key is not checked */ + status = this->prf->set_key(this->prf,concatenated_nonces); + + status = this->prf->allocate_bytes(this->prf,dh_shared_secret,&skeyseed); + if (status != SUCCESS) + { + allocator_free_chunk(concatenated_nonces); + this->logger->log(this->logger, ERROR, "Fatal errror: Could not allocate bytes for skeyseed"); + return status; + } + allocator_free_chunk(concatenated_nonces); + + prf_plus_seed.len = (initiator_nonce.len + responder_nonce.len + 16); + prf_plus_seed.ptr = allocator_alloc(prf_plus_seed.len); + if (prf_plus_seed.ptr == NULL) + { + this->logger->log(this->logger, ERROR, "Fatal errror: Could not allocate memory for prf+ seed"); + allocator_free_chunk(skeyseed); + return FAILED; + } + + /* first is initiator */ + memcpy(prf_plus_seed.ptr,initiator_nonce.ptr,initiator_nonce.len); + /* second is responder */ + memcpy(prf_plus_seed.ptr + initiator_nonce.len,responder_nonce.ptr,responder_nonce.len); + /* third is initiator spi */ + initiator_spi = this->ike_sa_id->get_initiator_spi(this->ike_sa_id); + memcpy(prf_plus_seed.ptr + initiator_nonce.len + responder_nonce.len,&initiator_spi,8); + /* fourth is responder spi */ + responder_spi = this->ike_sa_id->get_responder_spi(this->ike_sa_id); + memcpy(prf_plus_seed.ptr + initiator_nonce.len + responder_nonce.len + 8,&responder_spi,8); + + this->logger->log_chunk(this->logger, PRIVATE, "Keyseed", &skeyseed); + this->logger->log_chunk(this->logger, PRIVATE, "PRF+ Seed", &prf_plus_seed); + + this->logger->log(this->logger, CONTROL | MOST, "Set new key of prf object"); + status = this->prf->set_key(this->prf,skeyseed); + allocator_free_chunk(skeyseed); + if (status != SUCCESS) + { + this->logger->log(this->logger, ERROR, "Fatal errror: Could not allocate memory for prf+ seed"); + allocator_free_chunk(prf_plus_seed); + return FAILED; + } + + this->logger->log(this->logger, CONTROL | MOST, "Create new prf+ object"); + prf_plus = prf_plus_create(this->prf, prf_plus_seed); + allocator_free_chunk(prf_plus_seed); + if (prf_plus == NULL) + { + this->logger->log(this->logger, ERROR, "Fatal errror: prf+ object could not be created"); + return FAILED; + } + + prf_plus->allocate_bytes(prf_plus,100,&secrets_raw); + + this->logger->log_chunk(this->logger, PRIVATE, "Secrets", &secrets_raw); + + allocator_free_chunk(secrets_raw); + + prf_plus->destroy(prf_plus); + + return SUCCESS; +} + +/** + * @brief implements function resend_last_reply of protected_ike_sa_t + */ +status_t resend_last_reply (protected_ike_sa_t *this) +{ + packet_t *packet; + status_t status; + + status = this->last_responded_message->generate(this->last_responded_message, &packet); + if (status != SUCCESS) + { + this->logger->log(this->logger, ERROR, "Could not generate message to resent"); + return status; + } + + status = global_send_queue->add(global_send_queue, packet); + if (status != SUCCESS) + { + this->logger->log(this->logger, ERROR, "Could not add packet to send queue"); + packet->destroy(packet); + return status; + } + return SUCCESS; +} + +status_t create_delete_job (protected_ike_sa_t *this) +{ + job_t *delete_job; + status_t status; + + this->logger->log(this->logger, CONTROL | MORE, "Going to create job to delete this IKE_SA"); + + delete_job = (job_t *) delete_ike_sa_job_create(this->ike_sa_id); + if (delete_job == NULL) + { + this->logger->log(this->logger, ERROR, "Job to delete IKE SA could not be created"); + return FAILED; + } + + status = global_job_queue->add(global_job_queue,delete_job); + if (status != SUCCESS) + { + this->logger->log(this->logger, ERROR, "%s Job to delete IKE SA could not be added to job queue",mapping_find(status_m,status)); + delete_job->destroy_all(delete_job); + return status; + } + return SUCCESS; +} + +/** + * @brief implements function destroy of protected_ike_sa_t + */ +static status_t destroy (protected_ike_sa_t *this) +{ + + this->logger->log(this->logger, CONTROL | MORE, "Going to destroy IKE_SA"); + + /* destroy child sa's */ + this->logger->log(this->logger, CONTROL | MOST, "Destroy all child_sa's"); + while (this->child_sas->get_count(this->child_sas) > 0) + { + void *child_sa; + if (this->child_sas->remove_first(this->child_sas,&child_sa) != SUCCESS) + { + break; + } + /* destroy child sa */ + } + this->child_sas->destroy(this->child_sas); + + this->logger->log(this->logger, CONTROL | MOST, "Destroy secrets"); + if (this->secrets.d_key.ptr != NULL) + { + allocator_free(this->secrets.d_key.ptr); + } + if (this->secrets.ai_key.ptr != NULL) + { + allocator_free(this->secrets.ai_key.ptr); + } + if (this->secrets.ar_key.ptr != NULL) + { + allocator_free(this->secrets.ar_key.ptr); + } + if (this->secrets.ei_key.ptr != NULL) + { + allocator_free(this->secrets.ei_key.ptr); + } + if (this->secrets.er_key.ptr != NULL) + { + allocator_free(this->secrets.er_key.ptr); + } + if (this->secrets.pi_key.ptr != NULL) + { + allocator_free(this->secrets.pi_key.ptr); + } + if (this->secrets.pr_key.ptr != NULL) + { + allocator_free(this->secrets.pr_key.ptr); + } + + if ( this->crypter_initiator != NULL) + { + this->logger->log(this->logger, CONTROL | MOST, "Destroy initiator crypter"); + this->crypter_initiator->destroy(this->crypter_initiator); + } + + if ( this->crypter_responder != NULL) + { + this->logger->log(this->logger, CONTROL | MOST, "Destroy responder crypter"); + this->crypter_responder->destroy(this->crypter_responder); + } + + if ( this->signer_initiator != NULL) + { + this->logger->log(this->logger, CONTROL | MOST, "Destroy initiator signer"); + this->signer_initiator->destroy(this->signer_initiator); + } + + if (this->signer_responder != NULL) + { + this->logger->log(this->logger, CONTROL | MOST, "Destroy responder signer"); + this->signer_responder->destroy(this->signer_responder); + } + + if (this->prf != NULL) + { + this->logger->log(this->logger, CONTROL | MOST, "Destroy prf"); + this->prf->destroy(this->prf); + } + + /* destroy ike_sa_id */ + this->logger->log(this->logger, CONTROL | MOST, "Destroy assigned ike_sa_id"); + this->ike_sa_id->destroy(this->ike_sa_id); + + /* destroy stored requested message */ + if (this->last_requested_message != NULL) + { + this->logger->log(this->logger, CONTROL | MOST, "Destroy last requested message"); + this->last_requested_message->destroy(this->last_requested_message); + } + + /* destroy stored responded messages */ + if (this->last_responded_message != NULL) + { + this->logger->log(this->logger, CONTROL | MOST, "Destroy last responded message"); + this->last_responded_message->destroy(this->last_responded_message); + } + + this->logger->log(this->logger, CONTROL | MOST, "Destroy randomizer"); + this->randomizer->destroy(this->randomizer); + + if (this->me.host != NULL) + { + this->logger->log(this->logger, CONTROL | MOST, "Destroy host informations of me"); + this->me.host->destroy(this->me.host); + } + + if (this->other.host != NULL) + { + this->logger->log(this->logger, CONTROL | MOST, "Destroy host informations of other"); + this->other.host->destroy(this->other.host); + } + + this->logger->log(this->logger, CONTROL | MOST, "Destroy current state object"); + this->current_state->destroy(this->current_state); + + this->logger->log(this->logger, CONTROL | MOST, "Destroy logger of IKE_SA"); + + global_logger_manager->destroy_logger(global_logger_manager, this->logger); + + allocator_free(this); + return SUCCESS; +} + +/* + * Described in Header + */ +ike_sa_t * ike_sa_create(ike_sa_id_t *ike_sa_id) +{ + protected_ike_sa_t *this = allocator_alloc_thing(protected_ike_sa_t); + if (this == NULL) + { + return NULL; + } + + /* Public functions */ + this->public.process_message = (status_t(*)(ike_sa_t*, message_t*)) process_message; + this->public.initialize_connection = (status_t(*)(ike_sa_t*, char*)) initialize_connection; + this->public.get_id = (ike_sa_id_t*(*)(ike_sa_t*)) get_id; + this->public.destroy = (status_t(*)(ike_sa_t*))destroy; + + /* private functions */ + this->build_message = build_message; + this->resend_last_reply = resend_last_reply; + this->create_delete_job = create_delete_job; + this->compute_secrets = compute_secrets; + + /* initialize private fields */ + this->logger = global_logger_manager->create_logger(global_logger_manager, IKE_SA, NULL); + if (this->logger == NULL) + { + allocator_free(this); + } + + if (ike_sa_id->clone(ike_sa_id,&(this->ike_sa_id)) != SUCCESS) + { + this->logger->log(this->logger, ERROR, "Fatal error: Could not clone ike_sa_id"); + global_logger_manager->destroy_logger(global_logger_manager,this->logger); + allocator_free(this); + return NULL; + } + this->child_sas = linked_list_create(); + if (this->child_sas == NULL) + { + this->logger->log(this->logger, ERROR, "Fatal error: Could not create list for child_sa's"); + this->ike_sa_id->destroy(this->ike_sa_id); + global_logger_manager->destroy_logger(global_logger_manager,this->logger); + allocator_free(this); + return NULL; + } + this->randomizer = randomizer_create(); + if (this->randomizer == NULL) + { + this->logger->log(this->logger, ERROR, "Fatal error: Could not create list for child_sa's"); + this->child_sas->destroy(this->child_sas); + this->ike_sa_id->destroy(this->ike_sa_id); + global_logger_manager->destroy_logger(global_logger_manager,this->logger); + allocator_free(this); + } + + this->me.host = NULL; + this->other.host = NULL; + this->last_requested_message = NULL; + this->last_responded_message = NULL; + this->message_id_out = 0; + this->message_id_in = 0; + this->secrets.d_key.ptr = NULL; + this->secrets.d_key.len = 0; + this->secrets.ai_key.ptr = NULL; + this->secrets.ai_key.len = 0; + this->secrets.ar_key.ptr = NULL; + this->secrets.ar_key.len = 0; + this->secrets.ei_key.ptr = NULL; + this->secrets.ei_key.len = 0; + this->secrets.er_key.ptr = NULL; + this->secrets.er_key.len = 0; + this->secrets.pi_key.ptr = NULL; + this->secrets.pi_key.len = 0; + this->secrets.pr_key.ptr = NULL; + this->secrets.pr_key.len = 0; + this->crypter_initiator = NULL; + this->crypter_responder = NULL; + this->signer_initiator = NULL; + this->signer_responder = NULL; + this->prf = NULL; + + + + + /* at creation time, IKE_SA is in a initiator state */ + if (ike_sa_id->is_initiator(ike_sa_id)) + { + this->current_state = (state_t *) initiator_init_create(this); + } + else + { + this->current_state = (state_t *) responder_init_create(this); + } + + if (this->current_state == NULL) + { + this->logger->log(this->logger, ERROR, "Fatal error: Could not create state object"); + this->child_sas->destroy(this->child_sas); + this->ike_sa_id->destroy(this->ike_sa_id); + global_logger_manager->destroy_logger(global_logger_manager,this->logger); + this->randomizer->destroy(this->randomizer); + allocator_free(this); + } + + + return &(this->public); +} |