File: main.h

package info (click to toggle)
ptlib 2.10.4~dfsg-1
  • links: PTS, VCS
  • area: main
  • in suites: wheezy
  • size: 58,836 kB
  • sloc: cpp: 135,080; ansic: 8,534; yacc: 3,059; sh: 2,776; makefile: 1,082; lex: 390
file content (140 lines) | stat: -rw-r--r-- 3,763 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
/*
 * main.h
 *
 * PWLib application header file for SyncPoints
 *
 * Copyright 2009 Derek J Smithies
 *
 * $Revision: 23418 $
 * $Author: rjongbloed $
 * $Date: 2009-09-10 19:13:48 -0500 (Thu, 10 Sep 2009) $
 */

#ifndef _SyncPoints_MAIN_H
#define _SyncPoints_MAIN_H

#include <ptlib/pprocess.h>

#include <list>
class Runner;

/*! \mainpage  SyncPoints 

The purpose of this program is to find out which is faster:
\li windows
\li linux

when dealing with waiting on SyncPoints. PTLib provides the structure,
PSyncPoint. One programming approach is to have a thread wait on a
PSyncPoint instance. When the PSyncPoint instance is signalled, the
waiting thread should be activated to do things.

This program creates a PThread derivative, that contains a PSyncPoint instance.
There are N instances of the derivative. 

The first instance signals the second. The second instance signals the
third. And so on. The time taken to move the signalling along the
queue to end is record. Since this program runs on windows and linux,
we can compare the two times. Maybe, issues in linux scheduling can be
exposed..


Results so far indicate that a linux AMD dual core box will take 6
microseconds for a thread to respond to a syncpoint signal. this time
was recorded when the box was running at 100% cpu load, with both
processors doing a compile. When the machine was idle, the average
time per syncpoint was 4.7 microseconds.

If there are far fewer syncpoints made (just 1, loops of 10000) the
time was (unloaded box) 2.2 microseconds per sync point. With this
test set, and loading the box, the time was between 2 and 35
microseconds.

*/

typedef std::list<Runner *> RunnerList;

/**This is where all the activity happens. This class is launched on
   program startup, and does timing runs */
class SyncPoints : public PProcess
{
  PCLASSINFO(SyncPoints, PProcess)

  public:
  /**Constructor */
    SyncPoints();

    /**Destructor */
    ~SyncPoints();

    /**Program execution starts here */
    void Main();

    /**When all the elements have been processed, it calls this */
    void ListFinished();
    
    /**Simple counter of the number of signalled PSyncPoint classes
       there are.  This is used as a simple verification of the
       code. 

      The count in here is higher than expected. For 200 threads, 10
       loops, the count is 2010, which is correct. The reason is there
       are 201 sync points signalled per loop - the 200 threads and 1
       (allDone) in this class. */
    PAtomicInteger syncPointsSignalled;
 protected:
    
    /**The list of Runner Elements that do all the signaling */
    RunnerList list;
    
    /**The number of times we go over all the elements */
    PINDEX loops;
    
    /**The number of elements to put in the list of Runners */
    PINDEX size;
    
    /**The terminating sync point, which indicates all the threads
       were processed */
    PSyncPoint allDone;
};




/**This class is the core of the thing. It is a derivative of PThread,
   and contains 2 things - a pointer to the next thread, and and
   PSyncPoint. */
class Runner : public PThread
{
  PCLASSINFO(Runner, PThread);

 public:
  Runner(SyncPoints & app, Runner * nextThread, PINDEX id);

  /**Where the work in this thread happens; */
  virtual void Main();

  /**Where the other thread calls to activate us */
  void RunNow();
  
 protected:

  /**reference back to the master class */
  SyncPoints & app;

  /**The next thread, which we push to life when we are pushed to life */
  Runner *nextThread;

  /**The sync point we wait on */
  PSyncPoint syncPoint;
  
  /**The numerical ID of us */
  PINDEX id;
};



#endif  // _SyncPoints_MAIN_H


// End of File ///////////////////////////////////////////////////////////////