blob: 609786924cc3069917d37e24d3d1e4bae4140d37 [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
10import os
11
Mike Frysinger06a51c82021-04-06 11:39:17 -040012from chromite.lib import build_target_lib
Mike Frysinger807d8282022-04-28 22:45:17 -040013from chromite.lib import commandline
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
Ryan Cui6290f032012-11-20 15:44:43 -080016from chromite.lib import osutils
17from chromite.lib import sudo
Bertrand SIMONNETa5d8b552015-03-25 16:03:34 -070018from chromite.lib import sysroot_lib
Alex Klein8212d492018-08-27 07:47:23 -060019from chromite.lib import toolchain
Ryan Cui6290f032012-11-20 15:44:43 -080020
Mike Frysinger807d8282022-04-28 22:45:17 -040021
Ryan Cui6290f032012-11-20 15:44:43 -080022DEFAULT_NAME = 'sysroot_%(package)s.tar.xz'
23PACKAGE_SEPARATOR = '/'
24SYSROOT = 'sysroot'
25
26
27def ParseCommandLine(argv):
28 """Parse args, and run environment-independent checks."""
29 parser = commandline.ArgumentParser(description=__doc__)
30 parser.add_argument('--board', required=True,
Alex Klein8212d492018-08-27 07:47:23 -060031 help='The board to generate the sysroot for.')
Ryan Cui6290f032012-11-20 15:44:43 -080032 parser.add_argument('--package', required=True,
Alex Klein8212d492018-08-27 07:47:23 -060033 help='The packages to generate the sysroot for.')
Manoj Gupta2c2a2a22018-02-12 13:13:32 -080034 parser.add_argument('--deps-only', action='store_true',
35 default=False,
36 help='Build dependencies only.')
Michael Spang46c52fb2015-05-28 00:20:18 -040037 parser.add_argument('--out-dir', type='path', required=True,
Ryan Cui6290f032012-11-20 15:44:43 -080038 help='Directory to place the generated tarball.')
Mike Frysinger8557a292017-08-15 11:23:51 -040039 parser.add_argument('--out-file', default=DEFAULT_NAME,
40 help='The name to give to the tarball. '
41 'Defaults to %(default)s.')
Ryan Cui6290f032012-11-20 15:44:43 -080042 options = parser.parse_args(argv)
43
Mike Frysinger8557a292017-08-15 11:23:51 -040044 options.out_file %= {
Manoj Gupta2c2a2a22018-02-12 13:13:32 -080045 'package': options.package.split()[0].replace(PACKAGE_SEPARATOR, '_'),
Mike Frysinger8557a292017-08-15 11:23:51 -040046 }
Ryan Cui6290f032012-11-20 15:44:43 -080047
48 return options
49
50
51class GenerateSysroot(object):
52 """Wrapper for generation functionality."""
53
54 PARALLEL_EMERGE = os.path.join(constants.CHROMITE_BIN_DIR, 'parallel_emerge')
55
56 def __init__(self, sysroot, options):
57 """Initialize
58
Mike Frysinger02e1e072013-11-10 22:11:34 -050059 Args:
Ryan Cui6290f032012-11-20 15:44:43 -080060 sysroot: Path to sysroot.
61 options: Parsed options.
62 """
63 self.sysroot = sysroot
64 self.options = options
David James78d2e942013-07-31 15:34:45 -070065 self.extra_env = {'ROOT': self.sysroot, 'USE': os.environ.get('USE', '')}
66
67 def _Emerge(self, *args, **kwargs):
68 """Emerge the given packages using parallel_emerge."""
69 cmd = [self.PARALLEL_EMERGE, '--board=%s' % self.options.board,
70 '--usepkgonly', '--noreplace'] + list(args)
71 kwargs.setdefault('extra_env', self.extra_env)
Mike Frysinger45602c72019-09-22 02:15:11 -040072 cros_build_lib.sudo_run(cmd, **kwargs)
Ryan Cui6290f032012-11-20 15:44:43 -080073
Sloan Johnsona85640f2021-10-01 22:32:40 +000074 def _WriteConfig(self, sysroot):
75 sysroot.WriteConfig(sysroot.GenerateBoardSetupConfig(self.options.board))
76 # For the config to be correctly read, a stub make.conf is needed.
77 # pylint: disable=protected-access
78 make_conf_path = os.path.join(self.sysroot, sysroot_lib._MAKE_CONF)
79 assert not os.path.exists(make_conf_path), 'Expecting an empty sysroot.'
80 osutils.WriteFile(os.path.join(make_conf_path),
81 'source make.conf.board_setup', makedirs=True, sudo=True)
82
Ryan Cui6290f032012-11-20 15:44:43 -080083 def _InstallToolchain(self):
Bertrand SIMONNETa5d8b552015-03-25 16:03:34 -070084 # Create the sysroot's config.
Bertrand SIMONNETe2cec3f2015-04-06 16:12:54 -070085 sysroot = sysroot_lib.Sysroot(self.sysroot)
Sloan Johnsona85640f2021-10-01 22:32:40 +000086 self._WriteConfig(sysroot)
Alex Klein8212d492018-08-27 07:47:23 -060087 toolchain.InstallToolchain(sysroot, configure=False)
Ryan Cui6290f032012-11-20 15:44:43 -080088
89 def _InstallKernelHeaders(self):
David James78d2e942013-07-31 15:34:45 -070090 self._Emerge('sys-kernel/linux-headers')
Ryan Cui6290f032012-11-20 15:44:43 -080091
92 def _InstallBuildDependencies(self):
David James78d2e942013-07-31 15:34:45 -070093 # Calculate buildtime deps that are not runtime deps.
Mike Frysinger06a51c82021-04-06 11:39:17 -040094 raw_sysroot = build_target_lib.get_default_sysroot_path(self.options.board)
Manoj Gupta2c2a2a22018-02-12 13:13:32 -080095 packages = []
96 if not self.options.deps_only:
97 packages = self.options.package.split()
98 else:
99 for pkg in self.options.package.split():
100 cmd = ['qdepends', '-q', '-C', pkg]
Mike Frysinger45602c72019-09-22 02:15:11 -0400101 output = cros_build_lib.run(
Mike Frysingerc04ddb92019-12-09 21:19:47 -0500102 cmd, extra_env={'ROOT': raw_sysroot}, capture_output=True,
103 encoding='utf-8').stdout
David James78d2e942013-07-31 15:34:45 -0700104
Manoj Gupta2c2a2a22018-02-12 13:13:32 -0800105 if output.count('\n') > 1:
106 raise AssertionError('Too many packages matched for given pattern')
David James78d2e942013-07-31 15:34:45 -0700107
Manoj Gupta2c2a2a22018-02-12 13:13:32 -0800108 # qdepend outputs "package: deps", so only grab the deps.
109 deps = output.partition(':')[2].split()
110 packages.extend(deps)
111 # Install the required packages.
112 if packages:
113 self._Emerge(*packages)
Ryan Cui6290f032012-11-20 15:44:43 -0800114
115 def _CreateTarball(self):
Eliot Courtney93f76212020-10-22 11:16:33 +0900116 tarball_path = os.path.join(self.options.out_dir, self.options.out_file)
117 cros_build_lib.CreateTarball(tarball_path, self.sysroot, sudo=True)
Ryan Cui6290f032012-11-20 15:44:43 -0800118
119 def Perform(self):
120 """Generate the sysroot."""
121 self._InstallToolchain()
122 self._InstallKernelHeaders()
123 self._InstallBuildDependencies()
124 self._CreateTarball()
125
126
127def FinishParsing(options):
128 """Run environment dependent checks on parsed args."""
129 target = os.path.join(options.out_dir, options.out_file)
130 if os.path.exists(target):
131 cros_build_lib.Die('Output file %r already exists.' % target)
132
133 if not os.path.isdir(options.out_dir):
134 cros_build_lib.Die(
135 'Non-existent directory %r specified for --out-dir' % options.out_dir)
136
137
138def main(argv):
139 options = ParseCommandLine(argv)
140 FinishParsing(options)
141
Mike Frysinger8fd67dc2012-12-03 23:51:18 -0500142 cros_build_lib.AssertInsideChroot()
Ryan Cui6290f032012-11-20 15:44:43 -0800143
144 with sudo.SudoKeepAlive(ttyless_sudo=False):
David James4bc13702013-03-26 08:08:04 -0700145 with osutils.TempDir(set_global=True, sudo_rm=True) as tempdir:
Ryan Cui6290f032012-11-20 15:44:43 -0800146 sysroot = os.path.join(tempdir, SYSROOT)
147 os.mkdir(sysroot)
148 GenerateSysroot(sysroot, options).Perform()