3 # ***** BEGIN LICENSE BLOCK *****
4 # Version: MPL 1.1/GPL 2.0/LGPL 2.1
6 # The contents of this file are subject to the Mozilla Public License Version
7 # 1.1 (the "License"); you may not use this file except in compliance with
8 # the License. You may obtain a copy of the License at
9 # http://www.mozilla.org/MPL/
11 # Software distributed under the License is distributed on an "AS IS" basis,
12 # WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
13 # for the specific language governing rights and limitations under the
16 # The Original Code is standalone Firefox Windows performance test.
18 # The Initial Developer of the Original Code is Google Inc.
19 # Portions created by the Initial Developer are Copyright (C) 2006
20 # the Initial Developer. All Rights Reserved.
23 # Annie Sullivan <annie.sullivan@gmail.com> (original author)
24 # Ben Hearsum <bhearsum@wittydomain.com> (ported to linux)
25 # Zach Lipton <zach@zachlipton.com> (Mac port)
27 # Alternatively, the contents of this file may be used under the terms of
28 # either the GNU General Public License Version 2 or later (the "GPL"), or
29 # the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
30 # in which case the provisions of the GPL or the LGPL are applicable instead
31 # of those above. If you wish to allow use of your version of this file only
32 # under the terms of either the GPL or the LGPL, and not to allow others to
33 # use your version of this file under the terms of the MPL, indicate your
34 # decision by deleting the provisions above and replace them with the notice
35 # and other provisions required by the GPL or the LGPL. If you do not delete
36 # the provisions above, a recipient may use your version of this file under
37 # the terms of any one of the MPL, the GPL or the LGPL.
39 # ***** END LICENSE BLOCK *****
41 """A set of functions to run the Tp test.
43 The Tp test measures page load times in Firefox. It does this with a
44 JavaScript script that opens a new window and cycles through page loads
45 from the local disk, timing each one. The script dumps the sum of the
46 mean times to open each page, and the standard deviation, to standard out.
47 We can also measure performance attributes during the test. See below for
51 __author__
= 'annie.sullivan@gmail.com (Annie Sullivan)'
61 def GetProcessData(pid
):
62 """Runs a ps on the process identified by pid and returns the output line
63 as a list (pid, vsz, rss)
65 command
= ['ps -o pid,vsize,rss -p'+str(pid
)]
66 handle
= subprocess
.Popen(command
, stdout
=subprocess
.PIPE
, universal_newlines
=True, shell
=True)
68 data
= handle
.stdout
.readlines()
70 # First line is header output should look like:
75 if (line
[0] == str(pid
)):
78 def GetPrivateBytes(pid
):
79 """Calculate the amount of private, writeable memory allocated to a process.
81 psData
= GetProcessData(pid
)
85 def GetResidentSize(pid
):
86 """Retrieve the current resident memory for a given process"""
87 psData
= GetProcessData(pid
)
91 # return all zeros for now on this platform as per 7/18/07 perf meeting
95 counterDict
["Private Bytes"] = GetPrivateBytes
96 counterDict
["RSS"] = GetResidentSize
97 counterDict
["% Processor Time"] = GetCpuTime
99 class CounterManager(threading
.Thread
):
100 """This class manages the monitoring of a process with any number of
103 A counter can be any function that takes an argument of one pid and
104 returns a piece of data about that process.
105 Some examples are: CalcCPUTime, GetResidentSize, and GetPrivateBytes
110 def __init__(self
, process
, counters
=None):
112 counters: A list of counters to monitor. Any counters whose name does
113 not match a key in 'counterDict' will be ignored.
115 self
.allCounters
= {}
116 self
.registeredCounters
= {}
117 self
.process
= process
118 self
.runThread
= False
122 self
.registerCounters(counters
)
124 threading
.Thread
.__init
__(self
)
126 def _loadCounters(self
):
127 """Loads all of the counters defined in the counterDict"""
128 for counter
in counterDict
.keys():
129 self
.allCounters
[counter
] = counterDict
[counter
]
131 def registerCounters(self
, counters
):
132 """Registers a list of counters that will be monitoring.
133 Only counters whose names are found in allCounters will be added
135 for counter
in counters
:
136 if counter
in self
.allCounters
:
137 self
.registeredCounters
[counter
] = \
138 [self
.allCounters
[counter
], []]
140 def unregisterCounters(self
, counters
):
141 """Unregister a list of counters.
142 Only counters whose names are found in registeredCounters will be
145 for counter
in counters
:
146 if counter
in self
.registeredCounters
:
147 del self
.registeredCounters
[counter
]
149 def getRegisteredCounters(self
):
150 """Returns a list of the registered counters."""
151 return keys(self
.registeredCounters
)
153 def getCounterValue(self
, counterName
):
154 """Returns the last value of the counter 'counterName'"""
156 if counterName
is "% Processor Time":
157 return self
._getCounterAverage
(counterName
)
159 return self
.registeredCounters
[counterName
][1][-1]
163 def _getCounterAverage(self
, counterName
):
164 """Returns the average value of the counter 'counterName'"""
167 for v
in self
.registeredCounters
[counterName
][1]:
169 return total
/ len(self
.registeredCounters
[counterName
][1])
173 def getProcess(self
):
174 """Returns the process currently associated with this CounterManager"""
177 def startMonitor(self
):
178 """Starts the monitoring process.
179 Throws an exception if any error occurs
181 # TODO: make this function less ugly
183 # the last process is the useful one
184 self
.pid
= ffprocess
.GetPidsByName(self
.process
)[-1]
185 self
.runThread
= True
188 print 'WARNING: problem starting counter monitor'
190 def stopMonitor(self
):
191 """Stops the monitor"""
192 # TODO: should probably wait until we know run() is completely stopped
193 # before setting self.pid to None. Use a lock?
194 self
.runThread
= False
197 """Performs the actual monitoring of the process. Will keep running
198 until stopMonitor() is called
200 while self
.runThread
:
201 for counter
in self
.registeredCounters
.keys():
202 # counter[0] is a function that gets the current value for
204 # counter[1] is a list of recorded values
206 self
.registeredCounters
[counter
][1].append(
207 self
.registeredCounters
[counter
][0](self
.pid
))
209 # if a counter throws an exception, remove it
210 self
.unregisterCounters([counter
])
212 time
.sleep(self
.pollInterval
)