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
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
|
#!/usr/bin/python
# Copyright (c) 2006-2008 The Chromium Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
# purify_test.py
'''Runs an exe through Valgrind and puts the intermediate files in a
directory.
'''
import datetime
import glob
import logging
import optparse
import os
import re
import shutil
import sys
import time
import google.path_utils
import common
import valgrind_analyze
rmtree = shutil.rmtree
class Valgrind():
TMP_DIR = "valgrind.tmp"
def __init__(self):
self._suppressions_files = []
def CreateOptionParser(self):
self._parser = optparse.OptionParser("usage: %prog [options] <program to "
"test>")
self._parser.add_option("-e", "--echo_to_stdout",
dest="echo_to_stdout", action="store_true", default=False,
help="echo purify output to standard output")
self._parser.add_option("-t", "--timeout",
dest="timeout", metavar="TIMEOUT", default=10000,
help="timeout in seconds for the run (default 10000)")
self._parser.add_option("", "--source_dir",
help="path to top of source tree for this build"
"(used to normalize source paths in baseline)")
self._parser.add_option("", "--suppressions", default=["."],
action="append",
help="path to a valgrind suppression file")
self._parser.add_option("", "--generate_suppressions", action="store_true",
default=False,
help="Skip analysis and generate suppressions")
self._parser.description = __doc__
def ParseArgv(self):
self.CreateOptionParser()
self._options, self._args = self._parser.parse_args()
self._timeout = int(self._options.timeout)
self._suppressions = self._options.suppressions
self._generate_suppressions = self._options.generate_suppressions
self._source_dir = self._options.source_dir
return True
def Setup(self):
return self.ParseArgv()
def Execute(self):
''' Execute the app to be tested after successful instrumentation.
Full execution command-line provided by subclassers via proc.'''
logging.info("starting execution...")
# note that self._args begins with the exe to be run
# TODO(erg): We probably want to get a version of valgrind that supports
# the "--track-origins" option...
proc = ["valgrind", "--smc-check=all", "--leak-check=full",
"--num-callers=30"]
# Either generate suppressions or load them.
if self._generate_suppressions:
proc += ["--gen-suppressions=all"]
else:
proc += ["--xml=yes"]
suppression_count = 0
for suppression_file in self._suppressions:
if os.path.exists(suppression_file):
suppression_count += 1
proc += ["--suppressions=%s" % suppression_file]
if not suppression_count:
logging.warning("WARNING: NOT USING SUPPRESSIONS!")
proc += ["--log-file=" + self.TMP_DIR + "/valgrind.%p"] + self._args
# If we have a valgrind.tmp directory, we failed to cleanup last time.
if os.path.exists(self.TMP_DIR):
shutil.rmtree(self.TMP_DIR)
os.mkdir(self.TMP_DIR)
common.RunSubprocess(proc, self._timeout)
# Always return true, even if running the subprocess failed. We depend on
# Analyze to determine if the run was valid. (This behaviour copied from
# the purify_test.py script.)
return True
def Analyze(self):
# Glob all the files in the "valgrind.tmp" directory
filenames = glob.glob(self.TMP_DIR + "/valgrind.*")
analyzer = valgrind_analyze.ValgrindAnalyze(self._source_dir, filenames)
analyzer.Report()
return 1
def Cleanup(self):
# Right now, we can cleanup by deleting our temporary directory. Other
# cleanup is still a TODO?
shutil.rmtree(self.TMP_DIR)
return True
def RunTestsAndAnalyze(self):
self.Execute()
if self._generate_suppressions:
logging.info("Skipping analysis to let you look at the raw output...")
return 0
retcode = self.Analyze()
if retcode:
logging.error("Analyze failed.")
return retcode
logging.info("Execution and analysis completed successfully.")
return 0
def Main(self):
'''Call this to run through the whole process: Setup, Execute, Analyze'''
start = datetime.datetime.now()
retcode = -1
if self.Setup():
retcode = self.RunTestsAndAnalyze()
# Skip cleanup on generate.
if not self._generate_suppressions:
self.Cleanup()
else:
logging.error("Setup failed")
end = datetime.datetime.now()
seconds = (end - start).seconds
hours = seconds / 3600
seconds = seconds % 3600
minutes = seconds / 60
seconds = seconds % 60
logging.info("elapsed time: %02d:%02d:%02d" % (hours, minutes, seconds))
return retcode
if __name__ == "__main__":
valgrind = Valgrind()
retcode = valgrind.Main()
sys.exit(retcode)
|