Clone of official tools

Committer:
The Other Jimmy
Date:
Thu Jun 22 11:12:28 2017 -0500
Revision:
36:96847d42f010
Parent:
31:8ea194f6145b
Child:
38:399953da035d
Tools release 5.5.1

Who changed what in which revision?

UserRevisionLine numberNew contents of line
screamer 0:66f3b5499f7f 1 #! /usr/bin/env python2
screamer 0:66f3b5499f7f 2 """
screamer 0:66f3b5499f7f 3 mbed SDK
screamer 0:66f3b5499f7f 4 Copyright (c) 2011-2013 ARM Limited
screamer 0:66f3b5499f7f 5
screamer 0:66f3b5499f7f 6 Licensed under the Apache License, Version 2.0 (the "License");
screamer 0:66f3b5499f7f 7 you may not use this file except in compliance with the License.
screamer 0:66f3b5499f7f 8 You may obtain a copy of the License at
screamer 0:66f3b5499f7f 9
screamer 0:66f3b5499f7f 10 http://www.apache.org/licenses/LICENSE-2.0
screamer 0:66f3b5499f7f 11
screamer 0:66f3b5499f7f 12 Unless required by applicable law or agreed to in writing, software
screamer 0:66f3b5499f7f 13 distributed under the License is distributed on an "AS IS" BASIS,
screamer 0:66f3b5499f7f 14 WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
screamer 0:66f3b5499f7f 15 See the License for the specific language governing permissions and
screamer 0:66f3b5499f7f 16 limitations under the License.
screamer 0:66f3b5499f7f 17
screamer 0:66f3b5499f7f 18
screamer 0:66f3b5499f7f 19 TEST BUILD & RUN
screamer 0:66f3b5499f7f 20 """
screamer 0:66f3b5499f7f 21 import sys
The Other Jimmy 31:8ea194f6145b 22 import json
screamer 0:66f3b5499f7f 23 from time import sleep
screamer 0:66f3b5499f7f 24 from shutil import copy
The Other Jimmy 31:8ea194f6145b 25 from os.path import join, abspath, dirname
The Other Jimmy 36:96847d42f010 26 from json import load, dump
screamer 0:66f3b5499f7f 27
screamer 0:66f3b5499f7f 28 # Be sure that the tools directory is in the search path
screamer 0:66f3b5499f7f 29 ROOT = abspath(join(dirname(__file__), ".."))
screamer 0:66f3b5499f7f 30 sys.path.insert(0, ROOT)
screamer 0:66f3b5499f7f 31
screamer 0:66f3b5499f7f 32 from tools.utils import args_error
The Other Jimmy 31:8ea194f6145b 33 from tools.utils import NotSupportedException
screamer 0:66f3b5499f7f 34 from tools.paths import BUILD_DIR
The Other Jimmy 31:8ea194f6145b 35 from tools.paths import MBED_LIBRARIES
screamer 0:66f3b5499f7f 36 from tools.paths import RPC_LIBRARY
The Other Jimmy 36:96847d42f010 37 from tools.paths import USB_LIBRARIES
screamer 0:66f3b5499f7f 38 from tools.paths import DSP_LIBRARIES
screamer 0:66f3b5499f7f 39 from tools.tests import TESTS, Test, TEST_MAP
screamer 0:66f3b5499f7f 40 from tools.tests import TEST_MBED_LIB
screamer 22:9e85236d8716 41 from tools.tests import test_known, test_name_known
screamer 0:66f3b5499f7f 42 from tools.targets import TARGET_MAP
screamer 0:66f3b5499f7f 43 from tools.options import get_default_options_parser
The Other Jimmy 31:8ea194f6145b 44 from tools.options import extract_profile
screamer 0:66f3b5499f7f 45 from tools.build_api import build_project
Screamer@Y5070-M.virtuoso 9:2d27d77ada5c 46 from tools.build_api import mcu_toolchain_matrix
The Other Jimmy 36:96847d42f010 47 from tools.build_api import mcu_toolchain_list
The Other Jimmy 36:96847d42f010 48 from tools.build_api import mcu_target_list
The Other Jimmy 36:96847d42f010 49 from tools.build_api import merge_build_data
screamer 22:9e85236d8716 50 from utils import argparse_filestring_type
screamer 22:9e85236d8716 51 from utils import argparse_many
screamer 24:25bff2709c20 52 from utils import argparse_dir_not_parent
The Other Jimmy 31:8ea194f6145b 53 from tools.toolchains import mbedToolchain, TOOLCHAIN_CLASSES, TOOLCHAIN_PATHS
screamer 22:9e85236d8716 54 from tools.settings import CLI_COLOR_MAP
screamer 0:66f3b5499f7f 55
screamer 0:66f3b5499f7f 56 if __name__ == '__main__':
screamer 0:66f3b5499f7f 57 # Parse Options
The Other Jimmy 31:8ea194f6145b 58 parser = get_default_options_parser(add_app_config=True)
screamer 22:9e85236d8716 59 group = parser.add_mutually_exclusive_group(required=False)
screamer 22:9e85236d8716 60 group.add_argument("-p",
screamer 22:9e85236d8716 61 type=argparse_many(test_known),
screamer 0:66f3b5499f7f 62 dest="program",
screamer 0:66f3b5499f7f 63 help="The index of the desired test program: [0-%d]" % (len(TESTS)-1))
screamer 0:66f3b5499f7f 64
screamer 22:9e85236d8716 65 group.add_argument("-n",
screamer 22:9e85236d8716 66 type=argparse_many(test_name_known),
screamer 22:9e85236d8716 67 dest="program",
screamer 0:66f3b5499f7f 68 help="The name of the desired test program")
screamer 0:66f3b5499f7f 69
screamer 22:9e85236d8716 70 parser.add_argument("-j", "--jobs",
screamer 22:9e85236d8716 71 type=int,
screamer 0:66f3b5499f7f 72 dest="jobs",
screamer 0:66f3b5499f7f 73 default=0,
screamer 0:66f3b5499f7f 74 help="Number of concurrent jobs. Default: 0/auto (based on host machine's number of CPUs)")
screamer 0:66f3b5499f7f 75
screamer 22:9e85236d8716 76 parser.add_argument("-v", "--verbose",
screamer 0:66f3b5499f7f 77 action="store_true",
screamer 0:66f3b5499f7f 78 dest="verbose",
screamer 0:66f3b5499f7f 79 default=False,
screamer 0:66f3b5499f7f 80 help="Verbose diagnostic output")
screamer 0:66f3b5499f7f 81
screamer 22:9e85236d8716 82 parser.add_argument("--silent",
screamer 0:66f3b5499f7f 83 action="store_true",
screamer 0:66f3b5499f7f 84 dest="silent",
screamer 0:66f3b5499f7f 85 default=False,
screamer 0:66f3b5499f7f 86 help="Silent diagnostic output (no copy, compile notification)")
screamer 0:66f3b5499f7f 87
screamer 22:9e85236d8716 88 parser.add_argument("-D",
screamer 0:66f3b5499f7f 89 action="append",
screamer 0:66f3b5499f7f 90 dest="macros",
screamer 0:66f3b5499f7f 91 help="Add a macro definition")
screamer 0:66f3b5499f7f 92
screamer 22:9e85236d8716 93 group.add_argument("-S", "--supported-toolchains",
Screamer@Y5070-M.virtuoso 9:2d27d77ada5c 94 dest="supported_toolchains",
Screamer@Y5070-M.virtuoso 9:2d27d77ada5c 95 default=False,
The Other Jimmy 36:96847d42f010 96 const="matrix",
The Other Jimmy 36:96847d42f010 97 choices=["matrix", "toolchains", "targets"],
The Other Jimmy 36:96847d42f010 98 nargs="?",
Screamer@Y5070-M.virtuoso 9:2d27d77ada5c 99 help="Displays supported matrix of MCUs and toolchains")
Screamer@Y5070-M.virtuoso 9:2d27d77ada5c 100
screamer 22:9e85236d8716 101 parser.add_argument('-f', '--filter',
Screamer@Y5070-M.virtuoso 9:2d27d77ada5c 102 dest='general_filter_regex',
Screamer@Y5070-M.virtuoso 9:2d27d77ada5c 103 default=None,
Screamer@Y5070-M.virtuoso 9:2d27d77ada5c 104 help='For some commands you can use filter to filter out results')
Screamer@Y5070-M.virtuoso 9:2d27d77ada5c 105
screamer 0:66f3b5499f7f 106 # Local run
screamer 22:9e85236d8716 107 parser.add_argument("--automated", action="store_true", dest="automated",
screamer 0:66f3b5499f7f 108 default=False, help="Automated test")
screamer 22:9e85236d8716 109 parser.add_argument("--host", dest="host_test",
screamer 0:66f3b5499f7f 110 default=None, help="Host test")
screamer 22:9e85236d8716 111 parser.add_argument("--extra", dest="extra",
screamer 0:66f3b5499f7f 112 default=None, help="Extra files")
screamer 22:9e85236d8716 113 parser.add_argument("--peripherals", dest="peripherals",
screamer 0:66f3b5499f7f 114 default=None, help="Required peripherals")
screamer 22:9e85236d8716 115 parser.add_argument("--dep", dest="dependencies",
screamer 0:66f3b5499f7f 116 default=None, help="Dependencies")
screamer 22:9e85236d8716 117 parser.add_argument("--source", dest="source_dir", type=argparse_filestring_type,
screamer 22:9e85236d8716 118 default=None, help="The source (input) directory", action="append")
screamer 22:9e85236d8716 119 parser.add_argument("--duration", type=int, dest="duration",
screamer 0:66f3b5499f7f 120 default=None, help="Duration of the test")
screamer 24:25bff2709c20 121 parser.add_argument("--build", dest="build_dir", type=argparse_dir_not_parent(ROOT),
screamer 0:66f3b5499f7f 122 default=None, help="The build (output) directory")
screamer 22:9e85236d8716 123 parser.add_argument("-N", "--artifact-name", dest="artifact_name",
screamer 0:66f3b5499f7f 124 default=None, help="The built project's name")
screamer 22:9e85236d8716 125 parser.add_argument("-d", "--disk", dest="disk",
screamer 0:66f3b5499f7f 126 default=None, help="The mbed disk")
screamer 22:9e85236d8716 127 parser.add_argument("-s", "--serial", dest="serial",
screamer 0:66f3b5499f7f 128 default=None, help="The mbed serial port")
screamer 22:9e85236d8716 129 parser.add_argument("-b", "--baud", type=int, dest="baud",
screamer 0:66f3b5499f7f 130 default=None, help="The mbed serial baud rate")
screamer 22:9e85236d8716 131 group.add_argument("-L", "--list-tests", action="store_true", dest="list_tests",
screamer 0:66f3b5499f7f 132 default=False, help="List available tests in order and exit")
screamer 0:66f3b5499f7f 133
screamer 0:66f3b5499f7f 134 # Ideally, all the tests with a single "main" thread can be run with, or
The Other Jimmy 36:96847d42f010 135 # without the usb, dsp
screamer 22:9e85236d8716 136 parser.add_argument("--rpc",
screamer 0:66f3b5499f7f 137 action="store_true", dest="rpc",
screamer 0:66f3b5499f7f 138 default=False, help="Link with RPC library")
screamer 0:66f3b5499f7f 139
screamer 22:9e85236d8716 140 parser.add_argument("--usb",
screamer 0:66f3b5499f7f 141 action="store_true",
screamer 0:66f3b5499f7f 142 dest="usb",
screamer 0:66f3b5499f7f 143 default=False,
screamer 0:66f3b5499f7f 144 help="Link with USB Device library")
screamer 0:66f3b5499f7f 145
screamer 22:9e85236d8716 146 parser.add_argument("--dsp",
screamer 0:66f3b5499f7f 147 action="store_true",
screamer 0:66f3b5499f7f 148 dest="dsp",
screamer 0:66f3b5499f7f 149 default=False,
screamer 0:66f3b5499f7f 150 help="Link with DSP library")
screamer 0:66f3b5499f7f 151
screamer 22:9e85236d8716 152 parser.add_argument("--testlib",
screamer 0:66f3b5499f7f 153 action="store_true",
screamer 0:66f3b5499f7f 154 dest="testlib",
screamer 0:66f3b5499f7f 155 default=False,
screamer 0:66f3b5499f7f 156 help="Link with mbed test library")
screamer 0:66f3b5499f7f 157
The Other Jimmy 36:96847d42f010 158 parser.add_argument("--build-data",
The Other Jimmy 36:96847d42f010 159 dest="build_data",
The Other Jimmy 36:96847d42f010 160 default=None,
The Other Jimmy 36:96847d42f010 161 help="Dump build_data to this file")
The Other Jimmy 36:96847d42f010 162
screamer 0:66f3b5499f7f 163 # Specify a different linker script
screamer 22:9e85236d8716 164 parser.add_argument("-l", "--linker", dest="linker_script",
screamer 22:9e85236d8716 165 type=argparse_filestring_type,
screamer 0:66f3b5499f7f 166 default=None, help="use the specified linker script")
screamer 0:66f3b5499f7f 167
screamer 22:9e85236d8716 168 options = parser.parse_args()
screamer 0:66f3b5499f7f 169
Screamer@Y5070-M.virtuoso 9:2d27d77ada5c 170 # Only prints matrix of supported toolchains
Screamer@Y5070-M.virtuoso 9:2d27d77ada5c 171 if options.supported_toolchains:
The Other Jimmy 36:96847d42f010 172 if options.supported_toolchains == "matrix":
The Other Jimmy 36:96847d42f010 173 print mcu_toolchain_matrix(platform_filter=options.general_filter_regex)
The Other Jimmy 36:96847d42f010 174 elif options.supported_toolchains == "toolchains":
The Other Jimmy 36:96847d42f010 175 toolchain_list = mcu_toolchain_list()
The Other Jimmy 36:96847d42f010 176 # Only print the lines that matter
The Other Jimmy 36:96847d42f010 177 for line in toolchain_list.split("\n"):
The Other Jimmy 36:96847d42f010 178 if not "mbed" in line:
The Other Jimmy 36:96847d42f010 179 print line
The Other Jimmy 36:96847d42f010 180 elif options.supported_toolchains == "targets":
The Other Jimmy 36:96847d42f010 181 print mcu_target_list()
Screamer@Y5070-M.virtuoso 9:2d27d77ada5c 182 exit(0)
Screamer@Y5070-M.virtuoso 9:2d27d77ada5c 183
screamer 0:66f3b5499f7f 184 # Print available tests in order and exit
screamer 0:66f3b5499f7f 185 if options.list_tests is True:
screamer 0:66f3b5499f7f 186 print '\n'.join(map(str, sorted(TEST_MAP.values())))
screamer 0:66f3b5499f7f 187 sys.exit()
screamer 0:66f3b5499f7f 188
screamer 0:66f3b5499f7f 189 # force program to "0" if a source dir is specified
screamer 0:66f3b5499f7f 190 if options.source_dir is not None:
screamer 0:66f3b5499f7f 191 p = 0
screamer 0:66f3b5499f7f 192 else:
screamer 0:66f3b5499f7f 193 # Program Number or name
screamer 22:9e85236d8716 194 p = options.program
screamer 0:66f3b5499f7f 195
screamer 0:66f3b5499f7f 196 # If 'p' was set via -n to list of numbers make this a single element integer list
screamer 0:66f3b5499f7f 197 if type(p) != type([]):
screamer 0:66f3b5499f7f 198 p = [p]
screamer 0:66f3b5499f7f 199
screamer 0:66f3b5499f7f 200 # Target
screamer 0:66f3b5499f7f 201 if options.mcu is None :
The Other Jimmy 31:8ea194f6145b 202 args_error(parser, "argument -m/--mcu is required")
screamer 22:9e85236d8716 203 mcu = options.mcu[0]
screamer 0:66f3b5499f7f 204
screamer 0:66f3b5499f7f 205 # Toolchain
screamer 0:66f3b5499f7f 206 if options.tool is None:
The Other Jimmy 31:8ea194f6145b 207 args_error(parser, "argument -t/--tool is required")
screamer 22:9e85236d8716 208 toolchain = options.tool[0]
screamer 22:9e85236d8716 209
The Other Jimmy 31:8ea194f6145b 210 if (options.program is None) and (not options.source_dir):
The Other Jimmy 31:8ea194f6145b 211 args_error(parser, "one of -p, -n, or --source is required")
The Other Jimmy 31:8ea194f6145b 212
The Other Jimmy 31:8ea194f6145b 213 if options.source_dir and not options.build_dir:
The Other Jimmy 31:8ea194f6145b 214 args_error(parser, "argument --build is required when argument --source is provided")
The Other Jimmy 31:8ea194f6145b 215
The Other Jimmy 31:8ea194f6145b 216
screamer 22:9e85236d8716 217 if options.color:
screamer 22:9e85236d8716 218 # This import happens late to prevent initializing colorization when we don't need it
screamer 22:9e85236d8716 219 import colorize
screamer 22:9e85236d8716 220 if options.verbose:
screamer 22:9e85236d8716 221 notify = mbedToolchain.print_notify_verbose
screamer 22:9e85236d8716 222 else:
screamer 22:9e85236d8716 223 notify = mbedToolchain.print_notify
screamer 22:9e85236d8716 224 notify = colorize.print_in_color_notifier(CLI_COLOR_MAP, notify)
screamer 22:9e85236d8716 225 else:
screamer 22:9e85236d8716 226 notify = None
screamer 0:66f3b5499f7f 227
The Other Jimmy 31:8ea194f6145b 228 if not TOOLCHAIN_CLASSES[toolchain].check_executable():
The Other Jimmy 31:8ea194f6145b 229 search_path = TOOLCHAIN_PATHS[toolchain] or "No path set"
The Other Jimmy 31:8ea194f6145b 230 args_error(parser, "Could not find executable for %s.\n"
The Other Jimmy 31:8ea194f6145b 231 "Currently set search path: %s"
The Other Jimmy 31:8ea194f6145b 232 %(toolchain,search_path))
The Other Jimmy 31:8ea194f6145b 233
screamer 0:66f3b5499f7f 234 # Test
The Other Jimmy 36:96847d42f010 235 build_data_blob = {} if options.build_data else None
screamer 0:66f3b5499f7f 236 for test_no in p:
screamer 0:66f3b5499f7f 237 test = Test(test_no)
screamer 0:66f3b5499f7f 238 if options.automated is not None: test.automated = options.automated
screamer 0:66f3b5499f7f 239 if options.dependencies is not None: test.dependencies = options.dependencies
screamer 0:66f3b5499f7f 240 if options.host_test is not None: test.host_test = options.host_test;
screamer 0:66f3b5499f7f 241 if options.peripherals is not None: test.peripherals = options.peripherals;
screamer 0:66f3b5499f7f 242 if options.duration is not None: test.duration = options.duration;
screamer 0:66f3b5499f7f 243 if options.extra is not None: test.extra_files = options.extra
screamer 0:66f3b5499f7f 244
screamer 0:66f3b5499f7f 245 if not test.is_supported(mcu, toolchain):
screamer 0:66f3b5499f7f 246 print 'The selected test is not supported on target %s with toolchain %s' % (mcu, toolchain)
screamer 0:66f3b5499f7f 247 sys.exit()
screamer 0:66f3b5499f7f 248
screamer 0:66f3b5499f7f 249 # Linking with extra libraries
screamer 0:66f3b5499f7f 250 if options.rpc: test.dependencies.append(RPC_LIBRARY)
screamer 0:66f3b5499f7f 251 if options.usb: test.dependencies.append(USB_LIBRARIES)
screamer 0:66f3b5499f7f 252 if options.dsp: test.dependencies.append(DSP_LIBRARIES)
screamer 0:66f3b5499f7f 253 if options.testlib: test.dependencies.append(TEST_MBED_LIB)
screamer 0:66f3b5499f7f 254
screamer 0:66f3b5499f7f 255 build_dir = join(BUILD_DIR, "test", mcu, toolchain, test.id)
screamer 0:66f3b5499f7f 256 if options.source_dir is not None:
screamer 0:66f3b5499f7f 257 test.source_dir = options.source_dir
screamer 0:66f3b5499f7f 258 build_dir = options.source_dir
screamer 0:66f3b5499f7f 259
screamer 0:66f3b5499f7f 260 if options.build_dir is not None:
screamer 0:66f3b5499f7f 261 build_dir = options.build_dir
screamer 0:66f3b5499f7f 262
screamer 0:66f3b5499f7f 263 try:
The Other Jimmy 31:8ea194f6145b 264 bin_file = build_project(test.source_dir, build_dir, mcu, toolchain,
The Other Jimmy 36:96847d42f010 265 set(test.dependencies),
screamer 0:66f3b5499f7f 266 linker_script=options.linker_script,
screamer 0:66f3b5499f7f 267 clean=options.clean,
screamer 0:66f3b5499f7f 268 verbose=options.verbose,
screamer 22:9e85236d8716 269 notify=notify,
The Other Jimmy 36:96847d42f010 270 report=build_data_blob,
screamer 0:66f3b5499f7f 271 silent=options.silent,
screamer 0:66f3b5499f7f 272 macros=options.macros,
screamer 0:66f3b5499f7f 273 jobs=options.jobs,
The Other Jimmy 31:8ea194f6145b 274 name=options.artifact_name,
The Other Jimmy 31:8ea194f6145b 275 app_config=options.app_config,
The Other Jimmy 31:8ea194f6145b 276 inc_dirs=[dirname(MBED_LIBRARIES)],
The Other Jimmy 31:8ea194f6145b 277 build_profile=extract_profile(parser,
The Other Jimmy 31:8ea194f6145b 278 options,
The Other Jimmy 31:8ea194f6145b 279 toolchain))
screamer 0:66f3b5499f7f 280 print 'Image: %s'% bin_file
screamer 0:66f3b5499f7f 281
screamer 0:66f3b5499f7f 282 if options.disk:
screamer 0:66f3b5499f7f 283 # Simple copy to the mbed disk
screamer 0:66f3b5499f7f 284 copy(bin_file, options.disk)
screamer 0:66f3b5499f7f 285
screamer 0:66f3b5499f7f 286 if options.serial:
screamer 0:66f3b5499f7f 287 # Import pyserial: https://pypi.python.org/pypi/pyserial
screamer 0:66f3b5499f7f 288 from serial import Serial
screamer 0:66f3b5499f7f 289
screamer 13:ab47a20b66f0 290 sleep(TARGET_MAP[mcu].program_cycle_s)
screamer 0:66f3b5499f7f 291
screamer 0:66f3b5499f7f 292 serial = Serial(options.serial, timeout = 1)
screamer 0:66f3b5499f7f 293 if options.baud:
screamer 0:66f3b5499f7f 294 serial.setBaudrate(options.baud)
screamer 0:66f3b5499f7f 295
screamer 0:66f3b5499f7f 296 serial.flushInput()
screamer 0:66f3b5499f7f 297 serial.flushOutput()
screamer 0:66f3b5499f7f 298
screamer 0:66f3b5499f7f 299 try:
screamer 0:66f3b5499f7f 300 serial.sendBreak()
screamer 0:66f3b5499f7f 301 except:
screamer 0:66f3b5499f7f 302 # In linux a termios.error is raised in sendBreak and in setBreak.
screamer 0:66f3b5499f7f 303 # The following setBreak() is needed to release the reset signal on the target mcu.
screamer 0:66f3b5499f7f 304 try:
screamer 0:66f3b5499f7f 305 serial.setBreak(False)
screamer 0:66f3b5499f7f 306 except:
screamer 0:66f3b5499f7f 307 pass
screamer 0:66f3b5499f7f 308
screamer 0:66f3b5499f7f 309 while True:
screamer 0:66f3b5499f7f 310 c = serial.read(512)
screamer 0:66f3b5499f7f 311 sys.stdout.write(c)
screamer 0:66f3b5499f7f 312 sys.stdout.flush()
screamer 0:66f3b5499f7f 313
screamer 0:66f3b5499f7f 314 except KeyboardInterrupt, e:
screamer 0:66f3b5499f7f 315 print "\n[CTRL+c] exit"
The Other Jimmy 36:96847d42f010 316 except NotSupportedException as e:
The Other Jimmy 36:96847d42f010 317 print "\nCould not compile for %s: %s" % (mcu, str(e))
screamer 0:66f3b5499f7f 318 except Exception,e:
screamer 0:66f3b5499f7f 319 if options.verbose:
screamer 0:66f3b5499f7f 320 import traceback
screamer 0:66f3b5499f7f 321 traceback.print_exc(file=sys.stdout)
screamer 0:66f3b5499f7f 322 else:
screamer 0:66f3b5499f7f 323 print "[ERROR] %s" % str(e)
screamer 0:66f3b5499f7f 324
screamer 0:66f3b5499f7f 325 sys.exit(1)
The Other Jimmy 36:96847d42f010 326 if options.build_data:
The Other Jimmy 36:96847d42f010 327 merge_build_data(options.build_data, build_data_blob, "application")