Back to index

lightning-sunbird  0.9+nobinonly
Classes | Defines | Typedefs | Functions | Variables
switch.c File Reference
#include "prinit.h"
#include "prcvar.h"
#include "prmem.h"
#include "prinrval.h"
#include "prlock.h"
#include "prlog.h"
#include "prthread.h"
#include "prprf.h"
#include "plerror.h"
#include "plgetopt.h"
#include "private/pprio.h"
#include <stdlib.h>

Go to the source code of this file.

Classes

struct  Shared

Defines

#define INNER_LOOPS   100
#define DEFAULT_LOOPS   100
#define DEFAULT_THREADS   10

Typedefs

typedef struct Shared Shared

Functions

static void Help (void)
static void PR_CALLBACK Notified (void *arg)
PRIntn PR_CALLBACK Switch (PRIntn argc, char **argv)
PRIntn main (PRIntn argc, char **argv)

Variables

static PRFileDescdebug_out = NULL
static PRBool debug_mode = PR_FALSE
static PRBool verbosity = PR_FALSE
static PRBool failed = PR_FALSE
static Shared home

Define Documentation

#define DEFAULT_LOOPS   100

Definition at line 65 of file switch.c.

Definition at line 66 of file switch.c.

#define INNER_LOOPS   100

Definition at line 64 of file switch.c.


Typedef Documentation

typedef struct Shared Shared

Function Documentation

static void Help ( void  ) [static]

Definition at line 80 of file switch.c.

{
    debug_out = PR_STDOUT;

    PR_fprintf(
              debug_out, "Usage: >./switch [-c n] [-t n] [-d] [-v] [-G] [-C n]\n");
    PR_fprintf(
              debug_out, "-c n\tloops at thread level (default: %d)\n", DEFAULT_LOOPS);
    PR_fprintf(
              debug_out, "-t n\tnumber of threads (default: %d)\n", DEFAULT_THREADS);
    PR_fprintf(debug_out, "-d\tturn on debugging output (default: FALSE)\n");
    PR_fprintf(debug_out, "-v\tturn on verbose output (default: FALSE)\n");
    PR_fprintf(debug_out, "-G\tglobal threads only (default: FALSE)\n");
    PR_fprintf(debug_out, "-C n\tconcurrency setting (default: 1)\n");
}  /* Help */
PRIntn main ( PRIntn  argc,
char **  argv 
)

Definition at line 266 of file switch.c.

{
    PRIntn result;
    PR_STDIO_INIT();
    result = PR_Initialize(Switch, argc, argv, 0);
    return result;
}  /* main */

Here is the call graph for this function:

static void PR_CALLBACK Notified ( void arg) [static]

Definition at line 96 of file switch.c.

{
    Shared *shared = (Shared*)arg;
    PRStatus status = PR_SUCCESS;
    while (PR_SUCCESS == status)
    {
        PR_Lock(shared->ml);
        while (shared->twiddle && (PR_SUCCESS == status))
            status = PR_WaitCondVar(shared->cv, PR_INTERVAL_NO_TIMEOUT);
              if (verbosity) PR_fprintf(debug_out, "+");
        shared->twiddle = PR_TRUE;
        shared->next->twiddle = PR_FALSE;
        PR_NotifyCondVar(shared->next->cv);
        PR_Unlock(shared->ml);
    }
}  /* Notified */

Here is the call graph for this function:

Here is the caller graph for this function:

PRIntn PR_CALLBACK Switch ( PRIntn  argc,
char **  argv 
)

Definition at line 114 of file switch.c.

