2006-12-21 06:30:28 +00:00
|
|
|
/*
|
|
|
|
* 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
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @file su_source.c
|
|
|
|
* @brief Wrapper for glib GSource.
|
|
|
|
* *
|
|
|
|
* @author Pekka Pessi <Pekka.Pessi@nokia.com>.
|
|
|
|
*
|
|
|
|
* @date Created: Thu Mar 4 15:15:15 2004 ppessi
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#if HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <limits.h>
|
|
|
|
|
|
|
|
#include <glib.h>
|
|
|
|
|
|
|
|
#define SU_PORT_IMPLEMENTATION 1
|
|
|
|
|
|
|
|
#define SU_MSG_ARG_T union { char anoymous[4]; }
|
|
|
|
|
|
|
|
#define su_port_s su_source_s
|
|
|
|
|
|
|
|
#include "sofia-sip/su_source.h"
|
|
|
|
#include "sofia-sip/su_glib.h"
|
|
|
|
|
|
|
|
#include "sofia-sip/su.h"
|
|
|
|
#include "su_port.h"
|
|
|
|
#include "sofia-sip/su_alloc.h"
|
|
|
|
|
|
|
|
static su_port_t *su_source_create(void) __attribute__((__malloc__));
|
|
|
|
static gboolean su_source_prepare(GSource *gs, gint *return_tout);
|
|
|
|
static gboolean su_source_check(GSource *gs);
|
|
|
|
static gboolean su_source_dispatch(GSource *gs,
|
|
|
|
GSourceFunc callback,
|
|
|
|
gpointer user_data);
|
|
|
|
static void su_source_finalize(GSource *source);
|
|
|
|
|
|
|
|
static int su_source_getmsgs(su_port_t *self);
|
|
|
|
|
|
|
|
static
|
|
|
|
GSourceFuncs su_source_funcs[1] = {{
|
|
|
|
su_source_prepare,
|
|
|
|
su_source_check,
|
|
|
|
su_source_dispatch,
|
|
|
|
su_source_finalize,
|
|
|
|
NULL,
|
|
|
|
NULL
|
|
|
|
}};
|
|
|
|
|
|
|
|
static void su_source_lock(su_port_t *self, char const *who);
|
|
|
|
static void su_source_unlock(su_port_t *self, char const *who);
|
|
|
|
static void su_source_incref(su_port_t *self, char const *who);
|
|
|
|
static void su_source_decref(su_port_t *self, int blocking, char const *who);
|
|
|
|
static struct _GSource *su_source_gsource(su_port_t *port);
|
|
|
|
|
|
|
|
static int su_source_send(su_port_t *self, su_msg_r rmsg);
|
|
|
|
|
|
|
|
static int su_source_register(su_port_t *self,
|
|
|
|
su_root_t *root,
|
|
|
|
su_wait_t *wait,
|
|
|
|
su_wakeup_f callback,
|
|
|
|
su_wakeup_arg_t *arg,
|
|
|
|
int priority);
|
|
|
|
static int su_source_unregister(su_port_t *port,
|
|
|
|
su_root_t *root,
|
|
|
|
su_wait_t *wait,
|
|
|
|
su_wakeup_f callback,
|
|
|
|
su_wakeup_arg_t *arg);
|
|
|
|
static int su_source_deregister(su_port_t *self, int i);
|
|
|
|
static int su_source_unregister_all(su_port_t *self,
|
|
|
|
su_root_t *root);
|
|
|
|
static int su_source_eventmask(su_port_t *self,
|
|
|
|
int index, int socket, int events);
|
|
|
|
static void su_source_run(su_port_t *self);
|
|
|
|
static void su_source_break(su_port_t *self);
|
|
|
|
static su_duration_t su_source_step(su_port_t *self, su_duration_t tout);
|
|
|
|
static int su_source_own_thread(su_port_t const *port);
|
|
|
|
static int su_source_add_prepoll(su_port_t *port,
|
|
|
|
su_root_t *root,
|
|
|
|
su_prepoll_f *,
|
|
|
|
su_prepoll_magic_t *);
|
|
|
|
static int su_source_remove_prepoll(su_port_t *port,
|
|
|
|
su_root_t *root);
|
|
|
|
static su_timer_t **su_source_timers(su_port_t *port);
|
|
|
|
static int su_source_multishot(su_port_t *self, int multishot);
|
|
|
|
static int su_source_threadsafe(su_port_t *port);
|
|
|
|
|
|
|
|
static
|
|
|
|
su_port_vtable_t const su_source_vtable[1] =
|
|
|
|
{{
|
|
|
|
/* su_vtable_size: */ sizeof su_source_vtable,
|
|
|
|
su_source_lock,
|
|
|
|
su_source_unlock,
|
|
|
|
su_source_incref,
|
|
|
|
su_source_decref,
|
|
|
|
|
|
|
|
su_source_gsource,
|
|
|
|
|
|
|
|
su_source_send,
|
|
|
|
su_source_register,
|
|
|
|
su_source_unregister,
|
|
|
|
su_source_deregister,
|
|
|
|
su_source_unregister_all,
|
|
|
|
su_source_eventmask,
|
|
|
|
su_source_run,
|
|
|
|
su_source_break,
|
|
|
|
su_source_step,
|
|
|
|
su_source_own_thread,
|
|
|
|
su_source_add_prepoll,
|
|
|
|
su_source_remove_prepoll,
|
|
|
|
su_source_timers,
|
|
|
|
su_source_multishot,
|
|
|
|
su_source_threadsafe
|
|
|
|
|
|
|
|
}};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Port is a per-thread reactor.
|
|
|
|
*
|
|
|
|
* Multiple root objects executed by single thread share a su_port_t object.
|
|
|
|
*/
|
|
|
|
struct su_source_s {
|
|
|
|
su_home_t sup_home[1];
|
|
|
|
su_port_vtable_t const *sup_vtable;
|
|
|
|
|
|
|
|
GThread *sup_tid;
|
|
|
|
GStaticMutex sup_mutex[1];
|
|
|
|
GStaticRWLock sup_ref[1];
|
|
|
|
|
|
|
|
GSource *sup_source;
|
|
|
|
GMainLoop *sup_main_loop;
|
|
|
|
|
|
|
|
/* Message list - this is protected by lock */
|
|
|
|
su_msg_t *sup_head;
|
|
|
|
su_msg_t **sup_tail;
|
|
|
|
|
|
|
|
/* Waits */
|
|
|
|
unsigned sup_registers; /** Counter incremented by
|
|
|
|
su_port_register() or
|
|
|
|
su_port_unregister()
|
|
|
|
*/
|
|
|
|
unsigned sup_n_waits;
|
|
|
|
unsigned sup_size_waits;
|
|
|
|
unsigned sup_max_index;
|
|
|
|
unsigned *sup_indices;
|
|
|
|
su_wait_t *sup_waits;
|
|
|
|
su_wakeup_f *sup_wait_cbs;
|
|
|
|
su_wakeup_arg_t**sup_wait_args;
|
|
|
|
su_root_t **sup_wait_roots;
|
|
|
|
|
|
|
|
/* Timer list */
|
|
|
|
su_timer_t *sup_timers;
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct _SuSource
|
|
|
|
{
|
|
|
|
GSource ss_source[1];
|
|
|
|
su_port_t ss_port[1];
|
|
|
|
} SuSource;
|
|
|
|
|
|
|
|
#define SU_SOURCE_OWN_THREAD(p) ((p)->sup_tid == g_thread_self())
|
|
|
|
|
|
|
|
#if 1
|
|
|
|
#define SU_SOURCE_INCREF(p, f) (g_source_ref(p->sup_source))
|
|
|
|
#define SU_SOURCE_DECREF(p, f) (g_source_unref(p->sup_source))
|
|
|
|
|
|
|
|
#define SU_SOURCE_INITLOCK(p) (g_static_mutex_init((p)->sup_mutex))
|
|
|
|
#define SU_SOURCE_LOCK(p, f) (g_static_mutex_lock((p)->sup_mutex))
|
|
|
|
#define SU_SOURCE_UNLOCK(p, f) (g_static_mutex_unlock((p)->sup_mutex))
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
/* Debugging versions */
|
|
|
|
#define SU_SOURCE_INCREF(p, f) (g_source_ref(p->sup_source), printf("incref(%p) by %s\n", (p), f))
|
|
|
|
#define SU_SOURCE_DECREF(p, f) do { printf("decref(%p) by %s\n", (p), f), \
|
|
|
|
g_source_unref(p->sup_source); } while(0)
|
|
|
|
|
|
|
|
#define SU_SOURCE_INITLOCK(p) \
|
|
|
|
(g_static_mutex_init((p)->sup_mutex), printf("init_lock(%p)\n", p))
|
|
|
|
|
|
|
|
#define SU_SOURCE_LOCK(p, f) \
|
|
|
|
(printf("%ld at %s locking(%p)...", g_thread_self(), f, p), g_static_mutex_lock((p)->sup_mutex), printf(" ...%ld at %s locked(%p)...", g_thread_self(), f, p))
|
|
|
|
|
|
|
|
#define SU_SOURCE_UNLOCK(p, f) \
|
|
|
|
(g_static_mutex_unlock((p)->sup_mutex), printf(" ...%ld at %s unlocked(%p)\n", g_thread_self(), f, p))
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if HAVE_FUNC
|
|
|
|
#define enter (void)SU_DEBUG_9(("%s: entering\n", __func__))
|
|
|
|
#elif HAVE_FUNCTION
|
|
|
|
#define enter (void)SU_DEBUG_9(("%s: entering\n", __FUNCTION__))
|
|
|
|
#else
|
|
|
|
#define enter (void)0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*=============== Public function definitions ===============*/
|
|
|
|
|
|
|
|
/** Create a root that uses GSource as reactor */
|
|
|
|
su_root_t *su_glib_root_create(su_root_magic_t *magic)
|
|
|
|
{
|
|
|
|
return su_root_create_with_port(magic, su_source_create());
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Deprecated */
|
|
|
|
su_root_t *su_root_source_create(su_root_magic_t *magic)
|
|
|
|
{
|
|
|
|
return su_glib_root_create(magic);
|
|
|
|
}
|
|
|
|
|
|
|
|
GSource *su_glib_root_gsource(su_root_t *root)
|
|
|
|
{
|
|
|
|
g_assert(root);
|
|
|
|
return su_root_gsource(root);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*=============== Private function definitions ===============*/
|
|
|
|
|
|
|
|
/**@internal
|
|
|
|
*
|
|
|
|
* Allocates and initializes a reactor and message port object.
|
|
|
|
*
|
|
|
|
* @return
|
|
|
|
* If successful a pointer to the new message port is returned, otherwise
|
|
|
|
* NULL is returned.
|
|
|
|
*/
|
|
|
|
su_port_t *su_source_create(void)
|
|
|
|
{
|
|
|
|
SuSource *ss;
|
|
|
|
|
|
|
|
SU_DEBUG_9(("su_source_create() called\n"));
|
|
|
|
|
|
|
|
ss = (SuSource *)g_source_new(su_source_funcs, (sizeof *ss));
|
|
|
|
|
|
|
|
if (ss) {
|
|
|
|
su_port_t *self = ss->ss_port;
|
|
|
|
|
|
|
|
self->sup_vtable = su_source_vtable;
|
|
|
|
self->sup_source = ss->ss_source;
|
|
|
|
|
|
|
|
SU_SOURCE_INITLOCK(self);
|
|
|
|
|
|
|
|
self->sup_tail = &self->sup_head;
|
|
|
|
self->sup_tid = g_thread_self();
|
|
|
|
|
|
|
|
SU_DEBUG_9(("su_source_with_main_context() returns %p\n", self));
|
|
|
|
|
|
|
|
return self;
|
|
|
|
} else {
|
|
|
|
su_perror("su_source_with_main_context(): su_home_clone");
|
|
|
|
SU_DEBUG_9(("su_source_with_main_context() fails\n"));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @internal Destroy a port. */
|
|
|
|
static
|
|
|
|
void su_source_finalize(GSource *gs)
|
|
|
|
{
|
|
|
|
SuSource *ss = (SuSource *)gs;
|
|
|
|
su_port_t *self = ss->ss_port;
|
|
|
|
|
|
|
|
assert(gs);
|
|
|
|
|
|
|
|
SU_DEBUG_9(("su_source_finalize() called\n"));
|
|
|
|
|
|
|
|
if (self->sup_waits)
|
|
|
|
free(self->sup_waits), self->sup_waits = NULL;
|
|
|
|
if (self->sup_wait_cbs)
|
|
|
|
free(self->sup_wait_cbs), self->sup_wait_cbs = NULL;
|
|
|
|
if (self->sup_wait_args)
|
|
|
|
free(self->sup_wait_args), self->sup_wait_args = NULL;
|
|
|
|
if (self->sup_wait_roots)
|
|
|
|
free(self->sup_wait_roots), self->sup_wait_roots = NULL;
|
|
|
|
if (self->sup_indices)
|
|
|
|
free(self->sup_indices), self->sup_indices = NULL;
|
|
|
|
|
|
|
|
su_home_deinit(self->sup_home);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Seconds from 1.1.1900 to 1.1.1970 */
|
|
|
|
#define NTP_EPOCH 2208988800UL
|
|
|
|
|
|
|
|
static
|
|
|
|
gboolean su_source_prepare(GSource *gs, gint *return_tout)
|
|
|
|
{
|
|
|
|
SuSource *ss = (SuSource *)gs;
|
|
|
|
su_port_t *self = ss->ss_port;
|
|
|
|
|
|
|
|
enter;
|
|
|
|
|
2007-01-02 15:11:55 +00:00
|
|
|
if (self->sup_head) {
|
|
|
|
*return_tout = 0;
|
2006-12-21 06:30:28 +00:00
|
|
|
return TRUE;
|
2007-01-02 15:11:55 +00:00
|
|
|
}
|
2006-12-21 06:30:28 +00:00
|
|
|
|
|
|
|
if (self->sup_timers) {
|
|
|
|
su_time_t now;
|
|
|
|
GTimeVal gtimeval;
|
|
|
|
su_duration_t tout;
|
|
|
|
|
|
|
|
g_source_get_current_time(gs, >imeval);
|
|
|
|
now.tv_sec = gtimeval.tv_sec + 2208988800UL;
|
|
|
|
now.tv_usec = gtimeval.tv_usec;
|
|
|
|
|
|
|
|
tout = su_timer_next_expires(self->sup_timers, now);
|
|
|
|
|
2007-01-02 15:11:55 +00:00
|
|
|
*return_tout = (tout < 0 || tout > (su_duration_t)G_MAXINT)?
|
|
|
|
-1 : (gint)tout;
|
2006-12-21 06:30:28 +00:00
|
|
|
|
2007-01-02 15:11:55 +00:00
|
|
|
return (tout == 0);
|
2006-12-21 06:30:28 +00:00
|
|
|
}
|
2007-01-02 15:11:55 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
gboolean su_source_check(GSource *gs)
|
|
|
|
{
|
|
|
|
SuSource *ss = (SuSource *)gs;
|
|
|
|
su_port_t *self = ss->ss_port;
|
|
|
|
gint tout;
|
|
|
|
unsigned i, I;
|
|
|
|
|
|
|
|
enter;
|
|
|
|
|
|
|
|
I = self->sup_n_waits;
|
|
|
|
|
|
|
|
#if SU_HAVE_POLL
|
|
|
|
for (i = 0; i < I; i++) {
|
|
|
|
if (self->sup_waits[i].revents)
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return su_source_prepare(gs, &tout);
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
gboolean su_source_dispatch(GSource *gs,
|
|
|
|
GSourceFunc callback,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
SuSource *ss = (SuSource *)gs;
|
|
|
|
su_port_t *self = ss->ss_port;
|
|
|
|
|
|
|
|
enter;
|
|
|
|
|
|
|
|
if (self->sup_head)
|
|
|
|
su_source_getmsgs(self);
|
|
|
|
|
|
|
|
if (self->sup_timers) {
|
|
|
|
su_time_t now;
|
|
|
|
GTimeVal gtimeval;
|
|
|
|
su_duration_t tout;
|
|
|
|
int timers = 0;
|
|
|
|
|
|
|
|
tout = SU_DURATION_MAX;
|
|
|
|
|
|
|
|
g_source_get_current_time(gs, >imeval);
|
|
|
|
|
|
|
|
now.tv_sec = gtimeval.tv_sec + 2208988800UL;
|
|
|
|
now.tv_usec = gtimeval.tv_usec;
|
|
|
|
|
|
|
|
timers = su_timer_expire(&self->sup_timers, &tout, now);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if SU_HAVE_POLL
|
|
|
|
{
|
|
|
|
su_root_t *root;
|
|
|
|
su_wait_t *waits = self->sup_waits;
|
|
|
|
unsigned i, n = self->sup_n_waits;
|
|
|
|
unsigned version = self->sup_registers;
|
|
|
|
|
|
|
|
for (i = 0; i < n; i++) {
|
|
|
|
if (waits[i].revents) {
|
|
|
|
root = self->sup_wait_roots[i];
|
|
|
|
self->sup_wait_cbs[i](root ? su_root_magic(root) : NULL,
|
|
|
|
&waits[i],
|
|
|
|
self->sup_wait_args[i]);
|
|
|
|
/* Callback used su_register()/su_unregister() */
|
|
|
|
if (version != self->sup_registers)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (!callback)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
return callback(user_data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void su_source_lock(su_port_t *self, char const *who)
|
|
|
|
{
|
|
|
|
SU_SOURCE_LOCK(self, who);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void su_source_unlock(su_port_t *self, char const *who)
|
|
|
|
{
|
|
|
|
SU_SOURCE_UNLOCK(self, who);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void su_source_incref(su_port_t *self, char const *who)
|
|
|
|
{
|
|
|
|
SU_SOURCE_INCREF(self, who);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void su_source_decref(su_port_t *self, int blocking, char const *who)
|
|
|
|
{
|
|
|
|
/* XXX - blocking? */
|
|
|
|
SU_SOURCE_DECREF(self, who);
|
|
|
|
}
|
|
|
|
|
|
|
|
GSource *su_source_gsource(su_port_t *self)
|
|
|
|
{
|
|
|
|
return self->sup_source;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @internal Send a message to the port. */
|
|
|
|
int su_source_send(su_port_t *self, su_msg_r rmsg)
|
|
|
|
{
|
|
|
|
enter;
|
|
|
|
|
|
|
|
if (self) {
|
|
|
|
su_msg_t *msg;
|
|
|
|
GMainContext *gmc;
|
|
|
|
|
|
|
|
SU_SOURCE_LOCK(self, "su_source_send");
|
|
|
|
|
|
|
|
msg = rmsg[0]; rmsg[0] = NULL;
|
|
|
|
*self->sup_tail = msg;
|
|
|
|
self->sup_tail = &msg->sum_next;
|
|
|
|
|
|
|
|
SU_SOURCE_UNLOCK(self, "su_source_send");
|
|
|
|
|
|
|
|
gmc = g_source_get_context(self->sup_source);
|
|
|
|
|
|
|
|
if (gmc)
|
|
|
|
g_main_context_wakeup(gmc);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
su_msg_destroy(rmsg);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @internal
|
|
|
|
* Execute the messages in the incoming queue until the queue is empty..
|
|
|
|
*
|
|
|
|
* @param self - pointer to a port object
|
|
|
|
*
|
|
|
|
* @retval 0 if there was a signal to handle,
|
|
|
|
* @retval -1 otherwise.
|
|
|
|
*/
|
|
|
|
static
|
|
|
|
int su_source_getmsgs(su_port_t *self)
|
|
|
|
{
|
|
|
|
enter;
|
|
|
|
|
|
|
|
if (self && self->sup_head) {
|
|
|
|
su_root_t *root;
|
|
|
|
su_msg_f f;
|
|
|
|
|
|
|
|
SU_SOURCE_INCREF(self, "su_source_getmsgs");
|
|
|
|
SU_SOURCE_LOCK(self, "su_source_getmsgs");
|
|
|
|
|
|
|
|
while (self->sup_head) {
|
|
|
|
su_msg_t *msg = self->sup_head;
|
|
|
|
self->sup_head = msg->sum_next;
|
|
|
|
if (!self->sup_head) {
|
|
|
|
assert(self->sup_tail == &msg->sum_next);
|
|
|
|
self->sup_tail = &self->sup_head;
|
|
|
|
}
|
|
|
|
root = msg->sum_to->sut_root;
|
|
|
|
f = msg->sum_func;
|
|
|
|
SU_SOURCE_UNLOCK(self, "su_source_getmsgs");
|
|
|
|
if (f)
|
|
|
|
f(su_root_magic(root), &msg, msg->sum_data);
|
|
|
|
su_msg_delivery_report(&msg);
|
|
|
|
SU_SOURCE_LOCK(self, "su_source_getmsgs");
|
|
|
|
}
|
|
|
|
|
|
|
|
SU_SOURCE_UNLOCK(self, "su_source_getmsgs");
|
|
|
|
SU_SOURCE_DECREF(self, "su_source_getmsgs");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @internal
|
|
|
|
*
|
|
|
|
* Register a @c su_wait_t object. The wait object, a callback function and
|
|
|
|
* a argument pointer is stored in the port object. The callback function
|
|
|
|
* will be called when the wait object is signaled.
|
|
|
|
*
|
|
|
|
* Please note if identical wait objects are inserted, only first one is
|
|
|
|
* ever signalled.
|
|
|
|
*
|
|
|
|
* @param self pointer to port
|
|
|
|
* @param root pointer to root object
|
|
|
|
* @param waits pointer to wait object
|
|
|
|
* @param callback callback function pointer
|
|
|
|
* @param arg argument given to callback function when it is invoked
|
|
|
|
* @param priority relative priority of the wait object
|
|
|
|
* (0 is normal, 1 important, 2 realtime)
|
|
|
|
*
|
|
|
|
* @return
|
|
|
|
* The function @su_source_register returns nonzero index of the wait object,
|
|
|
|
* or -1 upon an error. */
|
|
|
|
int su_source_register(su_port_t *self,
|
|
|
|
su_root_t *root,
|
|
|
|
su_wait_t *wait,
|
|
|
|
su_wakeup_f callback,
|
|
|
|
su_wakeup_arg_t *arg,
|
|
|
|
int priority)
|
|
|
|
{
|
|
|
|
unsigned i, j, I;
|
|
|
|
unsigned n;
|
|
|
|
|
|
|
|
enter;
|
|
|
|
|
|
|
|
assert(SU_SOURCE_OWN_THREAD(self));
|
|
|
|
|
|
|
|
n = self->sup_n_waits;
|
|
|
|
|
|
|
|
if (n >= self->sup_size_waits) {
|
|
|
|
/* Reallocate size arrays */
|
|
|
|
unsigned size;
|
|
|
|
unsigned *indices;
|
|
|
|
su_wait_t *waits;
|
|
|
|
su_wakeup_f *wait_cbs;
|
|
|
|
su_wakeup_arg_t **wait_args;
|
|
|
|
su_root_t **wait_tasks;
|
|
|
|
|
|
|
|
if (self->sup_size_waits == 0)
|
|
|
|
size = SU_WAIT_MIN;
|
|
|
|
else
|
|
|
|
size = 2 * self->sup_size_waits;
|
|
|
|
|
|
|
|
indices = realloc(self->sup_indices, size * sizeof(*indices));
|
|
|
|
if (indices) {
|
|
|
|
self->sup_indices = indices;
|
|
|
|
|
|
|
|
for (i = self->sup_size_waits; i < size; i++)
|
|
|
|
indices[i] = UINT_MAX;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < self->sup_n_waits; i++)
|
|
|
|
g_source_remove_poll(self->sup_source, (GPollFD*)&self->sup_waits[i]);
|
|
|
|
|
|
|
|
waits = realloc(self->sup_waits, size * sizeof(*waits));
|
|
|
|
if (waits)
|
|
|
|
self->sup_waits = waits;
|
|
|
|
|
|
|
|
for (i = 0; i < self->sup_n_waits; i++)
|
|
|
|
g_source_add_poll(self->sup_source, (GPollFD*)&waits[i]);
|
|
|
|
|
|
|
|
wait_cbs = realloc(self->sup_wait_cbs, size * sizeof(*wait_cbs));
|
|
|
|
if (wait_cbs)
|
|
|
|
self->sup_wait_cbs = wait_cbs;
|
|
|
|
|
|
|
|
wait_args = realloc(self->sup_wait_args, size * sizeof(*wait_args));
|
|
|
|
if (wait_args)
|
|
|
|
self->sup_wait_args = wait_args;
|
|
|
|
|
|
|
|
/* Add sup_wait_roots array, if needed */
|
|
|
|
wait_tasks = realloc(self->sup_wait_roots, size * sizeof(*wait_tasks));
|
|
|
|
if (wait_tasks)
|
|
|
|
self->sup_wait_roots = wait_tasks;
|
|
|
|
|
|
|
|
if (!(indices && waits && wait_cbs && wait_args && wait_tasks)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
self->sup_size_waits = size;
|
|
|
|
}
|
|
|
|
|
|
|
|
self->sup_n_waits++;
|
|
|
|
|
|
|
|
if (priority > 0) {
|
|
|
|
/* Insert */
|
|
|
|
for (; n > 0; n--) {
|
|
|
|
g_source_remove_poll(self->sup_source, (GPollFD*)&self->sup_waits[n-1]);
|
|
|
|
self->sup_waits[n] = self->sup_waits[n-1];
|
|
|
|
g_source_add_poll(self->sup_source, (GPollFD*)&self->sup_waits[n]);
|
|
|
|
self->sup_wait_cbs[n] = self->sup_wait_cbs[n-1];
|
|
|
|
self->sup_wait_args[n] = self->sup_wait_args[n-1];
|
|
|
|
self->sup_wait_roots[n] = self->sup_wait_roots[n-1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Append - no need to move anything */
|
|
|
|
}
|
|
|
|
|
|
|
|
self->sup_waits[n] = *wait;
|
|
|
|
g_source_add_poll(self->sup_source, (GPollFD*)&self->sup_waits[n]);
|
|
|
|
self->sup_wait_cbs[n] = callback;
|
|
|
|
self->sup_wait_args[n] = arg;
|
|
|
|
self->sup_wait_roots[n] = root;
|
|
|
|
|
|
|
|
I = self->sup_max_index;
|
|
|
|
|
|
|
|
for (i = 0; i < I; i++)
|
|
|
|
if (self->sup_indices[i] == UINT_MAX)
|
|
|
|
break;
|
|
|
|
else if (self->sup_indices[i] >= n)
|
|
|
|
self->sup_indices[i]++;
|
|
|
|
|
|
|
|
if (i == I)
|
|
|
|
self->sup_max_index++;
|
|
|
|
|
|
|
|
if (n + 1 < self->sup_n_waits)
|
|
|
|
for (j = i; j < I; j++)
|
|
|
|
if (self->sup_indices[j] != UINT_MAX &&
|
|
|
|
self->sup_indices[j] >= n)
|
|
|
|
self->sup_indices[j]++;
|
|
|
|
|
|
|
|
self->sup_indices[i] = n;
|
|
|
|
|
|
|
|
self->sup_registers++;
|
|
|
|
|
|
|
|
return i + 1; /* 0 is failure */
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Unregister a su_wait_t object.
|
|
|
|
*
|
|
|
|
* The function su_source_unregister() unregisters a su_wait_t object. The
|
|
|
|
* wait object, a callback function and a argument are removed from the
|
|
|
|
* port object.
|
|
|
|
*
|
|
|
|
* @param self - pointer to port object
|
|
|
|
* @param root - pointer to root object
|
|
|
|
* @param wait - pointer to wait object
|
|
|
|
* @param callback - callback function pointer (may be NULL)
|
|
|
|
* @param arg - argument given to callback function when it is invoked
|
|
|
|
* (may be NULL)
|
|
|
|
*
|
|
|
|
* @return Nonzero index of the wait object, or -1 upon an error.
|
|
|
|
*/
|
|
|
|
int su_source_unregister(su_port_t *self,
|
|
|
|
su_root_t *root,
|
|
|
|
su_wait_t *wait,
|
|
|
|
su_wakeup_f callback, /* XXX - ignored */
|
|
|
|
su_wakeup_arg_t *arg)
|
|
|
|
{
|
|
|
|
unsigned n, N;
|
|
|
|
unsigned i, I, j, *indices;
|
|
|
|
|
|
|
|
enter;
|
|
|
|
|
|
|
|
assert(self);
|
|
|
|
assert(SU_SOURCE_OWN_THREAD(self));
|
|
|
|
|
|
|
|
i = (unsigned)-1;
|
|
|
|
N = self->sup_n_waits;
|
|
|
|
I = self->sup_max_index;
|
|
|
|
indices = self->sup_indices;
|
|
|
|
|
|
|
|
for (n = 0; n < N; n++) {
|
|
|
|
if (SU_WAIT_CMP(wait[0], self->sup_waits[n]) != 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* Found - delete it */
|
|
|
|
if (indices[n] == n)
|
|
|
|
i = n;
|
|
|
|
else for (i = 0; i < I; i++)
|
|
|
|
if (indices[i] == n)
|
|
|
|
break;
|
|
|
|
|
|
|
|
assert(i < I);
|
|
|
|
|
|
|
|
indices[i] = UINT_MAX;
|
|
|
|
|
|
|
|
g_source_remove_poll(self->sup_source, (GPollFD*)&self->sup_waits[n]);
|
|
|
|
|
|
|
|
self->sup_n_waits = N = N - 1;
|
|
|
|
|
|
|
|
if (n < N)
|
|
|
|
for (j = 0; j < I; j++)
|
|
|
|
if (self->sup_indices[j] != UINT_MAX &&
|
|
|
|
self->sup_indices[j] > n)
|
|
|
|
self->sup_indices[j]--;
|
|
|
|
|
|
|
|
for (; n < N; n++) {
|
|
|
|
g_source_remove_poll(self->sup_source, (GPollFD*)&self->sup_waits[n+1]);
|
|
|
|
self->sup_waits[n] = self->sup_waits[n+1];
|
|
|
|
g_source_add_poll(self->sup_source, (GPollFD*)&self->sup_waits[n]);
|
|
|
|
self->sup_wait_cbs[n] = self->sup_wait_cbs[n+1];
|
|
|
|
self->sup_wait_args[n] = self->sup_wait_args[n+1];
|
|
|
|
self->sup_wait_roots[n] = self->sup_wait_roots[n+1];
|
|
|
|
}
|
|
|
|
|
|
|
|
i += 1; /* 0 is failure */
|
|
|
|
|
|
|
|
if (i == I)
|
|
|
|
self->sup_max_index--;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
self->sup_registers++;
|
|
|
|
|
|
|
|
return (int)i;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Deregister a su_wait_t object.
|
|
|
|
*
|
|
|
|
* The function su_source_deregister() deregisters a su_wait_t registrattion.
|
|
|
|
* The wait object, a callback function and a argument are removed from the
|
|
|
|
* port object.
|
|
|
|
*
|
|
|
|
* @param self - pointer to port object
|
|
|
|
* @param i - registration index
|
|
|
|
*
|
|
|
|
* @return Index of the wait object, or -1 upon an error.
|
|
|
|
*/
|
|
|
|
int su_source_deregister(su_port_t *self, int i)
|
|
|
|
{
|
|
|
|
unsigned j, n, N;
|
|
|
|
unsigned I, *indices;
|
|
|
|
su_wait_t wait[1];
|
|
|
|
|
|
|
|
enter;
|
|
|
|
|
|
|
|
assert(self);
|
|
|
|
assert(SU_SOURCE_OWN_THREAD(self));
|
|
|
|
|
|
|
|
if (i <= 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
N = self->sup_n_waits;
|
|
|
|
I = self->sup_max_index;
|
|
|
|
indices = self->sup_indices;
|
|
|
|
|
|
|
|
assert((unsigned)i < I + 1);
|
|
|
|
|
|
|
|
n = indices[i - 1];
|
|
|
|
|
|
|
|
if (n == UINT_MAX)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
self->sup_n_waits = N = N - 1;
|
|
|
|
|
|
|
|
wait[0] = self->sup_waits[n];
|
|
|
|
|
|
|
|
g_source_remove_poll(self->sup_source, (GPollFD*)&self->sup_waits[n]);
|
|
|
|
|
|
|
|
if (n < N)
|
|
|
|
for (j = 0; j < I; j++)
|
|
|
|
if (self->sup_indices[j] != UINT_MAX &&
|
|
|
|
self->sup_indices[j] > n)
|
|
|
|
self->sup_indices[j]--;
|
|
|
|
|
|
|
|
for (; n < N; n++) {
|
|
|
|
g_source_remove_poll(self->sup_source, (GPollFD*)&self->sup_waits[n + 1]);
|
|
|
|
self->sup_waits[n] = self->sup_waits[n+1];
|
|
|
|
g_source_add_poll(self->sup_source, (GPollFD*)&self->sup_waits[n]);
|
|
|
|
self->sup_wait_cbs[n] = self->sup_wait_cbs[n+1];
|
|
|
|
self->sup_wait_args[n] = self->sup_wait_args[n+1];
|
|
|
|
self->sup_wait_roots[n] = self->sup_wait_roots[n+1];
|
|
|
|
}
|
|
|
|
|
|
|
|
indices[i - 1] = UINT_MAX;
|
|
|
|
|
|
|
|
if ((unsigned)i == I)
|
|
|
|
self->sup_max_index--;
|
|
|
|
|
|
|
|
su_wait_destroy(wait);
|
|
|
|
|
|
|
|
self->sup_registers++;
|
|
|
|
|
|
|
|
return (int)i;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @internal
|
|
|
|
* Unregister all su_wait_t objects.
|
|
|
|
*
|
|
|
|
* The function su_source_unregister_all() unregisters all su_wait_t objects
|
|
|
|
* associated with given root object destroys all queued timers.
|
|
|
|
*
|
|
|
|
* @param self - pointer to port object
|
|
|
|
* @param root - pointer to root object
|
|
|
|
*
|
|
|
|
* @return Number of wait objects removed.
|
|
|
|
*/
|
|
|
|
int su_source_unregister_all(su_port_t *self,
|
|
|
|
su_root_t *root)
|
|
|
|
{
|
|
|
|
unsigned i, j;
|
|
|
|
unsigned n_waits;
|
|
|
|
su_wait_t *waits;
|
|
|
|
su_wakeup_f *wait_cbs;
|
|
|
|
su_wakeup_arg_t**wait_args;
|
|
|
|
su_root_t **wait_roots;
|
|
|
|
|
|
|
|
enter;
|
|
|
|
|
|
|
|
assert(SU_SOURCE_OWN_THREAD(self));
|
|
|
|
|
|
|
|
n_waits = self->sup_n_waits;
|
|
|
|
waits = self->sup_waits;
|
|
|
|
wait_cbs = self->sup_wait_cbs;
|
|
|
|
wait_args = self->sup_wait_args;
|
|
|
|
wait_roots = self->sup_wait_roots;
|
|
|
|
|
|
|
|
for (i = j = 0; (unsigned)i < n_waits; i++) {
|
|
|
|
if (wait_roots[i] == root) {
|
|
|
|
/* XXX - we should free all resources associated with this */
|
|
|
|
g_source_remove_poll(self->sup_source, (GPollFD*)&waits[i]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (i != j) {
|
|
|
|
g_source_remove_poll(self->sup_source, (GPollFD*)&waits[i]);
|
|
|
|
waits[j] = waits[i];
|
|
|
|
wait_cbs[j] = wait_cbs[i];
|
|
|
|
wait_args[j] = wait_args[i];
|
|
|
|
wait_roots[j] = wait_roots[i];
|
|
|
|
g_source_add_poll(self->sup_source, (GPollFD*)&waits[i]);
|
|
|
|
}
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
|
|
|
|
self->sup_n_waits = j;
|
|
|
|
self->sup_registers++;
|
|
|
|
|
|
|
|
return n_waits - j;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**Set mask for a registered event. @internal
|
|
|
|
*
|
|
|
|
* The function su_source_eventmask() sets the mask describing events that can
|
|
|
|
* signal the registered callback.
|
|
|
|
*
|
|
|
|
* @param port pointer to port object
|
|
|
|
* @param index registration index
|
|
|
|
* @param socket socket
|
|
|
|
* @param events new event mask
|
|
|
|
*
|
|
|
|
* @retval 0 when successful,
|
|
|
|
* @retval -1 upon an error.
|
|
|
|
*/
|
|
|
|
int su_source_eventmask(su_port_t *self, int index, int socket, int events)
|
|
|
|
{
|
|
|
|
unsigned n;
|
|
|
|
int retval;
|
|
|
|
|
|
|
|
enter;
|
|
|
|
|
|
|
|
assert(self);
|
|
|
|
assert(SU_SOURCE_OWN_THREAD(self));
|
|
|
|
assert(0 < index && (unsigned)index <= self->sup_max_index);
|
|
|
|
|
|
|
|
if (index <= 0 || (unsigned)index > self->sup_max_index)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
n = self->sup_indices[index - 1];
|
|
|
|
|
|
|
|
if (n == UINT_MAX)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
g_source_remove_poll(self->sup_source, (GPollFD*)&self->sup_waits[n]);
|
|
|
|
|
|
|
|
retval = su_wait_mask(&self->sup_waits[n], socket, events);
|
|
|
|
|
|
|
|
g_source_add_poll(self->sup_source, (GPollFD*)&self->sup_waits[n]);
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
int su_source_multishot(su_port_t *self, int multishot)
|
|
|
|
{
|
|
|
|
if (multishot == -1)
|
|
|
|
return 1;
|
|
|
|
else if (multishot == 0 || multishot == 1)
|
|
|
|
return 1; /* Always enabled */
|
|
|
|
else
|
|
|
|
return (errno = EINVAL), -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @internal Enable threadsafe operation. */
|
|
|
|
static
|
|
|
|
int su_source_threadsafe(su_port_t *port)
|
|
|
|
{
|
|
|
|
return su_home_threadsafe(port->sup_home);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** @internal Main loop.
|
|
|
|
*
|
|
|
|
* The function @c su_source_run() runs the main loop
|
|
|
|
*
|
|
|
|
* The function @c su_source_run() runs until @c su_source_break() is called
|
|
|
|
* from a callback.
|
|
|
|
*
|
|
|
|
* @param self pointer to root object
|
|
|
|
* */
|
|
|
|
void su_source_run(su_port_t *self)
|
|
|
|
{
|
|
|
|
GMainContext *gmc;
|
|
|
|
GMainLoop *gml;
|
|
|
|
|
|
|
|
enter;
|
|
|
|
|
|
|
|
gmc = g_source_get_context(self->sup_source);
|
|
|
|
if (gmc && g_main_context_acquire(gmc)) {
|
|
|
|
gml = g_main_loop_new(gmc, TRUE);
|
|
|
|
self->sup_main_loop = gml;
|
|
|
|
g_main_loop_run(gml);
|
|
|
|
g_main_loop_unref(gml);
|
|
|
|
self->sup_main_loop = NULL;
|
|
|
|
g_main_context_release(gmc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @internal
|
|
|
|
* The function @c su_source_break() is used to terminate execution of @c
|
|
|
|
* su_source_run(). It can be called from a callback function.
|
|
|
|
*
|
|
|
|
* @param self pointer to port
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
void su_source_break(su_port_t *self)
|
|
|
|
{
|
|
|
|
enter;
|
|
|
|
|
|
|
|
if (self->sup_main_loop)
|
|
|
|
g_main_loop_quit(self->sup_main_loop);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @internal Block until wait object is signaled or timeout.
|
|
|
|
*
|
|
|
|
* This function waits for wait objects and the timers associated with
|
|
|
|
* the root object. When any wait object is signaled or timer is
|
|
|
|
* expired, it invokes the callbacks.
|
|
|
|
*
|
|
|
|
* This function returns when a callback has been invoked or @c tout
|
|
|
|
* milliseconds is elapsed.
|
|
|
|
*
|
|
|
|
* @param self pointer to port
|
|
|
|
* @param tout timeout in milliseconds
|
|
|
|
*
|
|
|
|
* @Return
|
|
|
|
* Milliseconds to the next invocation of timer, or @c SU_WAIT_FOREVER if
|
|
|
|
* there are no active timers.
|
|
|
|
*/
|
|
|
|
su_duration_t su_source_step(su_port_t *self, su_duration_t tout)
|
|
|
|
{
|
|
|
|
GMainContext *gmc;
|
|
|
|
|
|
|
|
enter;
|
|
|
|
|
|
|
|
gmc = g_source_get_context(self->sup_source);
|
|
|
|
|
|
|
|
if (gmc && g_main_context_acquire(gmc)) {
|
2007-01-02 15:11:55 +00:00
|
|
|
GPollFD *fds = NULL;
|
|
|
|
gint fds_size = 0;
|
|
|
|
gint fds_wait;
|
2006-12-21 06:30:28 +00:00
|
|
|
gint priority = G_MAXINT;
|
2007-01-02 15:11:55 +00:00
|
|
|
gint src_tout = -1;
|
2006-12-21 06:30:28 +00:00
|
|
|
|
2007-01-02 15:11:55 +00:00
|
|
|
g_main_context_prepare(gmc, &priority);
|
2006-12-21 06:30:28 +00:00
|
|
|
|
2007-01-02 15:11:55 +00:00
|
|
|
fds_wait = g_main_context_query(gmc, priority, &src_tout, NULL, 0);
|
|
|
|
while (fds_wait > fds_size) {
|
|
|
|
fds = g_alloca(fds_wait * sizeof(fds[0]));
|
|
|
|
fds_size = fds_wait;
|
|
|
|
fds_wait = g_main_context_query(gmc, priority, &src_tout, fds, fds_size);
|
2006-12-21 06:30:28 +00:00
|
|
|
}
|
2007-01-02 15:11:55 +00:00
|
|
|
|
|
|
|
if (src_tout >= 0 && tout > (su_duration_t)src_tout)
|
|
|
|
tout = src_tout;
|
|
|
|
|
|
|
|
su_wait((su_wait_t *)fds, fds_wait, tout);
|
|
|
|
|
|
|
|
g_main_context_check(gmc, priority, fds, fds_wait);
|
|
|
|
|
|
|
|
g_main_context_dispatch(gmc);
|
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
g_main_context_release(gmc);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** @internal
|
|
|
|
* Checks if the calling thread owns the port object.
|
|
|
|
*
|
|
|
|
* @param self pointer to a port object
|
|
|
|
*
|
|
|
|
* @retval true (nonzero) if the calling thread owns the port,
|
|
|
|
* @retval false (zero) otherwise.
|
|
|
|
*/
|
|
|
|
int su_source_own_thread(su_port_t const *self)
|
|
|
|
{
|
|
|
|
return self == NULL || SU_SOURCE_OWN_THREAD(self);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
/** @internal
|
|
|
|
* Prints out the contents of the port.
|
|
|
|
*
|
|
|
|
* @param self pointer to a port
|
|
|
|
* @param f pointer to a file (if @c NULL, uses @c stdout).
|
|
|
|
*/
|
|
|
|
void su_source_dump(su_port_t const *self, FILE *f)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
#define IS_WAIT_IN(x) (((x)->events & SU_WAIT_IN) ? "IN" : "")
|
|
|
|
#define IS_WAIT_OUT(x) (((x)->events & SU_WAIT_OUT) ? "OUT" : "")
|
|
|
|
#define IS_WAIT_ACCEPT(x) (((x)->events & SU_WAIT_ACCEPT) ? "ACCEPT" : "")
|
|
|
|
|
|
|
|
if (f == NULL)
|
|
|
|
f = stdout;
|
|
|
|
|
|
|
|
fprintf(f, "su_port_t at %p:\n", self);
|
|
|
|
fprintf(f, "\tport is%s running\n", self->sup_running ? "" : "not ");
|
|
|
|
fprintf(f, "\tport tid %p\n", (void *)self->sup_tid);
|
|
|
|
fprintf(f, "\t%d wait objects\n", self->sup_n_waits);
|
|
|
|
for (i = 0; i < self->sup_n_waits; i++) {
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* =========================================================================
|
|
|
|
* Pre-poll() callback
|
|
|
|
*/
|
|
|
|
|
|
|
|
int su_source_add_prepoll(su_port_t *port,
|
|
|
|
su_root_t *root,
|
|
|
|
su_prepoll_f *callback,
|
|
|
|
su_prepoll_magic_t *magic)
|
|
|
|
{
|
|
|
|
#if 0
|
|
|
|
if (port->sup_prepoll)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
port->sup_prepoll = callback;
|
|
|
|
port->sup_pp_magic = magic;
|
|
|
|
port->sup_pp_root = root;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
#else
|
|
|
|
return -1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
int su_source_remove_prepoll(su_port_t *port,
|
|
|
|
su_root_t *root)
|
|
|
|
{
|
|
|
|
#if 0
|
|
|
|
if (port->sup_pp_root != root)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
port->sup_prepoll = NULL;
|
|
|
|
port->sup_pp_magic = NULL;
|
|
|
|
port->sup_pp_root = NULL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
#else
|
|
|
|
return -1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/* =========================================================================
|
|
|
|
* Timers
|
|
|
|
*/
|
|
|
|
|
|
|
|
static
|
|
|
|
su_timer_t **su_source_timers(su_port_t *self)
|
|
|
|
{
|
|
|
|
return &self->sup_timers;
|
|
|
|
}
|