blob: 932d077f6887b5a83cde8d18ccc75dda73557052 [file] [log] [blame] [edit]
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* ***** BEGIN LICENSE BLOCK *****
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
*
* The contents of this file are subject to the Mozilla Public License Version
* 1.1 (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
* http://www.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
* for the specific language governing rights and limitations under the
* License.
*
* The Original Code is the Netscape Portable Runtime (NSPR).
*
* The Initial Developer of the Original Code is
* Netscape Communications Corporation.
* Portions created by the Initial Developer are Copyright (C) 1998-2000
* the Initial Developer. All Rights Reserved.
*
* Contributor(s):
*
* Alternatively, the contents of this file may be used under the terms of
* either the GNU General Public License Version 2 or later (the "GPL"), or
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
* in which case the provisions of the GPL or the LGPL are applicable instead
* of those above. If you wish to allow use of your version of this file only
* under the terms of either the GPL or the LGPL, and not to allow others to
* use your version of this file under the terms of the MPL, indicate your
* decision by deleting the provisions above and replace them with the notice
* and other provisions required by the GPL or the LGPL. If you do not delete
* the provisions above, a recipient may use your version of this file under
* the terms of any one of the MPL, the GPL or the LGPL.
*
* ***** END LICENSE BLOCK ***** */
#include "primpl.h"
/**********************************************************************/
/******************************* PRALARM ******************************/
/**********************************************************************/
#include "obsolete/pralarm.h"
struct PRAlarmID { /* typedef'd in pralarm.h */
PRCList list; /* circular list linkage */
PRAlarm *alarm; /* back pointer to owning alarm */
PRPeriodicAlarmFn function; /* function to call for notify */
void *clientData; /* opaque client context */
PRIntervalTime period; /* the client defined period */
PRUint32 rate; /* rate of notification */
PRUint32 accumulator; /* keeps track of # notifies */
PRIntervalTime epoch; /* when timer was started */
PRIntervalTime nextNotify; /* when we'll next do our thing */
PRIntervalTime lastNotify; /* when we last did our thing */
};
typedef enum {alarm_active, alarm_inactive} _AlarmState;
struct PRAlarm { /* typedef'd in pralarm.h */
PRCList timers; /* base of alarm ids list */
PRLock *lock; /* lock used to protect data */
PRCondVar *cond; /* condition that used to wait */
PRThread *notifier; /* thread to deliver notifies */
PRAlarmID *current; /* current alarm being served */
_AlarmState state; /* used to delete the alarm */
};
static PRAlarmID *pr_getNextAlarm(PRAlarm *alarm, PRAlarmID *id)
{
/*
* Puts 'id' back into the sorted list iff it's not NULL.
* Removes the first element from the list and returns it (or NULL).
* List is "assumed" to be short.
*
* NB: Caller is providing locking
*/
PRCList *timer;
PRAlarmID *result = id;
PRIntervalTime now = PR_IntervalNow();
if (!PR_CLIST_IS_EMPTY(&alarm->timers))
{
if (id != NULL) /* have to put this id back in */
{
PRIntervalTime idDelta = now - id->nextNotify;
timer = alarm->timers.next;
do
{
result = (PRAlarmID*)timer;
if ((PRIntervalTime)(now - result->nextNotify) > idDelta)
{
PR_INSERT_BEFORE(&id->list, &alarm->timers);
break;
}
timer = timer->next;
} while (timer != &alarm->timers);
}
result = (PRAlarmID*)(timer = PR_LIST_HEAD(&alarm->timers));
PR_REMOVE_LINK(timer); /* remove it from the list */
}
return result;
} /* pr_getNextAlarm */
static PRIntervalTime pr_PredictNextNotifyTime(PRAlarmID *id)
{
PRIntervalTime delta;
PRFloat64 baseRate = (PRFloat64)id->period / (PRFloat64)id->rate;
PRFloat64 offsetFromEpoch = (PRFloat64)id->accumulator * baseRate;
id->accumulator += 1; /* every call advances to next period */
id->lastNotify = id->nextNotify; /* just keeping track of things */
id->nextNotify = (PRIntervalTime)(offsetFromEpoch + 0.5);
delta = id->nextNotify - id->lastNotify;
return delta;
} /* pr_PredictNextNotifyTime */
static void PR_CALLBACK pr_alarmNotifier(void *arg)
{
/*
* This is the root of the notifier thread. There is one such thread
* for each PRAlarm. It may service an arbitrary (though assumed to be
* small) number of alarms using the same thread and structure. It
* continues to run until the alarm is destroyed.
*/
PRAlarmID *id = NULL;
PRAlarm *alarm = (PRAlarm*)arg;
enum {notify, abort, scan} why = scan;
while (why != abort)
{
PRIntervalTime pause;
PR_Lock(alarm->lock);
while (why == scan)
{
alarm->current = NULL; /* reset current id */
if (alarm->state == alarm_inactive) why = abort; /* we're toast */
else if (why == scan) /* the dominant case */
{
id = pr_getNextAlarm(alarm, id); /* even if it's the same */
if (id == NULL) /* there are no alarms set */
(void)PR_WaitCondVar(alarm->cond, PR_INTERVAL_NO_TIMEOUT);
else
{
pause = id->nextNotify - (PR_IntervalNow() - id->epoch);
if ((PRInt32)pause <= 0) /* is this one's time up? */
{
why = notify; /* set up to do our thing */
alarm->current = id; /* id we're about to schedule */
}
else
(void)PR_WaitCondVar(alarm->cond, pause); /* dally */
}
}
}
PR_Unlock(alarm->lock);
if (why == notify)
{
(void)pr_PredictNextNotifyTime(id);
if (!id->function(id, id->clientData, ~pause))
{
/*
* Notified function decided not to continue. Free
* the alarm id to make sure it doesn't get back on
* the list.
*/
PR_DELETE(id); /* free notifier object */
id = NULL; /* so it doesn't get back into the list */
}
why = scan; /* so we can cycle through the loop again */
}
}
} /* pr_alarm_notifier */
PR_IMPLEMENT(PRAlarm*) PR_CreateAlarm(void)
{
PRAlarm *alarm = PR_NEWZAP(PRAlarm);
if (alarm != NULL)
{
if ((alarm->lock = PR_NewLock()) == NULL) goto done;
if ((alarm->cond = PR_NewCondVar(alarm->lock)) == NULL) goto done;
alarm->state = alarm_active;
PR_INIT_CLIST(&alarm->timers);
alarm->notifier = PR_CreateThread(
PR_USER_THREAD, pr_alarmNotifier, alarm,
PR_GetThreadPriority(PR_GetCurrentThread()),
PR_LOCAL_THREAD, PR_JOINABLE_THREAD, 0);
if (alarm->notifier == NULL) goto done;
}
return alarm;
done:
if (alarm->cond != NULL) PR_DestroyCondVar(alarm->cond);
if (alarm->lock != NULL) PR_DestroyLock(alarm->lock);
PR_DELETE(alarm);
return NULL;
} /* CreateAlarm */
PR_IMPLEMENT(PRStatus) PR_DestroyAlarm(PRAlarm *alarm)
{
PRStatus rv;
PR_Lock(alarm->lock);
alarm->state = alarm_inactive;
rv = PR_NotifyCondVar(alarm->cond);
PR_Unlock(alarm->lock);
if (rv == PR_SUCCESS)
rv = PR_JoinThread(alarm->notifier);
if (rv == PR_SUCCESS)
{
PR_DestroyCondVar(alarm->cond);
PR_DestroyLock(alarm->lock);
PR_DELETE(alarm);
}
return rv;
} /* PR_DestroyAlarm */
PR_IMPLEMENT(PRAlarmID*) PR_SetAlarm(
PRAlarm *alarm, PRIntervalTime period, PRUint32 rate,
PRPeriodicAlarmFn function, void *clientData)
{
/*
* Create a new periodic alarm an existing current structure.
* Set up the context and compute the first notify time (immediate).
* Link the new ID into the head of the list (since it's notifying
* immediately).
*/
PRAlarmID *id = PR_NEWZAP(PRAlarmID);
if (!id)
return NULL;
id->alarm = alarm;
PR_INIT_CLIST(&id->list);
id->function = function;
id->clientData = clientData;
id->period = period;
id->rate = rate;
id->epoch = id->nextNotify = PR_IntervalNow();
(void)pr_PredictNextNotifyTime(id);
PR_Lock(alarm->lock);
PR_INSERT_BEFORE(&id->list, &alarm->timers);
PR_NotifyCondVar(alarm->cond);
PR_Unlock(alarm->lock);
return id;
} /* PR_SetAlarm */
PR_IMPLEMENT(PRStatus) PR_ResetAlarm(
PRAlarmID *id, PRIntervalTime period, PRUint32 rate)
{
/*
* Can only be called from within the notify routine. Doesn't
* need locking because it can only be called from within the
* notify routine.
*/
if (id != id->alarm->current)
return PR_FAILURE;
id->period = period;
id->rate = rate;
id->accumulator = 1;
id->epoch = PR_IntervalNow();
(void)pr_PredictNextNotifyTime(id);
return PR_SUCCESS;
} /* PR_ResetAlarm */