{
       PLOptStatus os;
    PRStatus status;
    PRBool help = PR_FALSE;
    PRUintn concurrency = 1;
    Shared *shared, *link;
    PRIntervalTime timein, timeout;
    PRThreadScope thread_scope = PR_LOCAL_THREAD;
    PRUintn thread_count, inner_count, loop_count, average;
    PRUintn thread_limit = DEFAULT_THREADS, loop_limit = DEFAULT_LOOPS;
       PLOptState *opt = PL_CreateOptState(argc, argv, "hdvc:t:C:G");
       while (PL_OPT_EOL != (os = PL_GetNextOpt(opt)))
    {
              if (PL_OPT_BAD == os) continue;
        switch (opt->option)
        {
        case 'v':  /* verbose mode */
                     verbosity = PR_TRUE;
        case 'd':  /* debug mode */
                     debug_mode = PR_TRUE;
            break;
        case 'c':  /* loop counter */
                     loop_limit = atoi(opt->value);
            break;
        case 't':  /* thread limit */
                     thread_limit = atoi(opt->value);
            break;
        case 'C':  /* Concurrency limit */
                     concurrency = atoi(opt->value);
            break;
        case 'G':  /* global threads only */
                     thread_scope = PR_GLOBAL_THREAD;
            break;
        case 'h':  /* help message */
                     Help();
                     help = PR_TRUE;
            break;
         default:
            break;
        }
    }
       PL_DestroyOptState(opt);

    if (help) return -1;

       if (PR_TRUE == debug_mode)
       {
              debug_out = PR_STDOUT;
              PR_fprintf(debug_out, "Test parameters\n");
              PR_fprintf(debug_out, "\tThreads involved: %d\n", thread_limit);
              PR_fprintf(debug_out, "\tIteration limit: %d\n", loop_limit);
              PR_fprintf(debug_out, "\tConcurrency: %d\n", concurrency);
              PR_fprintf(
                     debug_out, "\tThread type: %s\n",
                     (PR_GLOBAL_THREAD == thread_scope) ? "GLOBAL" : "LOCAL");
       }

    PR_SetConcurrency(concurrency);

    link = &home;
    home.ml = PR_NewLock();
    home.cv = PR_NewCondVar(home.ml);
    home.twiddle = PR_FALSE;
    home.next = NULL;

    timeout = 0;

    for (thread_count = 1; thread_count <= thread_limit; ++thread_count)
    {
        shared = PR_NEWZAP(Shared);

        shared->ml = home.ml;
        shared->cv = PR_NewCondVar(home.ml);
        shared->twiddle = PR_TRUE;
        shared->next = link;
        link = shared;

        shared->thread = PR_CreateThread(
            PR_USER_THREAD, Notified, shared,
            PR_PRIORITY_HIGH, thread_scope,
            PR_JOINABLE_THREAD, 0);
        PR_ASSERT(shared->thread != NULL);
        if (NULL == shared->thread)
            failed = PR_TRUE;
       }

    for (loop_count = 1; loop_count <= loop_limit; ++loop_count)
    {
              timein = PR_IntervalNow();
              for (inner_count = 0; inner_count < INNER_LOOPS; ++inner_count)
              {
                     PR_Lock(home.ml);
                     home.twiddle = PR_TRUE;
                     shared->twiddle = PR_FALSE;
                     PR_NotifyCondVar(shared->cv);
                     while (home.twiddle)
            {
                            status = PR_WaitCondVar(home.cv, PR_INTERVAL_NO_TIMEOUT);
                            if (PR_FAILURE == status)
                                failed = PR_TRUE;
            }
                     PR_Unlock(home.ml);
              }
              timeout += (PR_IntervalNow() - timein);
       }

       if (debug_mode)
       {
              average = PR_IntervalToMicroseconds(timeout)
                     / (INNER_LOOPS * loop_limit * thread_count);
              PR_fprintf(
                     debug_out, "Average switch times %d usecs for %d threads\n",
            average, thread_limit);
       }

    link = shared;
    for (thread_count = 1; thread_count <= thread_limit; ++thread_count)
    {
        if (&home == link) break;
        status = PR_Interrupt(link->thread);
              if (PR_SUCCESS != status)
        {
            failed = PR_TRUE;
            if (debug_mode)
                         PL_FPrintError(debug_out, "Failed to interrupt");
        }
              link = link->next; 
    }

    for (thread_count = 1; thread_count <= thread_limit; ++thread_count)
    {
        link = shared->next;
        status = PR_JoinThread(shared->thread);
              if (PR_SUCCESS != status)
              {
            failed = PR_TRUE;
            if (debug_mode)
                         PL_FPrintError(debug_out, "Failed to join");
        }
        PR_DestroyCondVar(shared->cv);
        PR_DELETE(shared);
        if (&home == link) break;
        shared = link;
    }
    PR_DestroyCondVar(home.cv);
    PR_DestroyLock(home.ml);

    PR_fprintf(PR_STDOUT, ((failed) ? "FAILED\n" : "PASSED\n"));
    return ((failed) ? 1 : 0);
}  /* Switch */

Here is the call graph for this function:

Here is the caller graph for this function:


Variable Documentation

Definition at line 69 of file switch.c.

PRFileDesc* debug_out = NULL [static]

Definition at line 68 of file switch.c.

PRBool failed = PR_FALSE [static]

Definition at line 69 of file switch.c.

Shared home [static]

Definition at line 113 of file switch.c.

Definition at line 69 of file switch.c.