mirror of
git://git.yoctoproject.org/poky.git
synced 2025-07-19 21:09:03 +02:00

Add a build-sdk command which is only available within the extensible SDK that builds a derivative extensible SDK. The idea is recipes in the workspace become a part of the new SDK - for example, this allows taking a vendor provided SDK, adding a few libs and then producing a new SDK with those included. When normally building the extensible SDK, the workspace is excluded; here we need to copy into the new SDK (renaming it in the process); the recipes' task signatures become locked and thus the sources are no longer needed, so they are removed along with the workspace bbappends which would interfere with the locked signatures. Additionally we need to just copy the configuration files (i.e. local.conf and auto.conf) rather than filtering and appending to them since that work has already been done when constructing the original SDK. The extra sstate artifacts from workspace recipes are also determined and copied into the new SDK in minimal mode (on the assumption that you won't set up a new sstate mirror). This reuses some code from build-image, so that needed to be generalised to allow that. Implements [YOCTO #8892]. (From OE-Core rev: 59e207ff6dd4b50a8905e14bc9292cf2794f4e7a) Signed-off-by: Paul Eggleton <paul.eggleton@linux.intel.com> Signed-off-by: Richard Purdie <richard.purdie@linuxfoundation.org>
155 lines
6.0 KiB
Python
155 lines
6.0 KiB
Python
# Development tool - build-image plugin
|
|
#
|
|
# Copyright (C) 2015 Intel Corporation
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License version 2 as
|
|
# published by the Free Software Foundation.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License along
|
|
# with this program; if not, write to the Free Software Foundation, Inc.,
|
|
# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
"""Devtool plugin containing the build-image subcommand."""
|
|
|
|
import os
|
|
import logging
|
|
|
|
from bb.process import ExecutionError
|
|
from devtool import exec_build_env_command, setup_tinfoil, parse_recipe, DevtoolError
|
|
|
|
logger = logging.getLogger('devtool')
|
|
|
|
class TargetNotImageError(Exception):
|
|
pass
|
|
|
|
def _get_packages(tinfoil, workspace, config):
|
|
"""Get list of packages from recipes in the workspace."""
|
|
result = []
|
|
for recipe in workspace:
|
|
data = parse_recipe(config, tinfoil, recipe, True)
|
|
if 'class-target' in data.getVar('OVERRIDES', True).split(':'):
|
|
if recipe in data.getVar('PACKAGES', True):
|
|
result.append(recipe)
|
|
else:
|
|
logger.warning("Skipping recipe %s as it doesn't produce a "
|
|
"package with the same name", recipe)
|
|
return result
|
|
|
|
def build_image(args, config, basepath, workspace):
|
|
"""Entry point for the devtool 'build-image' subcommand."""
|
|
|
|
image = args.imagename
|
|
auto_image = False
|
|
if not image:
|
|
sdk_targets = config.get('SDK', 'sdk_targets', '').split()
|
|
if sdk_targets:
|
|
image = sdk_targets[0]
|
|
auto_image = True
|
|
if not image:
|
|
raise DevtoolError('Unable to determine image to build, please specify one')
|
|
|
|
try:
|
|
if args.add_packages:
|
|
add_packages = args.add_packages.split(',')
|
|
else:
|
|
add_packages = None
|
|
result, outputdir = build_image_task(config, basepath, workspace, image, add_packages)
|
|
except TargetNotImageError:
|
|
if auto_image:
|
|
raise DevtoolError('Unable to determine image to build, please specify one')
|
|
else:
|
|
raise DevtoolError('Specified recipe %s is not an image recipe' % image)
|
|
|
|
if result == 0:
|
|
logger.info('Successfully built %s. You can find output files in %s'
|
|
% (image, outputdir))
|
|
return result
|
|
|
|
def build_image_task(config, basepath, workspace, image, add_packages=None, task=None, extra_append=None):
|
|
appendfile = os.path.join(config.workspace_path, 'appends',
|
|
'%s.bbappend' % image)
|
|
|
|
# remove <image>.bbappend to make sure setup_tinfoil doesn't
|
|
# break because of it
|
|
if os.path.isfile(appendfile):
|
|
os.unlink(appendfile)
|
|
|
|
tinfoil = setup_tinfoil(basepath=basepath)
|
|
rd = parse_recipe(config, tinfoil, image, True)
|
|
if not rd:
|
|
# Error already shown
|
|
return (1, None)
|
|
if not bb.data.inherits_class('image', rd):
|
|
raise TargetNotImageError()
|
|
|
|
outputdir = None
|
|
try:
|
|
if workspace or add_packages:
|
|
if add_packages:
|
|
packages = add_packages
|
|
else:
|
|
packages = _get_packages(tinfoil, workspace, config)
|
|
else:
|
|
packages = None
|
|
if not task:
|
|
if not packages and not add_packages and workspace:
|
|
logger.warning('No recipes in workspace, building image %s unmodified', image)
|
|
elif not packages:
|
|
logger.warning('No packages to add, building image %s unmodified', image)
|
|
|
|
if packages or extra_append:
|
|
bb.utils.mkdirhier(os.path.dirname(appendfile))
|
|
with open(appendfile, 'w') as afile:
|
|
if packages:
|
|
# include packages from workspace recipes into the image
|
|
afile.write('IMAGE_INSTALL_append = " %s"\n' % ' '.join(packages))
|
|
if not task:
|
|
logger.info('Building image %s with the following '
|
|
'additional packages: %s', image, ' '.join(packages))
|
|
if extra_append:
|
|
for line in extra_append:
|
|
afile.write('%s\n' % line)
|
|
|
|
if task in ['populate_sdk', 'populate_sdk_ext']:
|
|
outputdir = rd.getVar('SDK_DEPLOY', True)
|
|
else:
|
|
outputdir = rd.getVar('DEPLOY_DIR_IMAGE', True)
|
|
|
|
tinfoil.shutdown()
|
|
|
|
options = ''
|
|
if task:
|
|
options += '-c %s' % task
|
|
|
|
# run bitbake to build image (or specified task)
|
|
try:
|
|
exec_build_env_command(config.init_path, basepath,
|
|
'bitbake %s %s' % (options, image), watch=True)
|
|
except ExecutionError as err:
|
|
return (err.exitcode, None)
|
|
finally:
|
|
if os.path.isfile(appendfile):
|
|
os.unlink(appendfile)
|
|
return (0, outputdir)
|
|
|
|
|
|
def register_commands(subparsers, context):
|
|
"""Register devtool subcommands from the build-image plugin"""
|
|
parser = subparsers.add_parser('build-image',
|
|
help='Build image including workspace recipe packages',
|
|
description='Builds an image, extending it to include '
|
|
'packages from recipes in the workspace',
|
|
group='testbuild', order=-10)
|
|
parser.add_argument('imagename', help='Image recipe to build', nargs='?')
|
|
parser.add_argument('-p', '--add-packages', help='Instead of adding packages for the '
|
|
'entire workspace, specify packages to be added to the image '
|
|
'(separate multiple packages by commas)',
|
|
metavar='PACKAGES')
|
|
parser.set_defaults(func=build_image)
|