⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 alarm.c

📁 Netscape NSPR库源码
💻 C
字号:
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- *//*  * 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 Netscape are  * Copyright (C) 1998-2000 Netscape Communications Corporation.  All * Rights Reserved. *  * Contributor(s): *  * Alternatively, the contents of this file may be used under the * terms of the GNU General Public License Version 2 or later (the * "GPL"), in which case the provisions of the GPL are applicable  * instead of those above.  If you wish to allow use of your  * version of this file only under the terms of the GPL and not to * allow others to use your version of this file under the MPL, * indicate your decision by deleting the provisions above and * replace them with the notice and other provisions required by * the GPL.  If you do not delete the provisions above, a recipient * may use your version of this file under either the MPL or the * GPL. *//*************************************************************************  1996 - Netscape Communications Corporation**** Name: alarmtst.c**** Description: Test alarms**** Modification History:** 13-May-97 AGarcia- Converted the test to accomodate the debug_mode flag.**	         The debug mode will print all of the printfs associated with this test.**			 The regress mode will be the default mode. Since the regress tool limits**           the output to a one line status:PASS or FAIL,all of the printf statements**			 have been handled with an if (debug_mode) statement.** 04-June-97 AGarcia removed the Test_Result function. Regress tool has been updated to**			recognize the return code from tha main program.***********************************************************************//************************************************************************* Includes***********************************************************************/#include "prlog.h"#include "prinit.h"#ifdef XP_MAC#include "pralarm.h"#else#include "obsolete/pralarm.h"#endif#include "prlock.h"#include "prlong.h"#include "prcvar.h"#include "prinrval.h"#include "prtime.h"/* Used to get the command line option */#include "plgetopt.h"#include <stdio.h>#include <stdlib.h>#if defined(XP_UNIX)#include <sys/time.h>#endif#ifdef XP_MAC#include "prlog.h"#define printf PR_LogPrintextern void SetupMacPrintfLog(char *logFile);#endifstatic PRIntn debug_mode;static PRIntn failed_already=0;static PRThreadScope thread_scope = PR_LOCAL_THREAD;typedef struct notifyData {    PRLock *ml;    PRCondVar *child;    PRCondVar *parent;    PRBool pending;    PRUint32 counter;} NotifyData;static void Notifier(void *arg){    NotifyData *notifyData = (NotifyData*)arg;    PR_Lock(notifyData->ml);    while (notifyData->counter > 0)    {        while (!notifyData->pending)            PR_WaitCondVar(notifyData->child, PR_INTERVAL_NO_TIMEOUT);        notifyData->counter -= 1;        notifyData->pending = PR_FALSE;        PR_NotifyCondVar(notifyData->parent);    }    PR_Unlock(notifyData->ml);}  /* Notifier *//************************************************************************* PRIVATE FUNCTION:    ConditionNotify** DESCRIPTION:** ** INPUTS:      loops** OUTPUTS:     None** RETURN:      overhead** SIDE EFFECTS:**      ** RESTRICTIONS:**      None** MEMORY:      NA** ALGORITHM:**      ***********************************************************************/static PRIntervalTime ConditionNotify(PRUint32 loops){    PRThread *thread;    NotifyData notifyData;    PRIntervalTime timein, overhead;        timein = PR_IntervalNow();    notifyData.counter = loops;    notifyData.ml = PR_NewLock();    notifyData.child = PR_NewCondVar(notifyData.ml);    notifyData.parent = PR_NewCondVar(notifyData.ml);    thread = PR_CreateThread(        PR_USER_THREAD, Notifier, &notifyData,        PR_GetThreadPriority(PR_GetCurrentThread()),        thread_scope, PR_JOINABLE_THREAD, 0);    overhead = PR_IntervalNow() - timein;  /* elapsed so far */    PR_Lock(notifyData.ml);    while (notifyData.counter > 0)    {        notifyData.pending = PR_TRUE;        PR_NotifyCondVar(notifyData.child);        while (notifyData.pending)            PR_WaitCondVar(notifyData.parent, PR_INTERVAL_NO_TIMEOUT);    }    PR_Unlock(notifyData.ml);    timein = PR_IntervalNow();    (void)PR_JoinThread(thread);    PR_DestroyCondVar(notifyData.child);    PR_DestroyCondVar(notifyData.parent);    PR_DestroyLock(notifyData.ml);        overhead += (PR_IntervalNow() - timein);  /* more overhead */    return overhead;}  /* ConditionNotify */static PRIntervalTime ConditionTimeout(PRUint32 loops){    PRUintn count;    PRIntervalTime overhead, timein = PR_IntervalNow();    PRLock *ml = PR_NewLock();    PRCondVar *cv = PR_NewCondVar(ml);    PRIntervalTime interval = PR_MillisecondsToInterval(50);    overhead = PR_IntervalNow() - timein;    PR_Lock(ml);    for (count = 0; count < loops; ++count)    {        overhead += interval;        PR_ASSERT(PR_WaitCondVar(cv, interval) == PR_SUCCESS);    }    PR_Unlock(ml);    timein = PR_IntervalNow();    PR_DestroyCondVar(cv);    PR_DestroyLock(ml);    overhead += (PR_IntervalNow() - timein);    return overhead;}  /* ConditionTimeout */typedef struct AlarmData {    PRLock *ml;    PRCondVar *cv;    PRUint32 rate, late, times;    PRIntervalTime duration, timein, period;} AlarmData;static PRBool AlarmFn1(PRAlarmID *id, void *clientData, PRUint32 late){    PRStatus rv = PR_SUCCESS;    PRBool keepGoing, resetAlarm;    PRIntervalTime interval, now = PR_IntervalNow();    AlarmData *ad = (AlarmData*)clientData;    PR_Lock(ad->ml);    ad->late += late;    ad->times += 1;    keepGoing = ((PRIntervalTime)(now - ad->timein) < ad->duration) ?        PR_TRUE : PR_FALSE;    if (!keepGoing)        rv = PR_NotifyCondVar(ad->cv);    resetAlarm = ((ad->times % 31) == 0) ? PR_TRUE : PR_FALSE;                                             interval = (ad->period + ad->rate - 1) / ad->rate;    if (!late && (interval > 10))    {        interval &= (now & 0x03) + 1;        PR_WaitCondVar(ad->cv, interval);    }              PR_Unlock(ad->ml);    if (rv != PR_SUCCESS)    {		if (!debug_mode) failed_already=1;		else		 printf("AlarmFn: notify status: FAIL\n");			}    if (resetAlarm)    {           ad->rate += 3;        ad->late = ad->times = 0;        if (PR_ResetAlarm(id, ad->period, ad->rate) != PR_SUCCESS)        {			if (!debug_mode)				failed_already=1;			else						printf("AlarmFn: Resetting alarm status: FAIL\n");            keepGoing = PR_FALSE;        }    }    return keepGoing;}  /* AlarmFn1 */static PRIntervalTime Alarms1(PRUint32 loops){    PRAlarm *alarm;    AlarmData ad;    PRIntervalTime overhead, timein = PR_IntervalNow();    PRIntervalTime duration = PR_SecondsToInterval(3);    PRLock *ml = PR_NewLock();    PRCondVar *cv = PR_NewCondVar(ml);    ad.ml = ml;    ad.cv = cv;    ad.rate = 1;    ad.times = loops;    ad.late = ad.times = 0;    ad.duration = duration;    ad.timein = PR_IntervalNow();    ad.period = PR_SecondsToInterval(1);    alarm = PR_CreateAlarm();    (void)PR_SetAlarm(        alarm, ad.period, ad.rate, AlarmFn1, &ad);            overhead = PR_IntervalNow() - timein;    PR_Lock(ml);    while ((PRIntervalTime)(PR_IntervalNow() - ad.timein) < duration)        PR_WaitCondVar(cv, PR_INTERVAL_NO_TIMEOUT);    PR_Unlock(ml);    timein = PR_IntervalNow();    (void)PR_DestroyAlarm(alarm);    PR_DestroyCondVar(cv);    PR_DestroyLock(ml);    overhead += (PR_IntervalNow() - timein);        return duration + overhead;}  /* Alarms1 */static PRBool AlarmFn2(PRAlarmID *id, void *clientData, PRUint32 late){#if defined(XP_MAC)#pragma unused (id)#endif    PRBool keepGoing;    PRStatus rv = PR_SUCCESS;    AlarmData *ad = (AlarmData*)clientData;    PRIntervalTime interval, now = PR_IntervalNow();    PR_Lock(ad->ml);    ad->times += 1;    keepGoing = ((PRIntervalTime)(now - ad->timein) < ad->duration) ?        PR_TRUE : PR_FALSE;    interval = (ad->period + ad->rate - 1) / ad->rate;    if (!late && (interval > 10))    {        interval &= (now & 0x03) + 1;        PR_WaitCondVar(ad->cv, interval);    }    if (!keepGoing) rv = PR_NotifyCondVar(ad->cv);    PR_Unlock(ad->ml);    if (rv != PR_SUCCESS)		failed_already=1;;    return keepGoing;}  /* AlarmFn2 */static PRIntervalTime Alarms2(PRUint32 loops){    PRStatus rv;    PRAlarm *alarm;    PRIntervalTime overhead, timein = PR_IntervalNow();    AlarmData ad;    PRIntervalTime duration = PR_SecondsToInterval(30);    PRLock *ml = PR_NewLock();    PRCondVar *cv = PR_NewCondVar(ml);    ad.ml = ml;    ad.cv = cv;    ad.rate = 1;    ad.times = loops;    ad.late = ad.times = 0;    ad.duration = duration;    ad.timein = PR_IntervalNow();    ad.period = PR_SecondsToInterval(1);    alarm = PR_CreateAlarm();    (void)PR_SetAlarm(        alarm, ad.period, ad.rate, AlarmFn2, &ad);            overhead = PR_IntervalNow() - timein;    PR_Lock(ml);    while ((PRIntervalTime)(PR_IntervalNow() - ad.timein) < duration)        PR_WaitCondVar(cv, PR_INTERVAL_NO_TIMEOUT);    PR_Unlock(ml);        timein = PR_IntervalNow();    rv = PR_DestroyAlarm(alarm);    if (rv != PR_SUCCESS)    {		if (!debug_mode)			failed_already=1;		else				printf("***Destroying alarm status: FAIL\n");    }		    PR_DestroyCondVar(cv);    PR_DestroyLock(ml);        overhead += (PR_IntervalNow() - timein);        return duration + overhead;}  /* Alarms2 */static PRIntervalTime Alarms3(PRUint32 loops){    PRIntn i;    PRStatus rv;    PRAlarm *alarm;    AlarmData ad[3];    PRIntervalTime duration = PR_SecondsToInterval(30);    PRIntervalTime overhead, timein = PR_IntervalNow();    PRLock *ml = PR_NewLock();    PRCondVar *cv = PR_NewCondVar(ml);    for (i = 0; i < 3; ++i)    {        ad[i].ml = ml;        ad[i].cv = cv;        ad[i].rate = 1;        ad[i].times = loops;        ad[i].duration = duration;        ad[i].late = ad[i].times = 0;        ad[i].timein = PR_IntervalNow();        ad[i].period = PR_SecondsToInterval(1);        /* more loops, faster rate => same elapsed time */        ad[i].times = (i + 1) * loops;        ad[i].rate = (i + 1) * 10;    }    alarm = PR_CreateAlarm();    for (i = 0; i < 3; ++i)    {        (void)PR_SetAlarm(            alarm, ad[i].period, ad[i].rate,            AlarmFn2, &ad[i]);    }            overhead = PR_IntervalNow() - timein;    PR_Lock(ml);    for (i = 0; i < 3; ++i)    {        while ((PRIntervalTime)(PR_IntervalNow() - ad[i].timein) < duration)            PR_WaitCondVar(cv, PR_INTERVAL_NO_TIMEOUT);    }    PR_Unlock(ml);    timein = PR_IntervalNow();	if (debug_mode)	printf        ("Alarms3 finished at %u, %u, %u\n",        ad[0].timein, ad[1].timein, ad[2].timein);        rv = PR_DestroyAlarm(alarm);    if (rv != PR_SUCCESS)    {		if (!debug_mode)					failed_already=1;		else			   printf("***Destroying alarm status: FAIL\n");	}    PR_DestroyCondVar(cv);    PR_DestroyLock(ml);        overhead += (duration / 3);    overhead += (PR_IntervalNow() - timein);    return overhead;}  /* Alarms3 */static PRUint32 TimeThis(    const char *msg, PRUint32 (*func)(PRUint32 loops), PRUint32 loops){    PRUint32 overhead, usecs;    PRIntervalTime predicted, timein, timeout, ticks; if (debug_mode)    printf("Testing %s ...", msg);    timein = PR_IntervalNow();    predicted = func(loops);    timeout = PR_IntervalNow();  if (debug_mode)    printf(" done\n");    ticks = timeout - timein;    usecs = PR_IntervalToMicroseconds(ticks);    overhead = PR_IntervalToMicroseconds(predicted);    if(ticks < predicted)    {		if (debug_mode) {        printf("\tFinished in negative time\n");        printf("\tpredicted overhead was %d usecs\n", overhead);        printf("\ttest completed in %d usecs\n\n", usecs);		}    }    else    {	if (debug_mode)		        printf(            "\ttotal: %d usecs\n\toverhead: %d usecs\n\tcost: %6.3f usecs\n\n",            usecs, overhead, ((double)(usecs - overhead) / (double)loops));    }    return overhead;}  /* TimeThis */int prmain(int argc, char** argv){    PRUint32 cpu, cpus = 0, loops = 0;	/* The command line argument: -d is used to determine if the test is being run	in debug mode. The regress tool requires only one line output:PASS or FAIL.	All of the printfs associated with this test has been handled with a if (debug_mode)	test.	Usage: test_name [-d]	*/	PLOptStatus os;	PLOptState *opt = PL_CreateOptState(argc, argv, "Gdl:c:");	while (PL_OPT_EOL != (os = PL_GetNextOpt(opt)))    {		if (PL_OPT_BAD == os) continue;        switch (opt->option)        {        case 'G':  /* GLOBAL threads */			thread_scope = PR_GLOBAL_THREAD;            break;        case 'd':  /* debug mode */			debug_mode = 1;            break;        case 'l':  /* loop count */			loops = atoi(opt->value);            break;        case 'c':  /* concurrency limit */			cpus = atoi(opt->value);            break;         default:            break;        }    }	PL_DestroyOptState(opt);    if (cpus == 0) cpus = 1;    if (loops == 0) loops = 4;	if (debug_mode)		printf("Alarm: Using %d loops\n", loops);	if (debug_mode)		        printf("Alarm: Using %d cpu(s)\n", cpus);#ifdef XP_MAC	SetupMacPrintfLog("alarm.log");	debug_mode = 1;#endif    for (cpu = 1; cpu <= cpus; ++cpu)    {    if (debug_mode)        printf("\nAlarm: Using %d CPU(s)\n", cpu);	PR_SetConcurrency(cpu);                /* some basic time test */        (void)TimeThis("ConditionNotify", ConditionNotify, loops);        (void)TimeThis("ConditionTimeout", ConditionTimeout, loops);        (void)TimeThis("Alarms1", Alarms1, loops);        (void)TimeThis("Alarms2", Alarms2, loops);        (void)TimeThis("Alarms3", Alarms3, loops);    }    return 0;}int main(int argc, char** argv){     PR_Initialize(prmain, argc, argv, 0);     PR_STDIO_INIT();	 if (failed_already) return 1;	 else return 0;}  /* main *//* alarmtst.c */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -