2 # Copyright (c) 2012 The Chromium Authors. All rights reserved.
3 # Use of this source code is governed by a BSD-style license that can be
4 # found in the LICENSE file.
6 """Sets environment variables needed to run a chromium unit test."""
13 # This is hardcoded to be src/ relative to this script.
14 ROOT_DIR
= os
.path
.dirname(os
.path
.dirname(os
.path
.abspath(__file__
)))
16 CHROME_SANDBOX_ENV
= 'CHROME_DEVEL_SANDBOX'
17 CHROME_SANDBOX_PATH
= '/opt/chromium/chrome_sandbox'
20 def should_enable_sandbox(cmd
, sandbox_path
):
21 """Return a boolean indicating that the current slave is capable of using the
22 sandbox and should enable it. This should return True iff the slave is a
23 Linux host with the sandbox file present and configured correctly."""
24 if not (sys
.platform
.startswith('linux') and
25 os
.path
.exists(sandbox_path
)):
28 # Copy the check in tools/build/scripts/slave/runtest.py.
32 sandbox_stat
= os
.stat(sandbox_path
)
33 if ((sandbox_stat
.st_mode
& stat
.S_ISUID
) and
34 (sandbox_stat
.st_mode
& stat
.S_IRUSR
) and
35 (sandbox_stat
.st_mode
& stat
.S_IXUSR
) and
36 (sandbox_stat
.st_uid
== 0)):
41 def get_sandbox_env(cmd
, env
, verbose
=False):
42 """Checks enables the sandbox if it is required, otherwise it disables it.
43 Returns the environment flags to set."""
45 chrome_sandbox_path
= env
.get(CHROME_SANDBOX_ENV
, CHROME_SANDBOX_PATH
)
47 if should_enable_sandbox(cmd
, chrome_sandbox_path
):
49 print 'Enabling sandbox. Setting environment variable:'
50 print ' %s="%s"' % (CHROME_SANDBOX_ENV
, chrome_sandbox_path
)
51 extra_env
[CHROME_SANDBOX_ENV
] = chrome_sandbox_path
54 print 'Disabling sandbox. Setting environment variable:'
55 print ' CHROME_DEVEL_SANDBOX=""'
56 extra_env
['CHROME_DEVEL_SANDBOX'] = ''
62 """Removes internal flags from cmd since they're just used to communicate from
63 the host machine to this script running on the swarm slaves."""
64 internal_flags
= frozenset(['--asan=0', '--asan=1', '--lsan=0', '--lsan=1'])
65 return [i
for i
in cmd
if i
not in internal_flags
]
68 def fix_python_path(cmd
):
69 """Returns the fixed command line to call the right python executable."""
71 if out
[0] == 'python':
72 out
[0] = sys
.executable
73 elif out
[0].endswith('.py'):
74 out
.insert(0, sys
.executable
)
78 def get_asan_env(cmd
, lsan
):
79 """Returns the envirnoment flags needed for ASan and LSan."""
83 # Instruct GTK to use malloc while running ASan or LSan tests.
84 extra_env
['G_SLICE'] = 'always-malloc'
86 extra_env
['NSS_DISABLE_ARENA_FREE_LIST'] = '1'
87 extra_env
['NSS_DISABLE_UNLOAD'] = '1'
89 # TODO(glider): remove the symbolizer path once
90 # https://code.google.com/p/address-sanitizer/issues/detail?id=134 is fixed.
91 symbolizer_path
= os
.path
.abspath(os
.path
.join(ROOT_DIR
, 'third_party',
92 'llvm-build', 'Release+Asserts', 'bin', 'llvm-symbolizer'))
96 asan_options
.append('detect_leaks=1')
97 if sys
.platform
== 'linux2':
98 # Use the debug version of libstdc++ under LSan. If we don't, there will
99 # be a lot of incomplete stack traces in the reports.
100 extra_env
['LD_LIBRARY_PATH'] = '/usr/lib/x86_64-linux-gnu/debug:'
102 # LSan is not sandbox-compatible, so we can use online symbolization. In
103 # fact, it needs symbolization to be able to apply suppressions.
104 symbolization_options
= ['symbolize=1',
105 'external_symbolizer_path=%s' % symbolizer_path
]
107 suppressions_file
= os
.path
.join(ROOT_DIR
, 'tools', 'lsan',
109 lsan_options
= ['suppressions=%s' % suppressions_file
,
110 'print_suppressions=1']
111 extra_env
['LSAN_OPTIONS'] = ' '.join(lsan_options
)
113 # ASan uses a script for offline symbolization.
114 # Important note: when running ASan with leak detection enabled, we must use
115 # the LSan symbolization options above.
116 symbolization_options
= ['symbolize=0']
117 # Set the path to llvm-symbolizer to be used by asan_symbolize.py
118 extra_env
['LLVM_SYMBOLIZER_PATH'] = symbolizer_path
120 asan_options
.extend(symbolization_options
)
122 extra_env
['ASAN_OPTIONS'] = ' '.join(asan_options
)
124 if sys
.platform
== 'darwin':
125 isolate_output_dir
= os
.path
.abspath(os
.path
.dirname(cmd
[0]))
126 # This is needed because the test binary has @executable_path embedded in it
127 # it that the OS tries to resolve to the cache directory and not the mapped
129 extra_env
['DYLD_LIBRARY_PATH'] = str(isolate_output_dir
)
134 def run_executable(cmd
, env
):
135 """Runs an executable with:
136 - environment variable CR_SOURCE_ROOT set to the root directory.
137 - environment variable LANGUAGE to en_US.UTF-8.
138 - environment variable CHROME_DEVEL_SANDBOX set if need
139 - Reuses sys.executable automatically.
142 # Many tests assume a English interface...
143 extra_env
['LANG'] = 'en_US.UTF-8'
144 # Used by base/base_paths_linux.cc as an override. Just make sure the default
146 env
.pop('CR_SOURCE_ROOT', None)
147 extra_env
.update(get_sandbox_env(cmd
, env
))
149 # Copy logic from tools/build/scripts/slave/runtest.py.
150 asan
= '--asan=1' in cmd
151 lsan
= '--lsan=1' in cmd
154 extra_env
.update(get_asan_env(cmd
, lsan
))
156 cmd
.append('--no-sandbox')
160 # Ensure paths are correctly separated on windows.
161 cmd
[0] = cmd
[0].replace('/', os
.path
.sep
)
162 cmd
= fix_python_path(cmd
)
164 print('Additional test environment:\n%s\n'
167 (k
, v
) for k
, v
in sorted(extra_env
.iteritems())),
169 env
.update(extra_env
or {})
171 # See above comment regarding offline symbolization.
172 if asan
and not lsan
:
173 # Need to pipe to the symbolizer script.
174 p1
= subprocess
.Popen(cmd
, env
=env
, stdout
=subprocess
.PIPE
,
176 p2
= subprocess
.Popen(["../tools/valgrind/asan/asan_symbolize.py"],
177 env
=env
, stdin
=p1
.stdout
)
178 p1
.stdout
.close() # Allow p1 to receive a SIGPIPE if p2 exits.
183 return subprocess
.call(cmd
, env
=env
)
185 print >> sys
.stderr
, 'Failed to start %s' % cmd
190 return run_executable(sys
.argv
[1:], os
.environ
.copy())
193 if __name__
== '__main__':