1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/ossrv/ofdbus/dbus/bus/expirelist.c Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,328 @@
1.4 +/* -*- mode: C; c-file-style: "gnu" -*- */
1.5 +/* expirelist.c List of items that expire
1.6 + *
1.7 + * Copyright (C) 2003 Red Hat, Inc.
1.8 + * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
1.9 + * Licensed under the Academic Free License version 2.1
1.10 + *
1.11 + * This program is free software; you can redistribute it and/or modify
1.12 + * it under the terms of the GNU General Public License as published by
1.13 + * the Free Software Foundation; either version 2 of the License, or
1.14 + * (at your option) any later version.
1.15 + *
1.16 + * This program is distributed in the hope that it will be useful,
1.17 + * but WITHOUT ANY WARRANTY; without even the implied warranty of
1.18 + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
1.19 + * GNU General Public License for more details.
1.20 + *
1.21 + * You should have received a copy of the GNU General Public License
1.22 + * along with this program; if not, write to the Free Software
1.23 + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
1.24 + *
1.25 + */
1.26 +
1.27 +#include "expirelist.h"
1.28 +#include "test.h"
1.29 +#ifndef __SYMBIAN32__
1.30 +#include <dbus/dbus-internals.h>
1.31 +#include <dbus/dbus-mainloop.h>
1.32 +#include <dbus/dbus-timeout.h>
1.33 +#else
1.34 +#include "dbus-internals.h"
1.35 +#include "dbus-mainloop.h"
1.36 +#include "dbus-timeout.h"
1.37 +#endif //__SYMBIAN32__
1.38 +
1.39 +static dbus_bool_t expire_timeout_handler (void *data);
1.40 +
1.41 +static void
1.42 +call_timeout_callback (DBusTimeout *timeout,
1.43 + void *data)
1.44 +{
1.45 + /* can return FALSE on OOM but we just let it fire again later */
1.46 + dbus_timeout_handle (timeout);
1.47 +}
1.48 +
1.49 +BusExpireList*
1.50 +bus_expire_list_new (DBusLoop *loop,
1.51 + int expire_after,
1.52 + BusExpireFunc expire_func,
1.53 + void *data)
1.54 +{
1.55 + BusExpireList *list;
1.56 +
1.57 + list = dbus_new0 (BusExpireList, 1);
1.58 + if (list == NULL)
1.59 + return NULL;
1.60 +
1.61 + list->expire_func = expire_func;
1.62 + list->data = data;
1.63 + list->loop = loop;
1.64 + list->expire_after = expire_after;
1.65 +
1.66 + list->timeout = _dbus_timeout_new (100, /* irrelevant */
1.67 + expire_timeout_handler,
1.68 + list, NULL);
1.69 + if (list->timeout == NULL)
1.70 + goto failed;
1.71 +
1.72 + _dbus_timeout_set_enabled (list->timeout, FALSE);
1.73 +
1.74 + if (!_dbus_loop_add_timeout (list->loop,
1.75 + list->timeout,
1.76 + call_timeout_callback, NULL, NULL))
1.77 + goto failed;
1.78 +
1.79 + return list;
1.80 +
1.81 + failed:
1.82 + if (list->timeout)
1.83 + _dbus_timeout_unref (list->timeout);
1.84 +
1.85 + dbus_free (list);
1.86 +
1.87 + return NULL;
1.88 +}
1.89 +
1.90 +void
1.91 +bus_expire_list_free (BusExpireList *list)
1.92 +{
1.93 + _dbus_assert (list->items == NULL);
1.94 +
1.95 + _dbus_loop_remove_timeout (list->loop, list->timeout,
1.96 + call_timeout_callback, NULL);
1.97 +
1.98 + _dbus_timeout_unref (list->timeout);
1.99 +
1.100 + dbus_free (list);
1.101 +}
1.102 +
1.103 +void
1.104 +bus_expire_timeout_set_interval (DBusTimeout *timeout,
1.105 + int next_interval)
1.106 +{
1.107 + if (next_interval >= 0)
1.108 + {
1.109 + _dbus_timeout_set_interval (timeout,
1.110 + next_interval);
1.111 + _dbus_timeout_set_enabled (timeout, TRUE);
1.112 +
1.113 + _dbus_verbose ("Enabled expire timeout with interval %d\n",
1.114 + next_interval);
1.115 + }
1.116 + else if (dbus_timeout_get_enabled (timeout))
1.117 + {
1.118 + _dbus_timeout_set_enabled (timeout, FALSE);
1.119 +
1.120 + _dbus_verbose ("Disabled expire timeout\n");
1.121 + }
1.122 + else
1.123 + _dbus_verbose ("No need to disable expire timeout\n");
1.124 +}
1.125 +
1.126 +static int
1.127 +do_expiration_with_current_time (BusExpireList *list,
1.128 + long tv_sec,
1.129 + long tv_usec)
1.130 +{
1.131 + DBusList *link;
1.132 + int next_interval;
1.133 +
1.134 + next_interval = -1;
1.135 +
1.136 + link = _dbus_list_get_first_link (&list->items);
1.137 + while (link != NULL)
1.138 + {
1.139 + DBusList *next = _dbus_list_get_next_link (&list->items, link);
1.140 + double elapsed;
1.141 + BusExpireItem *item;
1.142 +
1.143 + item = link->data;
1.144 +
1.145 + elapsed = ELAPSED_MILLISECONDS_SINCE (item->added_tv_sec,
1.146 + item->added_tv_usec,
1.147 + tv_sec, tv_usec);
1.148 +
1.149 + if (elapsed >= (double) list->expire_after)
1.150 + {
1.151 + _dbus_verbose ("Expiring an item %p\n", item);
1.152 +
1.153 + /* If the expire function fails, we just end up expiring
1.154 + * this item next time we walk through the list. This would
1.155 + * be an indeterminate time normally, so we set up the
1.156 + * next_interval to be "shortly" (just enough to avoid
1.157 + * a busy loop)
1.158 + */
1.159 + if (!(* list->expire_func) (list, link, list->data))
1.160 + {
1.161 + next_interval = _dbus_get_oom_wait ();
1.162 + break;
1.163 + }
1.164 + }
1.165 + else
1.166 + {
1.167 + /* We can end the loop, since the connections are in oldest-first order */
1.168 + next_interval = ((double)list->expire_after) - elapsed;
1.169 + _dbus_verbose ("Item %p expires in %d milliseconds\n",
1.170 + item, next_interval);
1.171 +
1.172 + break;
1.173 + }
1.174 +
1.175 + link = next;
1.176 + }
1.177 +
1.178 + return next_interval;
1.179 +}
1.180 +
1.181 +static void
1.182 +bus_expirelist_expire (BusExpireList *list)
1.183 +{
1.184 + int next_interval;
1.185 +
1.186 + next_interval = -1;
1.187 +
1.188 + if (list->items != NULL)
1.189 + {
1.190 + long tv_sec, tv_usec;
1.191 +
1.192 + _dbus_get_current_time (&tv_sec, &tv_usec);
1.193 +
1.194 + next_interval = do_expiration_with_current_time (list, tv_sec, tv_usec);
1.195 + }
1.196 +
1.197 + bus_expire_timeout_set_interval (list->timeout, next_interval);
1.198 +}
1.199 +
1.200 +static dbus_bool_t
1.201 +expire_timeout_handler (void *data)
1.202 +{
1.203 + BusExpireList *list = data;
1.204 +
1.205 + _dbus_verbose ("Running %s\n", _DBUS_FUNCTION_NAME);
1.206 +
1.207 + /* note that this may remove the timeout */
1.208 + bus_expirelist_expire (list);
1.209 +
1.210 + return TRUE;
1.211 +}
1.212 +
1.213 +#ifdef DBUS_BUILD_TESTS
1.214 +
1.215 +typedef struct
1.216 +{
1.217 + BusExpireItem item;
1.218 + int expire_count;
1.219 +} TestExpireItem;
1.220 +
1.221 +static dbus_bool_t
1.222 +test_expire_func (BusExpireList *list,
1.223 + DBusList *link,
1.224 + void *data)
1.225 +{
1.226 + TestExpireItem *t;
1.227 +
1.228 + t = (TestExpireItem*) link->data;
1.229 +
1.230 + t->expire_count += 1;
1.231 +
1.232 + return TRUE;
1.233 +}
1.234 +
1.235 +static void
1.236 +time_add_milliseconds (long *tv_sec,
1.237 + long *tv_usec,
1.238 + int milliseconds)
1.239 +{
1.240 + *tv_sec = *tv_sec + milliseconds / 1000;
1.241 + *tv_usec = *tv_usec + milliseconds * 1000;
1.242 + if (*tv_usec >= 1000000)
1.243 + {
1.244 + *tv_usec -= 1000000;
1.245 + *tv_sec += 1;
1.246 + }
1.247 +}
1.248 +
1.249 +dbus_bool_t
1.250 +bus_expire_list_test (const DBusString *test_data_dir)
1.251 +{
1.252 + DBusLoop *loop;
1.253 + BusExpireList *list;
1.254 + long tv_sec, tv_usec;
1.255 + long tv_sec_not_expired, tv_usec_not_expired;
1.256 + long tv_sec_expired, tv_usec_expired;
1.257 + long tv_sec_past, tv_usec_past;
1.258 + TestExpireItem *item;
1.259 + int next_interval;
1.260 + dbus_bool_t result = FALSE;
1.261 +
1.262 +
1.263 + loop = _dbus_loop_new ();
1.264 + _dbus_assert (loop != NULL);
1.265 +
1.266 +#define EXPIRE_AFTER 100
1.267 +
1.268 + list = bus_expire_list_new (loop, EXPIRE_AFTER,
1.269 + test_expire_func, NULL);
1.270 + _dbus_assert (list != NULL);
1.271 +
1.272 + _dbus_get_current_time (&tv_sec, &tv_usec);
1.273 +
1.274 + tv_sec_not_expired = tv_sec;
1.275 + tv_usec_not_expired = tv_usec;
1.276 + time_add_milliseconds (&tv_sec_not_expired,
1.277 + &tv_usec_not_expired, EXPIRE_AFTER - 1);
1.278 +
1.279 + tv_sec_expired = tv_sec;
1.280 + tv_usec_expired = tv_usec;
1.281 + time_add_milliseconds (&tv_sec_expired,
1.282 + &tv_usec_expired, EXPIRE_AFTER);
1.283 +
1.284 +
1.285 + tv_sec_past = tv_sec - 1;
1.286 + tv_usec_past = tv_usec;
1.287 +
1.288 + item = dbus_new0 (TestExpireItem, 1);
1.289 +
1.290 + if (item == NULL)
1.291 + goto oom;
1.292 +
1.293 + item->item.added_tv_sec = tv_sec;
1.294 + item->item.added_tv_usec = tv_usec;
1.295 + if (!_dbus_list_append (&list->items, item))
1.296 + _dbus_assert_not_reached ("out of memory");
1.297 +
1.298 + next_interval =
1.299 + do_expiration_with_current_time (list, tv_sec_not_expired,
1.300 + tv_usec_not_expired);
1.301 + _dbus_assert (item->expire_count == 0);
1.302 + _dbus_verbose ("next_interval = %d\n", next_interval);
1.303 + _dbus_assert (next_interval == 1);
1.304 +
1.305 + next_interval =
1.306 + do_expiration_with_current_time (list, tv_sec_expired,
1.307 + tv_usec_expired);
1.308 + _dbus_assert (item->expire_count == 1);
1.309 + _dbus_verbose ("next_interval = %d\n", next_interval);
1.310 + _dbus_assert (next_interval == -1);
1.311 +
1.312 + next_interval =
1.313 + do_expiration_with_current_time (list, tv_sec_past,
1.314 + tv_usec_past);
1.315 + _dbus_assert (item->expire_count == 1);
1.316 + _dbus_verbose ("next_interval = %d\n", next_interval);
1.317 + _dbus_assert (next_interval == 1000 + EXPIRE_AFTER);
1.318 +
1.319 + _dbus_list_clear (&list->items);
1.320 + dbus_free (item);
1.321 +
1.322 + bus_expire_list_free (list);
1.323 + _dbus_loop_unref (loop);
1.324 +
1.325 + result = TRUE;
1.326 +
1.327 + oom:
1.328 + return result;
1.329 +}
1.330 +
1.331 +#endif /* DBUS_BUILD_TESTS */