Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(78)

Side by Side Diff: third_party/instrumented_libraries/scripts/download_build_install.py

Issue 2735013004: Instrumented libraries: improve parallel build (Closed)
Patch Set: Created 3 years, 9 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « third_party/instrumented_libraries/scripts/build_and_package.py ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 #!/usr/bin/python 1 #!/usr/bin/python
2 # Copyright 2013 The Chromium Authors. All rights reserved. 2 # Copyright 2013 The Chromium Authors. All rights reserved.
3 # Use of this source code is governed by a BSD-style license that can be 3 # Use of this source code is governed by a BSD-style license that can be
4 # found in the LICENSE file. 4 # found in the LICENSE file.
5 5
6 """Downloads, builds (with instrumentation) and installs shared libraries.""" 6 """Downloads, builds (with instrumentation) and installs shared libraries."""
7 7
8 import argparse 8 import argparse
9 import ast 9 import ast
10 import fcntl
10 import os 11 import os
11 import platform 12 import platform
12 import re 13 import re
13 import shlex 14 import shlex
14 import shutil 15 import shutil
15 import subprocess 16 import subprocess
16 import sys 17 import sys
17 18
18 SCRIPT_ABSOLUTE_PATH = os.path.dirname(os.path.abspath(__file__)) 19 SCRIPT_ABSOLUTE_PATH = os.path.dirname(os.path.abspath(__file__))
19 20
(...skipping 22 matching lines...) Expand all
42 return os.path.realpath(os.path.join(SCRIPT_ABSOLUTE_PATH, '..', 43 return os.path.realpath(os.path.join(SCRIPT_ABSOLUTE_PATH, '..',
43 path_relative_to_gyp)) 44 path_relative_to_gyp))
44 45
45 46
46 class InstrumentedPackageBuilder(object): 47 class InstrumentedPackageBuilder(object):
47 """Checks out and builds a single instrumented package.""" 48 """Checks out and builds a single instrumented package."""
48 def __init__(self, args, clobber): 49 def __init__(self, args, clobber):
49 self._cc = args.cc 50 self._cc = args.cc
50 self._cxx = args.cxx 51 self._cxx = args.cxx
51 self._extra_configure_flags = unescape_flags(args.extra_configure_flags) 52 self._extra_configure_flags = unescape_flags(args.extra_configure_flags)
52 self._jobs = args.jobs
53 self._libdir = args.libdir 53 self._libdir = args.libdir
54 self._package = args.package 54 self._package = args.package
55 self._patch = real_path(args.patch) if args.patch else None 55 self._patch = real_path(args.patch) if args.patch else None
56 self._pre_build = \ 56 self._pre_build = \
57 real_path(args.pre_build) if args.pre_build else None 57 real_path(args.pre_build) if args.pre_build else None
58 self._sanitizer = args.sanitizer 58 self._sanitizer = args.sanitizer
59 self._verbose = args.verbose 59 self._verbose = args.verbose
60 self._clobber = clobber 60 self._clobber = clobber
61 self._working_dir = os.path.join( 61 self._working_dir = os.path.join(
62 real_path(args.intermediate_dir), self._package, '') 62 real_path(args.intermediate_dir), self._package, '')
(...skipping 28 matching lines...) Expand all
91 self._build_env['LDFLAGS'] = self._ldflags 91 self._build_env['LDFLAGS'] = self._ldflags
92 92
93 if self._sanitizer == 'asan': 93 if self._sanitizer == 'asan':
94 # Do not report leaks during the build process. 94 # Do not report leaks during the build process.
95 self._build_env['ASAN_OPTIONS'] = \ 95 self._build_env['ASAN_OPTIONS'] = \
96 '%s:detect_leaks=0' % self._build_env.get('ASAN_OPTIONS', '') 96 '%s:detect_leaks=0' % self._build_env.get('ASAN_OPTIONS', '')
97 97
98 # libappindicator1 needs this. 98 # libappindicator1 needs this.
99 self._build_env['CSC'] = '/usr/bin/mono-csc' 99 self._build_env['CSC'] = '/usr/bin/mono-csc'
100 100
101 def shell_call(self, command, env=None, cwd=None): 101 def shell_call(self, command, env=None, cwd=None, ignore_ret_code=False):
102 """Wrapper around subprocess.Popen(). 102 """Wrapper around subprocess.Popen().
103 103
104 Calls command with specific environment and verbosity using 104 Calls command with specific environment and verbosity using
105 subprocess.Popen(). 105 subprocess.Popen().
106 """ 106 """
107 child = subprocess.Popen( 107 child = subprocess.Popen(
108 command, stdout=subprocess.PIPE, stderr=subprocess.STDOUT, 108 command, stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
109 env=env, shell=True, cwd=cwd) 109 env=env, shell=True, cwd=cwd)
110 stdout, stderr = child.communicate() 110 stdout, stderr = child.communicate()
111 if ignore_ret_code:
112 if self._verbose:
113 print stdout
114 return
111 if self._verbose or child.returncode: 115 if self._verbose or child.returncode:
112 print stdout 116 print stdout
113 if child.returncode: 117 if child.returncode:
114 raise Exception('Failed to run: %s' % command) 118 raise Exception('Failed to run: %s' % command)
115 119
116 def maybe_download_source(self): 120 def maybe_download_source(self):
117 """Checks out the source code (if needed). 121 """Checks out the source code (if needed).
118 122
119 Checks out the source code for the package, if required (i.e. unless running 123 Checks out the source code for the package, if required (i.e. unless running
120 in no-clobber mode). Initializes self._source_dir and self._source_archives. 124 in no-clobber mode). Initializes self._source_dir and self._source_archives.
121 """ 125 """
122 get_fresh_source = self._clobber or not os.path.exists(self._working_dir) 126 get_fresh_source = self._clobber or not os.path.exists(self._working_dir)
123 if get_fresh_source: 127 if get_fresh_source:
124 shutil.rmtree(self._working_dir, ignore_errors=True) 128 shutil.rmtree(self._working_dir, ignore_errors=True)
125 os.makedirs(self._working_dir) 129 os.makedirs(self._working_dir)
130
131 # Download one source package at a time, otherwise, there will
132 # be connection errors in gnutls_handshake().
133 self.shell_call('touch apt-source-lock')
Evgeniy Stepanov 2017/03/07 22:42:02 Why not open('apt-source-lock', 'w') ?
Tom (Use chromium acct) 2017/03/08 00:24:58 Done.
134 lock = open('apt-source-lock')
135 fcntl.flock(lock, fcntl.LOCK_EX)
126 self.shell_call('apt-get source %s' % self._package, 136 self.shell_call('apt-get source %s' % self._package,
127 cwd=self._working_dir) 137 cwd=self._working_dir)
138 fcntl.flock(lock, fcntl.LOCK_UN)
128 139
129 (dirpath, dirnames, filenames) = os.walk(self._working_dir).next() 140 (dirpath, dirnames, filenames) = os.walk(self._working_dir).next()
130 141
131 if len(dirnames) != 1: 142 if len(dirnames) != 1:
132 raise Exception( 143 raise Exception(
133 '`apt-get source %s\' must create exactly one subdirectory.' 144 '`apt-get source %s\' must create exactly one subdirectory.'
134 % self._package) 145 % self._package)
135 self._source_dir = os.path.join(dirpath, dirnames[0], '') 146 self._source_dir = os.path.join(dirpath, dirnames[0], '')
136 147
137 if len(filenames) == 0: 148 if len(filenames) == 0:
(...skipping 70 matching lines...) Expand 10 before | Expand all | Expand 10 after
208 219
209 def cleanup_after_install(self): 220 def cleanup_after_install(self):
210 """Removes unneeded files in self.temp_libdir().""" 221 """Removes unneeded files in self.temp_libdir()."""
211 # .la files are not needed, nuke them. 222 # .la files are not needed, nuke them.
212 # In case --no-static is not supported, nuke any static libraries we built. 223 # In case --no-static is not supported, nuke any static libraries we built.
213 self.shell_call( 224 self.shell_call(
214 'find %s -name *.la -or -name *.a | xargs rm -f' % self.temp_libdir()) 225 'find %s -name *.la -or -name *.a | xargs rm -f' % self.temp_libdir())
215 # .pc files are not needed. 226 # .pc files are not needed.
216 self.shell_call('rm %s/pkgconfig -rf' % self.temp_libdir()) 227 self.shell_call('rm %s/pkgconfig -rf' % self.temp_libdir())
217 228
218 def make(self, args, jobs=None, env=None, cwd=None): 229 def make(self, args, env=None, cwd=None, ignore_ret_code=False):
219 """Invokes `make'. 230 """Invokes `make'.
220 231
221 Invokes `make' with the specified args, using self._build_env and 232 Invokes `make' with the specified args, using self._build_env and
222 self._source_dir by default. 233 self._source_dir by default.
223 """ 234 """
224 if jobs is None:
225 jobs = self._jobs
226 if cwd is None: 235 if cwd is None:
227 cwd = self._source_dir 236 cwd = self._source_dir
228 if env is None: 237 if env is None:
229 env = self._build_env 238 env = self._build_env
230 cmd = ['make', '-j%s' % jobs] + args 239 cmd = ['make'] + args
231 self.shell_call(' '.join(cmd), env=env, cwd=cwd) 240 self.shell_call(' '.join(cmd), env=env, cwd=cwd,
241 ignore_ret_code=ignore_ret_code)
232 242
233 def make_install(self, args, **kwargs): 243 def make_install(self, args, **kwargs):
234 """Invokes `make install'.""" 244 """Invokes `make install'."""
235 self.make(['install'] + args, **kwargs) 245 self.make(['install'] + args, **kwargs)
236 246
237 def build_and_install(self): 247 def build_and_install(self):
238 """Builds and installs the DSOs. 248 """Builds and installs the DSOs.
239 249
240 Builds the package with ./configure + make, installs it to a temporary 250 Builds the package with ./configure + make, installs it to a temporary
241 location, then moves the relevant files to their permanent location. 251 location, then moves the relevant files to their permanent location.
242 """ 252 """
243 configure_cmd = './configure --libdir=/%s/ %s' % ( 253 configure_cmd = './configure --libdir=/%s/ %s' % (
244 self._libdir, self._extra_configure_flags) 254 self._libdir, self._extra_configure_flags)
245 self.shell_call(configure_cmd, env=self._build_env, cwd=self._source_dir) 255 self.shell_call(configure_cmd, env=self._build_env, cwd=self._source_dir)
246 256
247 # Some makefiles use BUILDROOT or INSTALL_ROOT instead of DESTDIR. 257 # Some makefiles use BUILDROOT or INSTALL_ROOT instead of DESTDIR.
248 args = ['DESTDIR', 'BUILDROOT', 'INSTALL_ROOT'] 258 args = ['DESTDIR', 'BUILDROOT', 'INSTALL_ROOT']
249 make_args = ['%s=%s' % (name, self.temp_dir()) for name in args] 259 make_args = ['%s=%s' % (name, self.temp_dir()) for name in args]
250 self.make(make_args) 260 self.make(make_args)
251 261
252 # Some packages don't support parallel install. Use -j1 always. 262 self.make_install(make_args)
253 self.make_install(make_args, jobs=1)
254 263
255 self.cleanup_after_install() 264 self.cleanup_after_install()
256 265
257 self.fix_rpaths(self.temp_libdir()) 266 self.fix_rpaths(self.temp_libdir())
258 267
259 # Now move the contents of the temporary destdir to their final place. 268 # Now move the contents of the temporary destdir to their final place.
260 # We only care for the contents of LIBDIR. 269 # We only care for the contents of LIBDIR.
261 self.shell_call('cp %s/* %s/ -rdf' % (self.temp_libdir(), 270 self.shell_call('cp %s/* %s/ -rdf' % (self.temp_libdir(),
262 self.dest_libdir())) 271 self.dest_libdir()))
263 272
(...skipping 100 matching lines...) Expand 10 before | Expand all | Expand 10 after
364 make_args.append('USE_64=1') 373 make_args.append('USE_64=1')
365 374
366 # Make sure we don't override the default flags in the makefile. 375 # Make sure we don't override the default flags in the makefile.
367 for variable in ['CFLAGS', 'CXXFLAGS', 'LDFLAGS']: 376 for variable in ['CFLAGS', 'CXXFLAGS', 'LDFLAGS']:
368 del self._build_env[variable] 377 del self._build_env[variable]
369 378
370 # Hardcoded paths. 379 # Hardcoded paths.
371 temp_dir = os.path.join(self._source_dir, 'nss') 380 temp_dir = os.path.join(self._source_dir, 'nss')
372 temp_libdir = os.path.join(temp_dir, 'lib') 381 temp_libdir = os.path.join(temp_dir, 'lib')
373 382
374 # Parallel build is not supported. Also, the build happens in 383 # The build happens in <source_dir>/nss. Building fails after all
375 # <source_dir>/nss. 384 # the required DSOs have been built, so ignore the error.
376 try: 385 self.make(make_args, cwd=temp_dir, ignore_ret_code=True)
377 self.make(make_args, jobs=1, cwd=temp_dir)
378 except Exception:
379 # Building fails after all the required DSOs have been built, so ignore
380 # the error.
381 pass
382 386
383 self.fix_rpaths(temp_libdir) 387 self.fix_rpaths(temp_libdir)
384 388
385 # 'make install' is not supported. Copy the DSOs manually. 389 # 'make install' is not supported. Copy the DSOs manually.
386 for (dirpath, dirnames, filenames) in os.walk(temp_libdir): 390 for (dirpath, dirnames, filenames) in os.walk(temp_libdir):
387 for filename in filenames: 391 for filename in filenames:
388 if filename.endswith('.so'): 392 if filename.endswith('.so'):
389 full_path = os.path.join(dirpath, filename) 393 full_path = os.path.join(dirpath, filename)
390 if self._verbose: 394 if self._verbose:
391 print 'download_build_install.py: installing %s' % full_path 395 print 'download_build_install.py: installing %s' % full_path
392 shutil.copy(full_path, self.dest_libdir()) 396 shutil.copy(full_path, self.dest_libdir())
393 397
394 398
395 class StubBuilder(InstrumentedPackageBuilder): 399 class StubBuilder(InstrumentedPackageBuilder):
396 def download_build_install(self): 400 def download_build_install(self):
397 self._touch(os.path.join(self._destdir, '%s.txt' % self._package)) 401 self._touch(os.path.join(self._destdir, '%s.txt' % self._package))
402 self.shell_call('mkdir -p %s' % self.dest_libdir())
398 self._touch(os.path.join(self.dest_libdir(), '%s.so.0' % self._package)) 403 self._touch(os.path.join(self.dest_libdir(), '%s.so.0' % self._package))
399 404
400 def _touch(self, path): 405 def _touch(self, path):
401 with open(path, 'w'): 406 with open(path, 'w'):
402 pass 407 pass
403 408
404 409
405 def main(): 410 def main():
406 parser = argparse.ArgumentParser( 411 parser = argparse.ArgumentParser(
407 description='Download, build and install an instrumented package.') 412 description='Download, build and install an instrumented package.')
408 413
409 parser.add_argument('-j', '--jobs', type=int, default=1)
410 parser.add_argument('-p', '--package', required=True) 414 parser.add_argument('-p', '--package', required=True)
411 parser.add_argument( 415 parser.add_argument(
412 '-i', '--product-dir', default='.', 416 '-i', '--product-dir', default='.',
413 help='Relative path to the directory with chrome binaries') 417 help='Relative path to the directory with chrome binaries')
414 parser.add_argument( 418 parser.add_argument(
415 '-m', '--intermediate-dir', default='.', 419 '-m', '--intermediate-dir', default='.',
416 help='Relative path to the directory for temporary build files') 420 help='Relative path to the directory for temporary build files')
417 parser.add_argument('--extra-configure-flags', default='') 421 parser.add_argument('--extra-configure-flags', default='')
418 parser.add_argument('--cflags', default='') 422 parser.add_argument('--cflags', default='')
419 parser.add_argument('--ldflags', default='') 423 parser.add_argument('--ldflags', default='')
(...skipping 31 matching lines...) Expand 10 before | Expand all | Expand 10 after
451 builder = Libpci3Builder(args, clobber) 455 builder = Libpci3Builder(args, clobber)
452 elif args.build_method == 'stub': 456 elif args.build_method == 'stub':
453 builder = StubBuilder(args, clobber) 457 builder = StubBuilder(args, clobber)
454 else: 458 else:
455 raise Exception('Unrecognized build method: %s' % args.build_method) 459 raise Exception('Unrecognized build method: %s' % args.build_method)
456 460
457 builder.download_build_install() 461 builder.download_build_install()
458 462
459 if __name__ == '__main__': 463 if __name__ == '__main__':
460 main() 464 main()
OLDNEW
« no previous file with comments | « third_party/instrumented_libraries/scripts/build_and_package.py ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698