2006-12-21 06:30:28 +00:00
|
|
|
/*
|
|
|
|
* This file is part of the Sofia-SIP package
|
|
|
|
*
|
|
|
|
* Copyright (C) 2006 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
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**@CFILE nua_dialog.c
|
|
|
|
* @brief Dialog and dialog usage handling
|
|
|
|
*
|
|
|
|
* @author Pekka Pessi <Pekka.Pessi@nokia.com>
|
|
|
|
*
|
|
|
|
* @date Created: Wed Mar 8 11:48:49 EET 2006 ppessi
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <limits.h>
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
#include <sofia-sip/string0.h>
|
|
|
|
#include <sofia-sip/su_uniqueid.h>
|
|
|
|
|
|
|
|
#include <sofia-sip/sip_protos.h>
|
|
|
|
|
|
|
|
#define NUA_OWNER_T su_home_t
|
|
|
|
|
|
|
|
#include "nua_dialog.h"
|
|
|
|
|
|
|
|
#define SU_LOG (nua_log)
|
|
|
|
#include <sofia-sip/su_debug.h>
|
|
|
|
|
|
|
|
#ifndef NONE
|
|
|
|
#define NONE ((void *)-1)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* ======================================================================== */
|
|
|
|
/* Dialog handling */
|
|
|
|
|
|
|
|
static void nua_dialog_usage_remove_at(nua_owner_t*, nua_dialog_state_t*,
|
|
|
|
nua_dialog_usage_t**);
|
|
|
|
static void nua_dialog_log_usage(nua_owner_t *, nua_dialog_state_t *);
|
|
|
|
|
|
|
|
/**@internal
|
|
|
|
* UAS tag and route.
|
|
|
|
*
|
|
|
|
* Update dialog tags and route on the UAS side.
|
|
|
|
*
|
|
|
|
* @param own dialog owner
|
|
|
|
* @param ds dialog state
|
|
|
|
* @param sip SIP message containing response used to update dialog
|
|
|
|
* @param rtag if true, set remote tag within the leg
|
|
|
|
*/
|
|
|
|
void nua_dialog_uas_route(nua_owner_t *own,
|
|
|
|
nua_dialog_state_t *ds,
|
|
|
|
sip_t const *sip,
|
|
|
|
int rtag)
|
|
|
|
{
|
|
|
|
int established = nua_dialog_is_established(ds);
|
|
|
|
|
|
|
|
if (!established && sip->sip_from->a_tag)
|
|
|
|
ds->ds_remote_tag = su_strdup(own, sip->sip_from->a_tag);
|
|
|
|
|
|
|
|
if (ds->ds_leg == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
nta_leg_server_route(ds->ds_leg, sip->sip_record_route, sip->sip_contact);
|
|
|
|
ds->ds_route = ds->ds_route || sip->sip_record_route || sip->sip_contact;
|
|
|
|
|
|
|
|
if (rtag && !established && sip->sip_from->a_tag)
|
|
|
|
nta_leg_rtag(ds->ds_leg, sip->sip_from->a_tag);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**@internal
|
|
|
|
* UAC tag and route.
|
|
|
|
*
|
|
|
|
* Update dialog tags and route on the UAC side.
|
|
|
|
*
|
|
|
|
* @param own dialog owner
|
|
|
|
* @param ds dialog state
|
|
|
|
* @param sip SIP message containing response used to update dialog
|
|
|
|
* @param rtag if true, set remote tag within the leg
|
|
|
|
*/
|
|
|
|
void nua_dialog_uac_route(nua_owner_t *own,
|
|
|
|
nua_dialog_state_t *ds,
|
|
|
|
sip_t const *sip,
|
|
|
|
int rtag)
|
|
|
|
{
|
|
|
|
int established = nua_dialog_is_established(ds);
|
|
|
|
|
|
|
|
if (!established && sip->sip_to->a_tag)
|
|
|
|
ds->ds_remote_tag = su_strdup(own, sip->sip_to->a_tag);
|
|
|
|
|
|
|
|
if (ds->ds_leg == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
nta_leg_client_route(ds->ds_leg, sip->sip_record_route, sip->sip_contact);
|
|
|
|
ds->ds_route = ds->ds_route || sip->sip_record_route || sip->sip_contact;
|
|
|
|
|
|
|
|
if (rtag && !established && sip->sip_to->a_tag)
|
|
|
|
nta_leg_rtag(ds->ds_leg, sip->sip_to->a_tag);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**@internal Store information from remote endpoint. */
|
|
|
|
void nua_dialog_store_peer_info(nua_owner_t *own,
|
|
|
|
nua_dialog_state_t *ds,
|
|
|
|
sip_t const *sip)
|
|
|
|
{
|
|
|
|
nua_remote_t *nr = ds->ds_remote_ua;
|
|
|
|
nua_dialog_usage_t *du;
|
|
|
|
nua_remote_t old[1];
|
|
|
|
|
|
|
|
*old = *nr;
|
|
|
|
|
|
|
|
if (sip && sip->sip_status &&
|
|
|
|
sip->sip_status->st_status >= 300 &&
|
|
|
|
sip->sip_status->st_status <= 399)
|
|
|
|
sip = NULL; /* Redirected */
|
|
|
|
|
|
|
|
if (sip == NULL) {
|
|
|
|
nr->nr_allow = NULL, su_free(own, old->nr_allow);
|
|
|
|
nr->nr_accept = NULL, su_free(own, old->nr_accept);
|
|
|
|
nr->nr_require = NULL, su_free(own, old->nr_require);
|
|
|
|
nr->nr_supported = NULL, su_free(own, old->nr_supported);
|
|
|
|
nr->nr_user_agent = NULL, su_free(own, old->nr_user_agent);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sip->sip_allow) {
|
|
|
|
nr->nr_allow = sip_allow_dup(own, sip->sip_allow);
|
|
|
|
su_free(own, old->nr_allow);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sip->sip_accept) {
|
|
|
|
nr->nr_accept = sip_accept_dup(own, sip->sip_accept);
|
|
|
|
su_free(own, old->nr_accept);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sip->sip_require) {
|
|
|
|
nr->nr_require = sip_require_dup(own, sip->sip_require);
|
|
|
|
su_free(own, old->nr_require);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sip->sip_supported) {
|
|
|
|
nr->nr_supported = sip_supported_dup(own, sip->sip_supported);
|
|
|
|
su_free(own, old->nr_supported);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sip->sip_user_agent) {
|
|
|
|
nr->nr_user_agent = sip_user_agent_dup(own, sip->sip_user_agent);
|
|
|
|
su_free(own, old->nr_user_agent);
|
|
|
|
}
|
|
|
|
else if (sip->sip_server) {
|
|
|
|
nr->nr_user_agent = sip_user_agent_dup(own, sip->sip_server);
|
|
|
|
su_free(own, old->nr_user_agent);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (du = ds->ds_usage; du; du = du->du_next) {
|
|
|
|
if (du->du_class->usage_peer_info)
|
|
|
|
du->du_class->usage_peer_info(du, ds, sip);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Remove dialog (if there is no other usages). */
|
|
|
|
int nua_dialog_remove(nua_owner_t *own,
|
|
|
|
nua_dialog_state_t *ds,
|
|
|
|
nua_dialog_usage_t *usage)
|
|
|
|
{
|
|
|
|
if (ds->ds_usage == usage && (usage == NULL || usage->du_next == NULL)) {
|
|
|
|
nua_dialog_store_peer_info(own, ds, NULL); /* zap peer info */
|
|
|
|
nta_leg_destroy(ds->ds_leg), ds->ds_leg = NULL;
|
|
|
|
su_free(own, (void *)ds->ds_remote_tag), ds->ds_remote_tag = NULL;
|
|
|
|
ds->ds_route = 0;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** @internal Get dialog usage slot. */
|
|
|
|
nua_dialog_usage_t **
|
|
|
|
nua_dialog_usage_at(nua_dialog_state_t const *ds,
|
|
|
|
nua_usage_class const *kind,
|
|
|
|
sip_event_t const *event)
|
|
|
|
{
|
|
|
|
static nua_dialog_usage_t *none = NULL;
|
|
|
|
|
|
|
|
if (ds) {
|
|
|
|
nua_dialog_usage_t *du, * const * prev;
|
|
|
|
sip_event_t const *o;
|
|
|
|
|
|
|
|
for (prev = &ds->ds_usage; (du = *prev); prev = &du->du_next) {
|
|
|
|
if (du->du_class != kind)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (event == NONE)
|
|
|
|
return (nua_dialog_usage_t **)prev;
|
|
|
|
|
|
|
|
o = du->du_event;
|
|
|
|
|
|
|
|
if (!event && !o)
|
|
|
|
return (nua_dialog_usage_t **)prev;
|
|
|
|
|
|
|
|
if (event != o) {
|
|
|
|
if (event == NULL || o == NULL)
|
|
|
|
continue;
|
|
|
|
if (strcmp(event->o_type, o->o_type))
|
|
|
|
continue;
|
|
|
|
if (str0casecmp(event->o_id, o->o_id)) {
|
|
|
|
if (event->o_id || strcmp(event->o_type, "refer"))
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return (nua_dialog_usage_t **)prev;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return &none;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @internal Get a dialog usage */
|
|
|
|
nua_dialog_usage_t *nua_dialog_usage_get(nua_dialog_state_t const *ds,
|
|
|
|
nua_usage_class const *kind,
|
|
|
|
sip_event_t const *event)
|
|
|
|
{
|
|
|
|
return *nua_dialog_usage_at(ds, kind, event);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @internal Get dialog usage name */
|
|
|
|
char const *nua_dialog_usage_name(nua_dialog_usage_t const *du)
|
|
|
|
{
|
|
|
|
if (du == NULL)
|
|
|
|
return "<NULL>";
|
|
|
|
return du->du_class->usage_name(du);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @internal Add dialog usage */
|
|
|
|
nua_dialog_usage_t *nua_dialog_usage_add(nua_owner_t *own,
|
|
|
|
struct nua_dialog_state *ds,
|
|
|
|
nua_usage_class const *uclass,
|
|
|
|
sip_event_t const *event)
|
|
|
|
{
|
|
|
|
if (ds) {
|
|
|
|
sip_event_t *o;
|
|
|
|
nua_dialog_usage_t *du, **prev_du;
|
|
|
|
|
|
|
|
prev_du = nua_dialog_usage_at(ds, uclass, event);
|
|
|
|
du = *prev_du;
|
|
|
|
if (du) { /* Already exists */
|
|
|
|
SU_DEBUG_5(("nua(%p): adding already existing %s usage%s%s\n",
|
2007-02-17 06:25:21 +00:00
|
|
|
own, nua_dialog_usage_name(du),
|
2006-12-21 06:30:28 +00:00
|
|
|
event ? " with event " : "", event ? event->o_type : ""));
|
|
|
|
|
|
|
|
if (prev_du != &ds->ds_usage) {
|
|
|
|
/* Move as a first usage in the list */
|
|
|
|
*prev_du = du->du_next;
|
|
|
|
du->du_next = ds->ds_usage;
|
|
|
|
ds->ds_usage = du;
|
|
|
|
}
|
|
|
|
return du;
|
|
|
|
}
|
|
|
|
|
|
|
|
o = event ? sip_event_dup(own, event) : NULL;
|
|
|
|
|
|
|
|
if (o != NULL || event == NULL)
|
|
|
|
du = su_zalloc(own, sizeof *du + uclass->usage_size);
|
|
|
|
|
|
|
|
if (du) {
|
|
|
|
du->du_class = uclass;
|
|
|
|
du->du_event = o;
|
|
|
|
|
|
|
|
if (uclass->usage_add(own, ds, du) < 0) {
|
|
|
|
su_free(own, o);
|
|
|
|
su_free(own, du);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
SU_DEBUG_5(("nua(%p): adding %s usage%s%s\n",
|
2007-02-17 06:25:21 +00:00
|
|
|
own, nua_dialog_usage_name(du),
|
2006-12-21 06:30:28 +00:00
|
|
|
o ? " with event " : "", o ? o->o_type :""));
|
|
|
|
|
|
|
|
su_home_ref(own);
|
|
|
|
du->du_next = ds->ds_usage, ds->ds_usage = du;
|
|
|
|
|
|
|
|
return du;
|
|
|
|
}
|
|
|
|
|
|
|
|
su_free(own, o);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @internal Remove dialog usage. */
|
|
|
|
void nua_dialog_usage_remove(nua_owner_t *own,
|
|
|
|
nua_dialog_state_t *ds,
|
|
|
|
nua_dialog_usage_t *du)
|
|
|
|
{
|
|
|
|
nua_dialog_usage_t **at;
|
|
|
|
|
|
|
|
assert(own); assert(ds); assert(du);
|
|
|
|
|
|
|
|
for (at = &ds->ds_usage; *at; at = &(*at)->du_next)
|
|
|
|
if (du == *at)
|
|
|
|
break;
|
|
|
|
|
|
|
|
assert(*at);
|
|
|
|
|
|
|
|
nua_dialog_usage_remove_at(own, ds, at);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @internal Remove dialog usage.
|
|
|
|
*
|
|
|
|
* Zap dialog state (leg, tag and route) if no usages remain.
|
|
|
|
*/
|
|
|
|
static
|
|
|
|
void nua_dialog_usage_remove_at(nua_owner_t *own,
|
|
|
|
nua_dialog_state_t *ds,
|
|
|
|
nua_dialog_usage_t **at)
|
|
|
|
{
|
|
|
|
if (*at) {
|
|
|
|
nua_dialog_usage_t *du = *at;
|
|
|
|
sip_event_t const *o = NULL;
|
|
|
|
nua_client_request_t *cr, *cr_next;
|
|
|
|
nua_server_request_t *sr, *sr_next;
|
|
|
|
|
|
|
|
for (cr = ds->ds_cr; cr; cr = cr_next) {
|
|
|
|
cr_next = cr->cr_next;
|
|
|
|
if (cr->cr_usage == du)
|
|
|
|
cr->cr_usage = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (sr = ds->ds_sr; sr; sr = sr_next) {
|
|
|
|
sr_next = sr->sr_next;
|
|
|
|
if (sr->sr_usage == du)
|
|
|
|
nua_server_request_destroy(sr);
|
|
|
|
}
|
|
|
|
|
2007-02-17 06:25:21 +00:00
|
|
|
*at = du->du_next;
|
|
|
|
|
|
|
|
o = du->du_event;
|
|
|
|
|
|
|
|
SU_DEBUG_5(("nua(%p): removing %s usage%s%s\n",
|
|
|
|
own, nua_dialog_usage_name(du),
|
|
|
|
o ? " with event " : "", o ? o->o_type :""));
|
|
|
|
du->du_class->usage_remove(own, ds, du);
|
|
|
|
msg_destroy(du->du_msg), du->du_msg = NULL;
|
2006-12-21 06:30:28 +00:00
|
|
|
su_home_unref(own);
|
|
|
|
su_free(own, du);
|
|
|
|
}
|
|
|
|
|
2007-02-17 06:25:21 +00:00
|
|
|
/* Zap dialog if there is no more usages */
|
|
|
|
if (ds->ds_usage == NULL) {
|
|
|
|
nta_leg_destroy(ds->ds_leg), ds->ds_leg = NULL;
|
|
|
|
su_free(own, (void *)ds->ds_remote_tag), ds->ds_remote_tag = NULL;
|
|
|
|
ds->ds_route = 0;
|
2006-12-21 06:30:28 +00:00
|
|
|
ds->ds_has_events = 0;
|
|
|
|
ds->ds_terminated = 0;
|
|
|
|
return;
|
|
|
|
}
|
2007-02-17 06:25:21 +00:00
|
|
|
else if (!ds->ds_terminated) {
|
2006-12-21 06:30:28 +00:00
|
|
|
nua_dialog_log_usage(own, ds);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
void nua_dialog_log_usage(nua_owner_t *own, nua_dialog_state_t *ds)
|
|
|
|
{
|
|
|
|
nua_dialog_usage_t *du;
|
|
|
|
|
|
|
|
if (SU_LOG->log_level >= 3) {
|
|
|
|
char buffer[160];
|
|
|
|
size_t l = 0, N = sizeof buffer;
|
|
|
|
ssize_t n;
|
|
|
|
|
|
|
|
buffer[0] = '\0';
|
|
|
|
|
|
|
|
for (du = ds->ds_usage; du; du = du->du_next) {
|
|
|
|
msg_header_t const *h = (void *)du->du_event;
|
|
|
|
|
|
|
|
if (!h)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
n = sip_event_e(buffer + l, N - l, h, 0);
|
|
|
|
if (n == -1)
|
|
|
|
break;
|
|
|
|
l += (size_t)n;
|
|
|
|
if (du->du_next && l + 2 < sizeof(buffer)) {
|
|
|
|
strcpy(buffer + l, ", ");
|
|
|
|
l += 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-02-17 06:25:21 +00:00
|
|
|
SU_DEBUG_3(("nua(%p): handle with %s%s%s\n", own,
|
2006-12-21 06:30:28 +00:00
|
|
|
ds->ds_has_session ? "session and " : "",
|
|
|
|
ds->ds_has_events ? "events " : "",
|
|
|
|
buffer));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Deinitialize dialog and its usage. @internal */
|
|
|
|
void nua_dialog_deinit(nua_owner_t *own,
|
|
|
|
nua_dialog_state_t *ds)
|
|
|
|
{
|
|
|
|
while (ds->ds_usage) {
|
|
|
|
nua_dialog_usage_remove_at(own, ds, &ds->ds_usage);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** @internal Dialog has been terminated. Remove all usages. */
|
2007-02-17 06:25:21 +00:00
|
|
|
void nua_dialog_terminated(nua_owner_t *own,
|
|
|
|
struct nua_dialog_state *ds,
|
|
|
|
int status,
|
|
|
|
char const *phrase)
|
2006-12-21 06:30:28 +00:00
|
|
|
{
|
2007-02-17 06:25:21 +00:00
|
|
|
|
2006-12-21 06:30:28 +00:00
|
|
|
ds->ds_terminated = 1;
|
|
|
|
|
|
|
|
while (ds->ds_usage) {
|
|
|
|
#if 0
|
|
|
|
int call = 0;
|
|
|
|
|
|
|
|
if (ds->ds_usage->du_kind == nua_session_usage)
|
|
|
|
call = 1; /* Delay sending the event */
|
|
|
|
else
|
|
|
|
/* XXX */;
|
|
|
|
#endif
|
|
|
|
nua_dialog_usage_remove_at(own, ds, &ds->ds_usage);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**@internal
|
|
|
|
* Set expiration time.
|
|
|
|
*/
|
|
|
|
void nua_dialog_usage_set_expires(nua_dialog_usage_t *du,
|
|
|
|
unsigned delta)
|
|
|
|
{
|
|
|
|
if (delta) {
|
|
|
|
sip_time_t now = sip_now(), expires = now + delta;
|
|
|
|
if (expires < now)
|
|
|
|
expires = SIP_TIME_MAX;
|
|
|
|
du->du_expires = expires;
|
|
|
|
nua_dialog_usage_set_refresh(du, delta);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
du->du_expires = 0, du->du_refresh = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**@internal
|
|
|
|
* Set refresh value suitably.
|
|
|
|
*
|
|
|
|
* The refresh time is set either around half of the @a delta interval or,
|
|
|
|
* if @a delta is less than 5 minutes but longer than 90 seconds, 30..60
|
|
|
|
* seconds before end of interval.
|
|
|
|
*
|
|
|
|
* If @a delta is 0, the refresh time is set at the end of the world
|
|
|
|
* (maximum time, for 32-bit systems sometimes during 2036).
|
|
|
|
*/
|
|
|
|
void nua_dialog_usage_set_refresh(nua_dialog_usage_t *du, unsigned delta)
|
|
|
|
{
|
|
|
|
if (delta == 0)
|
|
|
|
du->du_refresh = 0;
|
|
|
|
else if (delta > 90 && delta < 5 * 60)
|
|
|
|
/* refresh 30..60 seconds before deadline */
|
|
|
|
nua_dialog_usage_refresh_range(du, delta - 60, delta - 30);
|
|
|
|
else {
|
|
|
|
/* By default, refresh around half time before deadline */
|
|
|
|
unsigned min = (delta + 2) / 4;
|
|
|
|
unsigned max = (delta + 2) / 4 + (delta + 1) / 2;
|
|
|
|
if (min == 0)
|
|
|
|
min = 1;
|
|
|
|
nua_dialog_usage_refresh_range(du, min, max);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**@internal Set refresh in range min..max seconds in the future. */
|
|
|
|
void nua_dialog_usage_refresh_range(nua_dialog_usage_t *du,
|
|
|
|
unsigned min, unsigned max)
|
|
|
|
{
|
|
|
|
sip_time_t now = sip_now(), target;
|
|
|
|
unsigned delta;
|
|
|
|
|
|
|
|
if (max < min)
|
|
|
|
max = min;
|
|
|
|
|
|
|
|
if (min != max)
|
|
|
|
delta = su_randint(min, max);
|
|
|
|
else
|
|
|
|
delta = min;
|
|
|
|
|
|
|
|
if (now + delta >= now)
|
|
|
|
target = now + delta;
|
|
|
|
else
|
|
|
|
target = SIP_TIME_MAX;
|
|
|
|
|
|
|
|
SU_DEBUG_7(("nua(): refresh %s after %lu seconds (in [%u..%u])\n",
|
|
|
|
nua_dialog_usage_name(du), target - now, min, max));
|
|
|
|
|
|
|
|
du->du_refresh = target;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**@internal Do not refresh. */
|
|
|
|
void nua_dialog_usage_reset_refresh(nua_dialog_usage_t *du)
|
|
|
|
{
|
2007-02-17 06:25:21 +00:00
|
|
|
du->du_refresh = 0;
|
2006-12-21 06:30:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/** @internal Refresh usage or shutdown usage if @a now is 0. */
|
|
|
|
void nua_dialog_usage_refresh(nua_owner_t *owner,
|
|
|
|
nua_dialog_state_t *ds,
|
|
|
|
nua_dialog_usage_t *du,
|
|
|
|
sip_time_t now)
|
|
|
|
{
|
|
|
|
if (du) {
|
|
|
|
du->du_refresh = 0;
|
|
|
|
|
|
|
|
if (now > 0) {
|
2007-02-17 06:25:21 +00:00
|
|
|
if (du->du_class->usage_refresh) {
|
|
|
|
du->du_class->usage_refresh(owner, ds, du, now);
|
|
|
|
return;
|
|
|
|
}
|
2006-12-21 06:30:28 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
du->du_shutdown = 1;
|
2007-02-17 06:25:21 +00:00
|
|
|
if (du->du_class->usage_shutdown) {
|
|
|
|
du->du_class->usage_shutdown(owner, ds, du);
|
|
|
|
return;
|
2007-02-07 21:59:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-02-17 06:25:21 +00:00
|
|
|
|