strongswan/src/charon/network/sender.c

150 lines
3.5 KiB
C
Raw Normal View History

2005-11-07 12:07:56 +00:00
/**
* @file sender.c
*
* @brief Implementation of sender_t.
*
2005-11-07 12:07:56 +00:00
*/
/*
2006-07-07 08:49:06 +00:00
* Copyright (C) 2005-2006 Martin Willi
* Copyright (C) 2005 Jan Hutter
2005-11-07 12:07:56 +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.
*/
2005-11-07 12:07:56 +00:00
#include <stdlib.h>
#include <pthread.h>
2005-11-07 12:07:56 +00:00
#include "sender.h"
2005-11-29 10:25:07 +00:00
#include <daemon.h>
#include <network/socket.h>
2005-12-06 16:04:39 +00:00
2005-11-24 11:30:19 +00:00
typedef struct private_sender_t private_sender_t;
2005-11-07 12:07:56 +00:00
/**
2005-12-06 16:04:39 +00:00
* Private data of a sender_t object.
2005-11-07 12:07:56 +00:00
*/
2005-11-24 11:30:19 +00:00
struct private_sender_t {
2005-11-07 12:07:56 +00:00
/**
2005-12-06 16:04:39 +00:00
* Public part of a sender_t object.
2005-11-07 12:07:56 +00:00
*/
sender_t public;
2005-11-07 12:07:56 +00:00
/**
2005-12-06 16:04:39 +00:00
* Assigned thread.
2005-11-07 12:07:56 +00:00
*/
pthread_t assigned_thread;
/**
* The packets are stored in a linked list
*/
linked_list_t *list;
/**
* mutex to synchronize access to list
*/
pthread_mutex_t mutex;
2005-11-07 12:07:56 +00:00
/**
* condvar to signal for packets in list
*/
pthread_cond_t condvar;
2005-11-07 12:07:56 +00:00
};
/**
* implements sender_t.send
*/
static void send_(private_sender_t *this, packet_t *packet)
{
host_t *src, *dst;
src = packet->get_source(packet);
dst = packet->get_destination(packet);
DBG1(DBG_NET, "sending packet: from %#H to %#H", src, dst);
pthread_mutex_lock(&this->mutex);
this->list->insert_last(this->list, packet);
pthread_mutex_unlock(&this->mutex);
pthread_cond_signal(&this->condvar);
}
2005-11-07 12:07:56 +00:00
/**
2005-12-06 16:04:39 +00:00
* Implementation of private_sender_t.send_packets.
2005-11-07 12:07:56 +00:00
*/
static void send_packets(private_sender_t * this)
2005-11-07 12:07:56 +00:00
{
2005-11-17 11:20:34 +00:00
2005-11-07 12:07:56 +00:00
/* cancellation disabled by default */
pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, NULL);
DBG1(DBG_NET, "sender thread running, thread_ID: %06u", (int)pthread_self());
2005-11-26 15:44:40 +00:00
while (TRUE)
2005-11-07 12:07:56 +00:00
{
packet_t *packet;
int oldstate;
pthread_mutex_lock(&this->mutex);
/* go to wait while no packets available */
while (this->list->get_count(this->list) == 0)
2005-11-07 12:07:56 +00:00
{
/* add cleanup handler, wait for packet, remove cleanup handler */
pthread_cleanup_push((void(*)(void*))pthread_mutex_unlock, (void*)&this->mutex);
pthread_setcancelstate(PTHREAD_CANCEL_ENABLE, &oldstate);
pthread_cond_wait(&this->condvar, &this->mutex);
pthread_setcancelstate(oldstate, NULL);
pthread_cleanup_pop(0);
2005-11-07 12:07:56 +00:00
}
this->list->remove_first(this->list, (void**)&packet);
pthread_mutex_unlock(&this->mutex);
charon->socket->send(charon->socket, packet);
packet->destroy(packet);
2005-11-07 12:07:56 +00:00
}
}
/**
2005-12-06 16:04:39 +00:00
* Implementation of sender_t.destroy.
2005-11-07 12:07:56 +00:00
*/
2005-11-28 20:29:47 +00:00
static void destroy(private_sender_t *this)
2005-11-07 12:07:56 +00:00
{
pthread_cancel(this->assigned_thread);
pthread_join(this->assigned_thread, NULL);
this->list->destroy_offset(this->list, offsetof(packet_t, destroy));
free(this);
2005-11-07 12:07:56 +00:00
}
/*
2005-12-06 16:04:39 +00:00
* Described in header.
*/
2005-11-07 12:07:56 +00:00
sender_t * sender_create()
{
private_sender_t *this = malloc_thing(private_sender_t);
this->public.send = (void(*)(sender_t*,packet_t*))send_;
2005-11-28 20:29:47 +00:00
this->public.destroy = (void(*)(sender_t*)) destroy;
this->list = linked_list_create();
pthread_mutex_init(&this->mutex, NULL);
pthread_cond_init(&this->condvar, NULL);
if (pthread_create(&this->assigned_thread, NULL,
(void*)send_packets, this) != 0)
2005-11-07 12:07:56 +00:00
{
charon->kill(charon, "unable to create sender thread");
2005-11-07 12:07:56 +00:00
}
2005-11-07 12:07:56 +00:00
return &(this->public);
}