Clone of official tools

Committer:
The Other Jimmy
Date:
Thu Jul 13 15:26:26 2017 -0500
Revision:
38:399953da035d
Parent:
36:96847d42f010
Child:
40:7d3fa6b99b2b
Update to tools release 5.5.2

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