471 lines
14 KiB
C
471 lines
14 KiB
C
/*
|
|
* This file is part of the Sofia-SIP package
|
|
*
|
|
* Copyright (C) 2005 Nokia Corporation.
|
|
*
|
|
* Contact: Pekka Pessi <pekka.pessi@nokia.com>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public License
|
|
* as published by the Free Software Foundation; either version 2.1 of
|
|
* the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
|
* 02110-1301 USA
|
|
*
|
|
*/
|
|
|
|
/**@ingroup su_wait
|
|
* @CFILE su_port.c
|
|
*
|
|
* OS-Independent Socket Syncronization Interface.
|
|
*
|
|
* This looks like nth reincarnation of "reactor". It implements the
|
|
* poll/select/WaitForMultipleObjects and message passing functionality.
|
|
* This is virtual implementation:
|
|
*
|
|
* @author Pekka Pessi <Pekka.Pessi@nokia.com>
|
|
* @author Kai Vehmanen <kai.vehmanen@nokia.com>
|
|
*
|
|
* @date Created: Tue Sep 14 15:51:04 1999 ppessi
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#define SU_CLONE_T su_msg_t
|
|
|
|
#define su_port_s su_virtual_port_s
|
|
|
|
#include "su_port.h"
|
|
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
|
|
/** Create the default su_port_t implementation. */
|
|
su_port_t *su_default_port_create(void)
|
|
{
|
|
#if HAVE_EPOLL
|
|
return su_epoll_port_create();
|
|
#elif HAVE_KQUEUE
|
|
return su_kqueue_port_create();
|
|
#elif HAVE_SYS_DEVPOLL_H
|
|
return su_devpoll_port_create();
|
|
#elif HAVE_POLL_PORT
|
|
return su_poll_port_create();
|
|
#elif HAVE_WIN32
|
|
return su_wsaevent_port_create();
|
|
#elif HAVE_SELECT
|
|
return su_select_port_create();
|
|
#else
|
|
return NULL;
|
|
#endif
|
|
}
|
|
|
|
int su_default_clone_start(su_root_t *parent,
|
|
su_clone_r return_clone,
|
|
su_root_magic_t *magic,
|
|
su_root_init_f init,
|
|
su_root_deinit_f deinit)
|
|
{
|
|
#if HAVE_EPOLL
|
|
return su_epoll_clone_start(parent, return_clone, magic, init, deinit);
|
|
#elif HAVE_KQUEUE
|
|
return su_kqueue_clone_start(parent, return_clone, magic, init, deinit);
|
|
#elif HAVE_SYS_DEVPOLL_H
|
|
return su_devpoll_clone_start(parent, return_clone, magic, init, deinit);
|
|
#elif HAVE_POLL_PORT
|
|
return su_poll_clone_start(parent, return_clone, magic, init, deinit);
|
|
#elif HAVE_WIN32
|
|
return su_wsaevent_clone_start(parent, return_clone, magic, init, deinit);
|
|
#elif HAVE_SELECT
|
|
return su_select_clone_start(parent, return_clone, magic, init, deinit);
|
|
#else
|
|
errno = ENOSYS;
|
|
return -1;
|
|
#endif
|
|
}
|
|
|
|
static su_port_create_f *preferred_su_port_create;
|
|
static su_clone_start_f *preferred_su_clone_start;
|
|
|
|
/** Explicitly set the preferred su_port_t implementation.
|
|
*
|
|
* @sa su_epoll_port_create(), su_poll_port_create(), su_select_port_create()
|
|
*/
|
|
void su_port_prefer(su_port_create_f *create,
|
|
su_clone_start_f *start)
|
|
{
|
|
if (create) preferred_su_port_create = create;
|
|
if (start) preferred_su_clone_start = start;
|
|
}
|
|
|
|
static
|
|
void su_port_set_system_preferences(char const *name)
|
|
{
|
|
su_port_create_f *create = preferred_su_port_create;
|
|
su_clone_start_f *start = preferred_su_clone_start;
|
|
|
|
if (name == NULL)
|
|
;
|
|
#if HAVE_EPOLL
|
|
else if (strcmp(name, "epoll") == 0) {
|
|
create = su_epoll_port_create;
|
|
start = su_epoll_clone_start;
|
|
}
|
|
#endif
|
|
#if HAVE_KQUEUE
|
|
else if (strcmp(name, "kqueue") == 0) {
|
|
create = su_kqueue_port_create;
|
|
start = su_kqueue_clone_start;
|
|
}
|
|
#endif
|
|
#if HAVE_SYS_DEVPOLL_H
|
|
else if (strcmp(name, "devpoll") == 0) {
|
|
create = su_devpoll_port_create;
|
|
start = su_devpoll_clone_start;
|
|
}
|
|
#endif
|
|
#if HAVE_POLL_PORT
|
|
else if (strcmp(name, "poll") == 0) {
|
|
create = su_poll_port_create;
|
|
start = su_poll_clone_start;
|
|
}
|
|
#endif
|
|
#if HAVE_WIN32
|
|
else if (strcasecmp(name, "wsaevent") == 0) {
|
|
create = su_wsaevent_port_create;
|
|
start = su_wsaevent_clone_start;
|
|
}
|
|
#elif HAVE_SELECT
|
|
else if (strcmp(name, "select") == 0) {
|
|
create = su_select_port_create;
|
|
start = su_select_clone_start;
|
|
}
|
|
#endif
|
|
|
|
if (create == NULL)
|
|
create = su_default_port_create;
|
|
|
|
if (!preferred_su_port_create ||
|
|
preferred_su_port_create == su_default_port_create)
|
|
preferred_su_port_create = create;
|
|
|
|
if (start == NULL)
|
|
start = su_default_clone_start;
|
|
|
|
if (!preferred_su_clone_start ||
|
|
preferred_su_clone_start == su_default_clone_start)
|
|
preferred_su_clone_start = start;
|
|
}
|
|
|
|
/** Create the preferred su_port_t implementation. */
|
|
su_port_t *su_port_create(void)
|
|
{
|
|
if (preferred_su_port_create == NULL)
|
|
su_port_set_system_preferences(getenv("SU_PORT"));
|
|
|
|
return preferred_su_port_create();
|
|
}
|
|
|
|
/** Return name of the su_port_t instance. */
|
|
char const *su_port_name(su_port_t const *port)
|
|
{
|
|
return port->sup_vtable->su_port_name(port);
|
|
}
|
|
|
|
/* ========================================================================
|
|
* su_clone_t
|
|
*/
|
|
|
|
/**@ingroup su_wait
|
|
*
|
|
* @page su_clone_t Clone Objects
|
|
*
|
|
* The process may be divided into many tasks via cloning. Several tasks may
|
|
* run in context of one thread, or each task may be run by its own thread.
|
|
* However, only a single thread can execute code within a task. There can
|
|
* be a 1-to-N mapping from thread to tasks. Thus, software using tasks can
|
|
* be executed by multiple threads in a multithreaded environment and by a
|
|
* single thread in a singlethreaded environment.
|
|
*
|
|
* The clones are useful for handling tasks that can be executed by a
|
|
* separate threads, but which do not block excessively. When threads are
|
|
* not available or they are not needed, clones can also be run in a
|
|
* single-threaded mode. Running in single-threaded mode is especially
|
|
* useful while debugging.
|
|
*
|
|
* A clone task is created with function su_clone_start(). Each clone has
|
|
* its own root object (su_root_t), which holds a context pointer
|
|
* (su_root_magic_t *). The context object can be different from that of
|
|
* parent task.
|
|
*
|
|
* When a clone is started, the clone initialization function is called. The
|
|
* initialization function should do whatever initialization there is to be
|
|
* performed, register I/O events and timers, and then return. If the
|
|
* initialization is successful, the clone task reverts to run the event
|
|
* loop and invoking the event callbacks until its parent stops it by
|
|
* calling su_clone_wait() which invokes the deinit function. The clone task
|
|
* is destroyed when the deinit function returns.
|
|
*
|
|
* The public API consists of following functions:
|
|
* - su_clone_start()
|
|
* - su_clone_task()
|
|
* - su_clone_wait()
|
|
* - su_clone_forget()
|
|
*
|
|
* @note
|
|
* There is only one event loop for each thread which can be shared by
|
|
* multiple clone tasks. Therefore, the clone tasks can not explicitly run
|
|
* or step the event loop, but they are limited to event callbacks. A clone
|
|
* task may not call su_root_break(), su_root_run() or su_root_step().
|
|
*/
|
|
|
|
static int su_root_init_nothing(su_root_t *root, su_root_magic_t *magic)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static void su_root_deinit_nothing(su_root_t *root, su_root_magic_t *magic)
|
|
{
|
|
}
|
|
|
|
/** Start a clone task.
|
|
*
|
|
* Allocate and initialize a sub-task. Depending on the su_root_threading()
|
|
* settings, a separate thread may be created to execute the sub-task. The
|
|
* sub-task is represented by clone handle to the rest of the application.
|
|
* The function su_clone_start() returns the clone handle in @a
|
|
* return_clone. The clone handle is used to communicate with the newly
|
|
* created clone task using messages.
|
|
*
|
|
* A new #su_root_t object is created for the sub-task with the @a magic as
|
|
* the root context pointer. Because the sub-task may or may not have its
|
|
* own thread, all its activity must be scheduled via this root object. In
|
|
* other words, the sub-task can be schedule
|
|
* -# I/O events with su_root_register()
|
|
* -# timers with su_timer_set(), su_timer_set_at() or su_timer_run()
|
|
* -# messages with su_msg_send().
|
|
*
|
|
* Messages can also be used to pass information between tasks or threads.
|
|
*
|
|
* In multi-threaded implementation, su_clone_start() launches a new thread,
|
|
* and the initialization routine is executed by this newly created thread.
|
|
* The calling thread blocks until the initialization routine completes. If
|
|
* the initialization routine returns #su_success (0), the sub-task is
|
|
* considered to be created successfully. After the successful
|
|
* initialization, the sub-task continues to execeute the function
|
|
* su_root_run().
|
|
*
|
|
* In single-threaded implementations, just a new root object is created.
|
|
* The initialization routine is called directly from su_clone_start().
|
|
*
|
|
* If the initalization function @a init fails, the sub-task (either the
|
|
* newly created thread or the current thread executing the su_clone_start()
|
|
* function) calls the deinitialization function, and su_clone_start()
|
|
* returns NULL.
|
|
*
|
|
* @param parent root to be cloned
|
|
* @param return_clone reference to a clone [OUT]
|
|
* @param magic pointer to user data
|
|
* @param init initialization function
|
|
* @param deinit deinitialization function
|
|
*
|
|
* @return 0 if successfull, -1 upon an error.
|
|
*
|
|
* @note Earlier documentation mentioned that @a parent could be NULL. That
|
|
* feature has never been implemented, however.
|
|
*
|
|
* @sa su_root_threading(), su_clone_task(), su_clone_stop(), su_clone_wait(),
|
|
* su_clone_forget().
|
|
*/
|
|
int su_clone_start(su_root_t *parent,
|
|
su_clone_r return_clone,
|
|
su_root_magic_t *magic,
|
|
su_root_init_f init,
|
|
su_root_deinit_f deinit)
|
|
{
|
|
su_port_vtable_t const *svp;
|
|
|
|
if (init == NULL)
|
|
init = su_root_init_nothing;
|
|
if (deinit == NULL)
|
|
deinit = su_root_deinit_nothing;
|
|
|
|
|
|
if (parent == NULL || parent->sur_threading) {
|
|
if (preferred_su_clone_start == NULL)
|
|
su_port_set_system_preferences(getenv("SU_PORT"));
|
|
return preferred_su_clone_start(parent, return_clone, magic, init, deinit);
|
|
}
|
|
|
|
svp = parent->sur_task->sut_port->sup_vtable;
|
|
|
|
if (svp->su_port_start_shared == NULL)
|
|
return su_seterrno(EINVAL);
|
|
|
|
/* Return a task sharing the same port. */
|
|
return svp->su_port_start_shared(parent, return_clone, magic, init, deinit);
|
|
}
|
|
|
|
/** Get reference to a clone task.
|
|
*
|
|
* @param clone Clone pointer
|
|
*
|
|
* @return A reference to the task structure of the clone.
|
|
*/
|
|
_su_task_r su_clone_task(su_clone_r clone)
|
|
{
|
|
return su_msg_to(clone);
|
|
}
|
|
|
|
/**Forget the clone.
|
|
*
|
|
* Normally, the clone task executes until it is stopped. If the parent
|
|
* task does not need to stop the task, it can "forget" the clone. The
|
|
* clone exits independently of the parent task.
|
|
*
|
|
* @param rclone Reference to the clone.
|
|
*/
|
|
void su_clone_forget(su_clone_r rclone)
|
|
{
|
|
su_msg_destroy(rclone);
|
|
}
|
|
|
|
/** Stop the clone.
|
|
*
|
|
* This can used only if clone task has sent no report messages (messages
|
|
* with delivery report sent back to clone).
|
|
*
|
|
* @deprecated. Use su_clone_wait().
|
|
*/
|
|
void su_clone_stop(su_clone_r rclone)
|
|
{
|
|
su_msg_send(rclone);
|
|
}
|
|
|
|
/** Stop a clone and wait until it is has completed.
|
|
*
|
|
* The function su_clone_wait() is used to stop the clone task and wait
|
|
* until it has cleaned up. The clone task is destroyed asynchronously. The
|
|
* parent sends a message to clone, clone deinitializes itself and then
|
|
* replies. After the reply message is received by the parent, it will send
|
|
* a third message back to clone.
|
|
*
|
|
* The parent destroy all messages to or from clone task before calling
|
|
* su_clone_wait(). The parent task may not send any messages to the clone
|
|
* after calling su_clone_wait(). The su_clone_wait() function blocks until
|
|
* the cloned task is destroyed. During that time, the parent task must be
|
|
* prepared to process all the messages sent by clone task. This includes
|
|
* all the messages sent by clone before destroy the message reached the
|
|
* clone.
|
|
*/
|
|
void su_clone_wait(su_root_t *root, su_clone_r rclone)
|
|
{
|
|
if (rclone[0]) {
|
|
assert(root == NULL || root == su_msg_from(rclone)->sut_root);
|
|
su_port_wait(rclone);
|
|
}
|
|
}
|
|
|
|
/** Pause a clone.
|
|
*
|
|
* Obtain an exclusive lock on clone's private data.
|
|
*
|
|
* @retval 0 if successful (and clone is paused)
|
|
* @retval -1 upon an error
|
|
*
|
|
* @deprecated Never implemented.
|
|
*/
|
|
int su_clone_pause(su_clone_r rclone)
|
|
{
|
|
#if 0
|
|
su_root_t *cloneroot = su_task_root(su_msg_to(rclone));
|
|
|
|
if (!cloneroot)
|
|
return (errno = EFAULT), -1;
|
|
|
|
if (SU_ROOT_OWN_THREAD(cloneroot))
|
|
/* We own it already */
|
|
return 0;
|
|
|
|
return su_port_pause(cloneroot->sur_port);
|
|
#else
|
|
return errno = ENOSYS, -1;
|
|
#endif
|
|
}
|
|
|
|
/** Resume a clone.
|
|
*
|
|
* Give up an exclusive lock on clone's private data.
|
|
*
|
|
* @retval 0 if successful (and clone is resumed)
|
|
* @retval -1 upon an error
|
|
*
|
|
* @deprecated Never implemented.
|
|
*/
|
|
int su_clone_resume(su_clone_r rclone)
|
|
{
|
|
#if 0
|
|
su_root_t *cloneroot = su_task_root(su_msg_to(rclone));
|
|
|
|
if (!cloneroot)
|
|
return (errno = EFAULT), -1;
|
|
|
|
if (SU_ROOT_OWN_THREAD(cloneroot))
|
|
/* We cannot give it away */
|
|
return 0;
|
|
|
|
return su_port_resume(cloneroot->sur_port);
|
|
#else
|
|
return errno = ENOSYS, -1;
|
|
#endif
|
|
}
|
|
|
|
/** Wait for clone to exit.
|
|
*
|
|
* @internal
|
|
*
|
|
* Called by su_clone_wait().
|
|
*/
|
|
void su_port_wait(su_clone_r rclone)
|
|
{
|
|
su_port_t *cloneport;
|
|
|
|
assert(*rclone);
|
|
|
|
cloneport = su_msg_to(rclone)->sut_port;
|
|
cloneport->sup_vtable->su_port_wait(rclone);
|
|
}
|
|
|
|
int su_port_execute(su_task_r const task,
|
|
int (*function)(void *), void *arg,
|
|
int *return_value)
|
|
{
|
|
if (!task->sut_port->sup_vtable->su_port_execute)
|
|
return errno = ENOSYS, -1;
|
|
|
|
return task->sut_port->sup_vtable->
|
|
su_port_execute(task, function, arg, return_value);
|
|
}
|
|
|
|
#if notyet && nomore
|
|
int su_port_pause(su_port_t *self)
|
|
{
|
|
assert(self->sup_vtable->su_port_pause);
|
|
return self->sup_vtable->su_port_pause(self);
|
|
}
|
|
|
|
int su_port_resume(su_port_t *self)
|
|
{
|
|
assert(self->sup_vtable->su_port_resume);
|
|
return self->sup_vtable->su_port_resume(self);
|
|
}
|
|
#endif
|