blob: dab2def135dc008810027420118808fffcf79058 [file] [log] [blame]
Ryan Cui6290f032012-11-20 15:44:43 -08001# Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
2# Use of this source code is governed by a BSD-style license that can be
3# found in the LICENSE file.
4
5"""Generates a sysroot tarball for building a specific package.
6
7Meant for use after setup_board and build_packages have been run.
8"""
9
Mike Frysinger383367e2014-09-16 15:06:17 -040010from __future__ import print_function
11
Ryan Cui6290f032012-11-20 15:44:43 -080012import os
13
Aviv Keshetb7519e12016-10-04 00:50:00 -070014from chromite.lib import constants
Ryan Cui6290f032012-11-20 15:44:43 -080015from chromite.lib import cros_build_lib
16from chromite.lib import commandline
17from chromite.lib import osutils
18from chromite.lib import sudo
Bertrand SIMONNETa5d8b552015-03-25 16:03:34 -070019from chromite.lib import sysroot_lib
Ryan Cui6290f032012-11-20 15:44:43 -080020
21DEFAULT_NAME = 'sysroot_%(package)s.tar.xz'
22PACKAGE_SEPARATOR = '/'
23SYSROOT = 'sysroot'
24
25
26def ParseCommandLine(argv):
27 """Parse args, and run environment-independent checks."""
28 parser = commandline.ArgumentParser(description=__doc__)
29 parser.add_argument('--board', required=True,
30 help=('The board to generate the sysroot for.'))
31 parser.add_argument('--package', required=True,
32 help=('The package to generate the sysroot for.'))
Michael Spang46c52fb2015-05-28 00:20:18 -040033 parser.add_argument('--out-dir', type='path', required=True,
Ryan Cui6290f032012-11-20 15:44:43 -080034 help='Directory to place the generated tarball.')
Mike Frysinger8557a292017-08-15 11:23:51 -040035 parser.add_argument('--out-file', default=DEFAULT_NAME,
36 help='The name to give to the tarball. '
37 'Defaults to %(default)s.')
Ryan Cui6290f032012-11-20 15:44:43 -080038 options = parser.parse_args(argv)
39
Mike Frysinger8557a292017-08-15 11:23:51 -040040 options.out_file %= {
41 'package': options.package.replace(PACKAGE_SEPARATOR, '_'),
42 }
Ryan Cui6290f032012-11-20 15:44:43 -080043
44 return options
45
46
47class GenerateSysroot(object):
48 """Wrapper for generation functionality."""
49
50 PARALLEL_EMERGE = os.path.join(constants.CHROMITE_BIN_DIR, 'parallel_emerge')
51
52 def __init__(self, sysroot, options):
53 """Initialize
54
Mike Frysinger02e1e072013-11-10 22:11:34 -050055 Args:
Ryan Cui6290f032012-11-20 15:44:43 -080056 sysroot: Path to sysroot.
57 options: Parsed options.
58 """
59 self.sysroot = sysroot
60 self.options = options
David James78d2e942013-07-31 15:34:45 -070061 self.extra_env = {'ROOT': self.sysroot, 'USE': os.environ.get('USE', '')}
62
63 def _Emerge(self, *args, **kwargs):
64 """Emerge the given packages using parallel_emerge."""
65 cmd = [self.PARALLEL_EMERGE, '--board=%s' % self.options.board,
66 '--usepkgonly', '--noreplace'] + list(args)
67 kwargs.setdefault('extra_env', self.extra_env)
68 cros_build_lib.SudoRunCommand(cmd, **kwargs)
Ryan Cui6290f032012-11-20 15:44:43 -080069
70 def _InstallToolchain(self):
Bertrand SIMONNETa5d8b552015-03-25 16:03:34 -070071 # Create the sysroot's config.
Bertrand SIMONNETe2cec3f2015-04-06 16:12:54 -070072 sysroot = sysroot_lib.Sysroot(self.sysroot)
73 sysroot.WriteConfig(sysroot.GenerateBoardConfig(self.options.board))
Ryan Cui6290f032012-11-20 15:44:43 -080074 cros_build_lib.RunCommand(
75 [os.path.join(constants.CROSUTILS_DIR, 'install_toolchain'),
Bertrand SIMONNETc1db71d2015-03-24 12:54:36 -070076 '--noconfigure', '--sysroot', self.sysroot])
Ryan Cui6290f032012-11-20 15:44:43 -080077
78 def _InstallKernelHeaders(self):
David James78d2e942013-07-31 15:34:45 -070079 self._Emerge('sys-kernel/linux-headers')
Ryan Cui6290f032012-11-20 15:44:43 -080080
81 def _InstallBuildDependencies(self):
David James78d2e942013-07-31 15:34:45 -070082 # Calculate buildtime deps that are not runtime deps.
Yu-Ju Hongdd9bb2b2014-01-03 17:08:26 -080083 raw_sysroot = cros_build_lib.GetSysroot(board=self.options.board)
David James78d2e942013-07-31 15:34:45 -070084 cmd = ['qdepends', '-q', '-C', self.options.package]
Yu-Ju Hong3add4432014-01-30 11:46:15 -080085 output = cros_build_lib.RunCommand(
86 cmd, extra_env={'ROOT': raw_sysroot}, capture_output=True).output
David James78d2e942013-07-31 15:34:45 -070087
88 if output.count('\n') > 1:
89 raise AssertionError('Too many packages matched given package pattern')
90
91 # qdepend outputs "package: deps", so only grab the deps.
92 atoms = output.partition(':')[2].split()
93
94 # Install the buildtime deps.
95 if atoms:
96 self._Emerge(*atoms)
Ryan Cui6290f032012-11-20 15:44:43 -080097
98 def _CreateTarball(self):
Ryan Cuid6d13332012-11-28 16:35:22 -080099 target = os.path.join(self.options.out_dir, self.options.out_file)
100 cros_build_lib.CreateTarball(target, self.sysroot, sudo=True)
Ryan Cui6290f032012-11-20 15:44:43 -0800101
102 def Perform(self):
103 """Generate the sysroot."""
104 self._InstallToolchain()
105 self._InstallKernelHeaders()
106 self._InstallBuildDependencies()
107 self._CreateTarball()
108
109
110def FinishParsing(options):
111 """Run environment dependent checks on parsed args."""
112 target = os.path.join(options.out_dir, options.out_file)
113 if os.path.exists(target):
114 cros_build_lib.Die('Output file %r already exists.' % target)
115
116 if not os.path.isdir(options.out_dir):
117 cros_build_lib.Die(
118 'Non-existent directory %r specified for --out-dir' % options.out_dir)
119
120
121def main(argv):
122 options = ParseCommandLine(argv)
123 FinishParsing(options)
124
Mike Frysinger8fd67dc2012-12-03 23:51:18 -0500125 cros_build_lib.AssertInsideChroot()
Ryan Cui6290f032012-11-20 15:44:43 -0800126
127 with sudo.SudoKeepAlive(ttyless_sudo=False):
David James4bc13702013-03-26 08:08:04 -0700128 with osutils.TempDir(set_global=True, sudo_rm=True) as tempdir:
Ryan Cui6290f032012-11-20 15:44:43 -0800129 sysroot = os.path.join(tempdir, SYSROOT)
130 os.mkdir(sysroot)
131 GenerateSysroot(sysroot, options).Perform()