2015-04-22 22:23:19 +00:00
|
|
|
#! /usr/bin/env python2
|
2013-02-18 15:32:11 +00:00
|
|
|
"""
|
2013-08-06 13:38:00 +00:00
|
|
|
mbed SDK
|
|
|
|
Copyright (c) 2011-2013 ARM Limited
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
|
|
|
|
|
2013-02-18 15:32:11 +00:00
|
|
|
TEST BUILD & RUN
|
|
|
|
"""
|
|
|
|
import sys
|
2016-09-23 20:44:09 +00:00
|
|
|
import json
|
2013-02-18 15:32:11 +00:00
|
|
|
from time import sleep
|
2014-09-05 10:48:37 +00:00
|
|
|
from shutil import copy
|
2016-09-13 18:38:58 +00:00
|
|
|
from os.path import join, abspath, dirname
|
2017-04-04 16:35:37 +00:00
|
|
|
from json import load, dump
|
2013-02-18 15:32:11 +00:00
|
|
|
|
|
|
|
# Be sure that the tools directory is in the search path
|
|
|
|
ROOT = abspath(join(dirname(__file__), ".."))
|
2013-12-19 13:02:57 +00:00
|
|
|
sys.path.insert(0, ROOT)
|
2013-02-18 15:32:11 +00:00
|
|
|
|
2016-06-09 20:34:53 +00:00
|
|
|
from tools.utils import args_error
|
2016-10-26 16:33:36 +00:00
|
|
|
from tools.utils import NotSupportedException
|
2016-06-09 20:34:53 +00:00
|
|
|
from tools.paths import BUILD_DIR
|
2016-10-01 23:08:38 +00:00
|
|
|
from tools.paths import MBED_LIBRARIES
|
2016-06-09 20:34:53 +00:00
|
|
|
from tools.paths import RPC_LIBRARY
|
2017-05-04 13:59:49 +00:00
|
|
|
from tools.paths import USB_LIBRARIES
|
2016-06-09 20:34:53 +00:00
|
|
|
from tools.paths import DSP_LIBRARIES
|
|
|
|
from tools.tests import TESTS, Test, TEST_MAP
|
|
|
|
from tools.tests import TEST_MBED_LIB
|
2016-06-24 22:15:01 +00:00
|
|
|
from tools.tests import test_known, test_name_known
|
2016-06-09 20:34:53 +00:00
|
|
|
from tools.targets import TARGET_MAP
|
|
|
|
from tools.options import get_default_options_parser
|
2016-09-27 18:15:22 +00:00
|
|
|
from tools.options import extract_profile
|
2017-04-06 19:40:38 +00:00
|
|
|
from tools.options import extract_mcus
|
2016-06-09 20:34:53 +00:00
|
|
|
from tools.build_api import build_project
|
2016-06-12 00:06:15 +00:00
|
|
|
from tools.build_api import mcu_toolchain_matrix
|
2017-03-22 18:06:57 +00:00
|
|
|
from tools.build_api import mcu_toolchain_list
|
|
|
|
from tools.build_api import mcu_target_list
|
2017-04-06 16:17:54 +00:00
|
|
|
from tools.build_api import merge_build_data
|
2016-06-24 22:15:01 +00:00
|
|
|
from utils import argparse_filestring_type
|
2016-06-29 00:58:02 +00:00
|
|
|
from utils import argparse_many
|
2016-07-11 16:08:11 +00:00
|
|
|
from utils import argparse_dir_not_parent
|
2016-09-12 23:54:39 +00:00
|
|
|
from tools.toolchains import mbedToolchain, TOOLCHAIN_CLASSES, TOOLCHAIN_PATHS
|
2016-07-07 19:39:59 +00:00
|
|
|
from tools.settings import CLI_COLOR_MAP
|
2016-06-21 15:55:01 +00:00
|
|
|
|
2013-02-18 15:32:11 +00:00
|
|
|
if __name__ == '__main__':
|
|
|
|
# Parse Options
|
2016-09-01 10:41:19 +00:00
|
|
|
parser = get_default_options_parser(add_app_config=True)
|
2016-06-29 22:28:50 +00:00
|
|
|
group = parser.add_mutually_exclusive_group(required=False)
|
2017-05-25 11:09:18 +00:00
|
|
|
group.add_argument(
|
|
|
|
"-p",
|
|
|
|
type=argparse_many(test_known),
|
|
|
|
dest="program",
|
|
|
|
help="The index of the desired test program: [0-%d]" % (len(TESTS)-1))
|
|
|
|
|
|
|
|
group.add_argument(
|
|
|
|
"-n",
|
|
|
|
type=argparse_many(test_name_known),
|
|
|
|
dest="program",
|
|
|
|
help="The name of the desired test program")
|
|
|
|
|
|
|
|
parser.add_argument(
|
|
|
|
"-j", "--jobs",
|
|
|
|
type=int,
|
|
|
|
dest="jobs",
|
|
|
|
default=0,
|
|
|
|
help="Number of concurrent jobs. Default: 0/auto (based on host machine's number of CPUs)")
|
|
|
|
|
|
|
|
parser.add_argument(
|
|
|
|
"-v", "--verbose",
|
|
|
|
action="store_true",
|
|
|
|
dest="verbose",
|
|
|
|
default=False,
|
|
|
|
help="Verbose diagnostic output")
|
|
|
|
|
|
|
|
parser.add_argument(
|
|
|
|
"--silent",
|
|
|
|
action="store_true",
|
|
|
|
dest="silent",
|
|
|
|
default=False,
|
|
|
|
help="Silent diagnostic output (no copy, compile notification)")
|
|
|
|
|
|
|
|
parser.add_argument(
|
|
|
|
"-D",
|
|
|
|
action="append",
|
|
|
|
dest="macros",
|
|
|
|
help="Add a macro definition")
|
|
|
|
|
|
|
|
group.add_argument(
|
|
|
|
"-S", "--supported-toolchains",
|
|
|
|
dest="supported_toolchains",
|
|
|
|
default=False,
|
|
|
|
const="matrix",
|
|
|
|
choices=["matrix", "toolchains", "targets"],
|
|
|
|
nargs="?",
|
|
|
|
help="Displays supported matrix of MCUs and toolchains")
|
|
|
|
|
|
|
|
parser.add_argument(
|
|
|
|
'-f', '--filter',
|
|
|
|
dest='general_filter_regex',
|
|
|
|
default=None,
|
|
|
|
help='For some commands you can use filter to filter out results')
|
|
|
|
|
|
|
|
parser.add_argument(
|
|
|
|
"--stats-depth",
|
|
|
|
type=int,
|
|
|
|
dest="stats_depth",
|
2018-01-23 23:26:48 +00:00
|
|
|
default=None,
|
2017-05-25 11:09:18 +00:00
|
|
|
help="Depth level for static memory report")
|
2016-06-12 00:06:15 +00:00
|
|
|
|
2013-02-18 15:32:11 +00:00
|
|
|
# Local run
|
2016-06-24 22:15:01 +00:00
|
|
|
parser.add_argument("--automated", action="store_true", dest="automated",
|
2013-10-20 15:17:10 +00:00
|
|
|
default=False, help="Automated test")
|
2016-06-24 22:15:01 +00:00
|
|
|
parser.add_argument("--host", dest="host_test",
|
2013-10-20 15:17:10 +00:00
|
|
|
default=None, help="Host test")
|
2016-06-24 22:15:01 +00:00
|
|
|
parser.add_argument("--extra", dest="extra",
|
2013-10-20 15:17:10 +00:00
|
|
|
default=None, help="Extra files")
|
2016-06-24 22:15:01 +00:00
|
|
|
parser.add_argument("--peripherals", dest="peripherals",
|
2013-10-20 15:17:10 +00:00
|
|
|
default=None, help="Required peripherals")
|
2016-06-24 22:15:01 +00:00
|
|
|
parser.add_argument("--dep", dest="dependencies",
|
2013-10-20 15:17:10 +00:00
|
|
|
default=None, help="Dependencies")
|
2016-06-29 22:00:29 +00:00
|
|
|
parser.add_argument("--source", dest="source_dir", type=argparse_filestring_type,
|
|
|
|
default=None, help="The source (input) directory", action="append")
|
2016-06-24 22:15:01 +00:00
|
|
|
parser.add_argument("--duration", type=int, dest="duration",
|
2013-10-20 15:17:10 +00:00
|
|
|
default=None, help="Duration of the test")
|
2016-07-11 16:08:11 +00:00
|
|
|
parser.add_argument("--build", dest="build_dir", type=argparse_dir_not_parent(ROOT),
|
2013-10-20 15:17:10 +00:00
|
|
|
default=None, help="The build (output) directory")
|
2016-06-24 22:15:01 +00:00
|
|
|
parser.add_argument("-N", "--artifact-name", dest="artifact_name",
|
2016-06-09 22:11:23 +00:00
|
|
|
default=None, help="The built project's name")
|
2016-06-24 22:15:01 +00:00
|
|
|
parser.add_argument("-d", "--disk", dest="disk",
|
2013-02-18 15:32:11 +00:00
|
|
|
default=None, help="The mbed disk")
|
2016-06-24 22:15:01 +00:00
|
|
|
parser.add_argument("-s", "--serial", dest="serial",
|
2013-02-18 15:32:11 +00:00
|
|
|
default=None, help="The mbed serial port")
|
2016-06-24 22:15:01 +00:00
|
|
|
parser.add_argument("-b", "--baud", type=int, dest="baud",
|
2013-02-18 15:32:11 +00:00
|
|
|
default=None, help="The mbed serial baud rate")
|
2016-06-24 22:15:01 +00:00
|
|
|
group.add_argument("-L", "--list-tests", action="store_true", dest="list_tests",
|
2014-03-27 11:17:53 +00:00
|
|
|
default=False, help="List available tests in order and exit")
|
|
|
|
|
2013-02-18 15:32:11 +00:00
|
|
|
# Ideally, all the tests with a single "main" thread can be run with, or
|
2017-05-04 13:59:49 +00:00
|
|
|
# without the usb, dsp
|
2016-06-24 22:15:01 +00:00
|
|
|
parser.add_argument("--rpc",
|
2015-11-15 08:52:54 +00:00
|
|
|
action="store_true", dest="rpc",
|
|
|
|
default=False, help="Link with RPC library")
|
|
|
|
|
2016-06-24 22:15:01 +00:00
|
|
|
parser.add_argument("--usb",
|
2014-12-01 14:51:55 +00:00
|
|
|
action="store_true",
|
|
|
|
dest="usb",
|
|
|
|
default=False,
|
|
|
|
help="Link with USB Device library")
|
|
|
|
|
2016-06-24 22:15:01 +00:00
|
|
|
parser.add_argument("--dsp",
|
2014-12-01 14:51:55 +00:00
|
|
|
action="store_true",
|
|
|
|
dest="dsp",
|
|
|
|
default=False,
|
|
|
|
help="Link with DSP library")
|
|
|
|
|
2016-06-24 22:15:01 +00:00
|
|
|
parser.add_argument("--testlib",
|
2014-12-01 14:51:55 +00:00
|
|
|
action="store_true",
|
|
|
|
dest="testlib",
|
|
|
|
default=False,
|
|
|
|
help="Link with mbed test library")
|
2014-03-27 11:17:53 +00:00
|
|
|
|
2017-04-05 21:40:01 +00:00
|
|
|
parser.add_argument("--build-data",
|
|
|
|
dest="build_data",
|
2017-04-04 16:35:37 +00:00
|
|
|
default=None,
|
2017-04-05 21:40:01 +00:00
|
|
|
help="Dump build_data to this file")
|
2017-04-04 16:35:37 +00:00
|
|
|
|
2013-06-10 14:44:08 +00:00
|
|
|
# Specify a different linker script
|
2016-06-24 22:15:01 +00:00
|
|
|
parser.add_argument("-l", "--linker", dest="linker_script",
|
|
|
|
type=argparse_filestring_type,
|
2013-06-10 14:44:08 +00:00
|
|
|
default=None, help="use the specified linker script")
|
2014-03-27 11:17:53 +00:00
|
|
|
|
2016-06-24 22:15:01 +00:00
|
|
|
options = parser.parse_args()
|
2013-10-20 15:17:10 +00:00
|
|
|
|
2016-06-12 00:06:15 +00:00
|
|
|
# Only prints matrix of supported toolchains
|
|
|
|
if options.supported_toolchains:
|
2017-03-22 18:06:57 +00:00
|
|
|
if options.supported_toolchains == "matrix":
|
|
|
|
print mcu_toolchain_matrix(platform_filter=options.general_filter_regex)
|
|
|
|
elif options.supported_toolchains == "toolchains":
|
2017-03-30 16:22:42 +00:00
|
|
|
toolchain_list = mcu_toolchain_list()
|
|
|
|
# Only print the lines that matter
|
|
|
|
for line in toolchain_list.split("\n"):
|
|
|
|
if not "mbed" in line:
|
|
|
|
print line
|
2017-03-22 18:06:57 +00:00
|
|
|
elif options.supported_toolchains == "targets":
|
|
|
|
print mcu_target_list()
|
2016-06-12 00:06:15 +00:00
|
|
|
exit(0)
|
|
|
|
|
2014-03-27 11:17:53 +00:00
|
|
|
# Print available tests in order and exit
|
|
|
|
if options.list_tests is True:
|
|
|
|
print '\n'.join(map(str, sorted(TEST_MAP.values())))
|
|
|
|
sys.exit()
|
|
|
|
|
2013-10-20 15:17:10 +00:00
|
|
|
# force program to "0" if a source dir is specified
|
|
|
|
if options.source_dir is not None:
|
|
|
|
p = 0
|
|
|
|
else:
|
2013-06-24 13:32:08 +00:00
|
|
|
# Program Number or name
|
2016-06-24 22:15:01 +00:00
|
|
|
p = options.program
|
2014-12-01 15:33:30 +00:00
|
|
|
|
|
|
|
# If 'p' was set via -n to list of numbers make this a single element integer list
|
|
|
|
if type(p) != type([]):
|
|
|
|
p = [p]
|
2014-03-27 11:17:53 +00:00
|
|
|
|
2013-02-18 15:32:11 +00:00
|
|
|
# Target
|
|
|
|
if options.mcu is None :
|
2016-08-04 19:48:18 +00:00
|
|
|
args_error(parser, "argument -m/--mcu is required")
|
2017-04-06 19:40:38 +00:00
|
|
|
mcu = extract_mcus(parser, options)[0]
|
2014-03-27 11:17:53 +00:00
|
|
|
|
2013-02-18 15:32:11 +00:00
|
|
|
# Toolchain
|
|
|
|
if options.tool is None:
|
2016-08-04 21:18:20 +00:00
|
|
|
args_error(parser, "argument -t/--tool is required")
|
2016-06-24 22:15:01 +00:00
|
|
|
toolchain = options.tool[0]
|
2014-03-27 11:17:53 +00:00
|
|
|
|
2016-08-04 21:18:20 +00:00
|
|
|
if (options.program is None) and (not options.source_dir):
|
|
|
|
args_error(parser, "one of -p, -n, or --source is required")
|
|
|
|
|
|
|
|
if options.source_dir and not options.build_dir:
|
|
|
|
args_error(parser, "argument --build is required when argument --source is provided")
|
|
|
|
|
2016-09-23 20:44:09 +00:00
|
|
|
|
2016-07-07 19:39:59 +00:00
|
|
|
if options.color:
|
|
|
|
# This import happens late to prevent initializing colorization when we don't need it
|
|
|
|
import colorize
|
|
|
|
if options.verbose:
|
|
|
|
notify = mbedToolchain.print_notify_verbose
|
|
|
|
else:
|
|
|
|
notify = mbedToolchain.print_notify
|
|
|
|
notify = colorize.print_in_color_notifier(CLI_COLOR_MAP, notify)
|
|
|
|
else:
|
|
|
|
notify = None
|
|
|
|
|
2016-09-12 23:54:39 +00:00
|
|
|
if not TOOLCHAIN_CLASSES[toolchain].check_executable():
|
2016-09-13 17:06:01 +00:00
|
|
|
search_path = TOOLCHAIN_PATHS[toolchain] or "No path set"
|
2016-09-12 23:54:39 +00:00
|
|
|
args_error(parser, "Could not find executable for %s.\n"
|
|
|
|
"Currently set search path: %s"
|
2016-09-13 17:06:01 +00:00
|
|
|
%(toolchain,search_path))
|
2016-09-12 23:54:39 +00:00
|
|
|
|
2013-02-18 15:32:11 +00:00
|
|
|
# Test
|
2017-04-05 21:40:01 +00:00
|
|
|
build_data_blob = {} if options.build_data else None
|
2014-12-01 15:07:17 +00:00
|
|
|
for test_no in p:
|
|
|
|
test = Test(test_no)
|
|
|
|
if options.automated is not None: test.automated = options.automated
|
|
|
|
if options.dependencies is not None: test.dependencies = options.dependencies
|
|
|
|
if options.host_test is not None: test.host_test = options.host_test;
|
|
|
|
if options.peripherals is not None: test.peripherals = options.peripherals;
|
|
|
|
if options.duration is not None: test.duration = options.duration;
|
|
|
|
if options.extra is not None: test.extra_files = options.extra
|
|
|
|
|
|
|
|
if not test.is_supported(mcu, toolchain):
|
|
|
|
print 'The selected test is not supported on target %s with toolchain %s' % (mcu, toolchain)
|
|
|
|
sys.exit()
|
|
|
|
|
|
|
|
# Linking with extra libraries
|
2015-11-15 08:52:54 +00:00
|
|
|
if options.rpc: test.dependencies.append(RPC_LIBRARY)
|
2014-12-01 15:07:17 +00:00
|
|
|
if options.usb: test.dependencies.append(USB_LIBRARIES)
|
|
|
|
if options.dsp: test.dependencies.append(DSP_LIBRARIES)
|
|
|
|
if options.testlib: test.dependencies.append(TEST_MBED_LIB)
|
|
|
|
|
|
|
|
build_dir = join(BUILD_DIR, "test", mcu, toolchain, test.id)
|
|
|
|
if options.source_dir is not None:
|
|
|
|
test.source_dir = options.source_dir
|
|
|
|
build_dir = options.source_dir
|
|
|
|
|
|
|
|
if options.build_dir is not None:
|
|
|
|
build_dir = options.build_dir
|
|
|
|
|
|
|
|
try:
|
2016-09-27 18:15:22 +00:00
|
|
|
bin_file = build_project(test.source_dir, build_dir, mcu, toolchain,
|
2017-03-07 00:23:16 +00:00
|
|
|
set(test.dependencies),
|
2014-12-01 15:33:30 +00:00
|
|
|
linker_script=options.linker_script,
|
|
|
|
clean=options.clean,
|
|
|
|
verbose=options.verbose,
|
2016-07-07 19:39:59 +00:00
|
|
|
notify=notify,
|
2017-04-05 21:40:01 +00:00
|
|
|
report=build_data_blob,
|
2014-12-03 10:53:50 +00:00
|
|
|
silent=options.silent,
|
2014-12-01 15:33:30 +00:00
|
|
|
macros=options.macros,
|
2016-06-09 22:11:23 +00:00
|
|
|
jobs=options.jobs,
|
2016-09-01 10:41:19 +00:00
|
|
|
name=options.artifact_name,
|
2016-09-23 20:44:09 +00:00
|
|
|
app_config=options.app_config,
|
2016-10-01 23:08:38 +00:00
|
|
|
inc_dirs=[dirname(MBED_LIBRARIES)],
|
2016-09-27 18:15:22 +00:00
|
|
|
build_profile=extract_profile(parser,
|
|
|
|
options,
|
2017-05-25 11:09:18 +00:00
|
|
|
toolchain),
|
|
|
|
stats_depth=options.stats_depth)
|
2014-12-01 15:33:30 +00:00
|
|
|
print 'Image: %s'% bin_file
|
2014-12-01 15:07:17 +00:00
|
|
|
|
|
|
|
if options.disk:
|
|
|
|
# Simple copy to the mbed disk
|
2014-12-01 15:33:30 +00:00
|
|
|
copy(bin_file, options.disk)
|
2014-12-01 15:07:17 +00:00
|
|
|
|
|
|
|
if options.serial:
|
|
|
|
# Import pyserial: https://pypi.python.org/pypi/pyserial
|
|
|
|
from serial import Serial
|
|
|
|
|
2016-06-27 21:01:52 +00:00
|
|
|
sleep(TARGET_MAP[mcu].program_cycle_s)
|
2014-12-01 15:07:17 +00:00
|
|
|
|
|
|
|
serial = Serial(options.serial, timeout = 1)
|
|
|
|
if options.baud:
|
|
|
|
serial.setBaudrate(options.baud)
|
|
|
|
|
|
|
|
serial.flushInput()
|
|
|
|
serial.flushOutput()
|
2014-03-27 11:17:53 +00:00
|
|
|
|
2014-06-23 16:34:10 +00:00
|
|
|
try:
|
2014-12-01 15:07:17 +00:00
|
|
|
serial.sendBreak()
|
2014-06-23 16:34:10 +00:00
|
|
|
except:
|
2014-12-01 15:07:17 +00:00
|
|
|
# In linux a termios.error is raised in sendBreak and in setBreak.
|
|
|
|
# The following setBreak() is needed to release the reset signal on the target mcu.
|
|
|
|
try:
|
|
|
|
serial.setBreak(False)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
while True:
|
|
|
|
c = serial.read(512)
|
|
|
|
sys.stdout.write(c)
|
|
|
|
sys.stdout.flush()
|
|
|
|
|
|
|
|
except KeyboardInterrupt, e:
|
|
|
|
print "\n[CTRL+c] exit"
|
2017-05-31 18:08:15 +00:00
|
|
|
except NotSupportedException as e:
|
|
|
|
print "\nCould not compile for %s: %s" % (mcu, str(e))
|
2014-12-01 15:07:17 +00:00
|
|
|
except Exception,e:
|
|
|
|
if options.verbose:
|
|
|
|
import traceback
|
|
|
|
traceback.print_exc(file=sys.stdout)
|
|
|
|
else:
|
|
|
|
print "[ERROR] %s" % str(e)
|
2017-05-25 11:09:18 +00:00
|
|
|
|
2016-06-09 22:11:23 +00:00
|
|
|
sys.exit(1)
|
2017-04-05 21:40:01 +00:00
|
|
|
if options.build_data:
|
2017-04-07 17:38:49 +00:00
|
|
|
merge_build_data(options.build_data, build_data_blob, "application")
|