forked from eden-emu/eden
		
	
		
			
	
	
		
			410 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
		
		
			
		
	
	
			410 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
|  | #!/usr/bin/env python3 | ||
|  | 
 | ||
|  | # Greentea host test script for Mbed TLS on-target test suite testing. | ||
|  | # | ||
|  | # Copyright The Mbed TLS Contributors | ||
|  | # SPDX-License-Identifier: Apache-2.0 OR GPL-2.0-or-later | ||
|  | # | ||
|  | # This file is provided under the Apache License 2.0, or the | ||
|  | # GNU General Public License v2.0 or later. | ||
|  | # | ||
|  | # ********** | ||
|  | # Apache License 2.0: | ||
|  | # | ||
|  | # 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. | ||
|  | # | ||
|  | # ********** | ||
|  | # | ||
|  | # ********** | ||
|  | # GNU General Public License v2.0 or later: | ||
|  | # | ||
|  | # This program is free software; you can redistribute it and/or modify | ||
|  | # it under the terms of the GNU General Public License as published by | ||
|  | # the Free Software Foundation; either version 2 of the License, or | ||
|  | # (at your option) any later version. | ||
|  | # | ||
|  | # 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. | ||
|  | # | ||
|  | # ********** | ||
|  | 
 | ||
|  | 
 | ||
|  | """
 | ||
|  | Mbed TLS on-target test suite tests are implemented as Greentea | ||
|  | tests. Greentea tests are implemented in two parts: target test and | ||
|  | host test. Target test is a C application that is built for the | ||
|  | target platform and executes on the target. Host test is a Python | ||
|  | class derived from mbed_host_tests.BaseHostTest. Target communicates | ||
|  | with the host over serial for the test data and sends back the result. | ||
|  | 
 | ||
|  | Python tool mbedgt (Greentea) is responsible for flashing the test | ||
|  | binary on to the target and dynamically loading this host test module. | ||
|  | 
 | ||
|  | Greentea documentation can be found here: | ||
|  | https://github.com/ARMmbed/greentea | ||
|  | """
 | ||
|  | 
 | ||
|  | 
 | ||
|  | import re | ||
|  | import os | ||
|  | import binascii | ||
|  | 
 | ||
|  | from mbed_host_tests import BaseHostTest, event_callback # pylint: disable=import-error | ||
|  | 
 | ||
|  | 
 | ||
|  | class TestDataParserError(Exception): | ||
|  |     """Indicates error in test data, read from .data file.""" | ||
|  |     pass | ||
|  | 
 | ||
|  | 
 | ||
|  | class TestDataParser: | ||
|  |     """
 | ||
|  |     Parses test name, dependencies, test function name and test parameters | ||
|  |     from the data file. | ||
|  |     """
 | ||
|  | 
 | ||
|  |     def __init__(self): | ||
|  |         """
 | ||
|  |         Constructor | ||
|  |         """
 | ||
|  |         self.tests = [] | ||
|  | 
 | ||
|  |     def parse(self, data_file): | ||
|  |         """
 | ||
|  |         Data file parser. | ||
|  | 
 | ||
|  |         :param data_file: Data file path | ||
|  |         """
 | ||
|  |         with open(data_file, 'r') as data_f: | ||
|  |             self.__parse(data_f) | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def __escaped_split(inp_str, split_char): | ||
|  |         """
 | ||
|  |         Splits inp_str on split_char except when escaped. | ||
|  | 
 | ||
|  |         :param inp_str: String to split | ||
|  |         :param split_char: Split character | ||
|  |         :return: List of splits | ||
|  |         """
 | ||
|  |         split_colon_fn = lambda x: re.sub(r'\\' + split_char, split_char, x) | ||
|  |         if len(split_char) > 1: | ||
|  |             raise ValueError('Expected split character. Found string!') | ||
|  |         out = list(map(split_colon_fn, re.split(r'(?<!\\)' + split_char, inp_str))) | ||
|  |         out = [x for x in out if x] | ||
|  |         return out | ||
|  | 
 | ||
|  |     def __parse(self, data_f): | ||
|  |         """
 | ||
|  |         Parses data file using supplied file object. | ||
|  | 
 | ||
|  |         :param data_f: Data file object | ||
|  |         :return: | ||
|  |         """
 | ||
|  |         for line in data_f: | ||
|  |             line = line.strip() | ||
|  |             if not line: | ||
|  |                 continue | ||
|  |             # Read test name | ||
|  |             name = line | ||
|  | 
 | ||
|  |             # Check dependencies | ||
|  |             dependencies = [] | ||
|  |             line = next(data_f).strip() | ||
|  |             match = re.search('depends_on:(.*)', line) | ||
|  |             if match: | ||
|  |                 dependencies = [int(x) for x in match.group(1).split(':')] | ||
|  |                 line = next(data_f).strip() | ||
|  | 
 | ||
|  |             # Read test vectors | ||
|  |             line = line.replace('\\n', '\n') | ||
|  |             parts = self.__escaped_split(line, ':') | ||
|  |             function_name = int(parts[0]) | ||
|  |             args = parts[1:] | ||
|  |             args_count = len(args) | ||
|  |             if args_count % 2 != 0: | ||
|  |                 err_str_fmt = "Number of test arguments({}) should be even: {}" | ||
|  |                 raise TestDataParserError(err_str_fmt.format(args_count, line)) | ||
|  |             grouped_args = [(args[i * 2], args[(i * 2) + 1]) | ||
|  |                             for i in range(int(len(args)/2))] | ||
|  |             self.tests.append((name, function_name, dependencies, | ||
|  |                                grouped_args)) | ||
|  | 
 | ||
|  |     def get_test_data(self): | ||
|  |         """
 | ||
|  |         Returns test data. | ||
|  |         """
 | ||
|  |         return self.tests | ||
|  | 
 | ||
|  | 
 | ||
|  | class MbedTlsTest(BaseHostTest): | ||
|  |     """
 | ||
|  |     Host test for Mbed TLS unit tests. This script is loaded at | ||
|  |     run time by Greentea for executing Mbed TLS test suites. Each | ||
|  |     communication from the target is received in this object as | ||
|  |     an event, which is then handled by the event handler method | ||
|  |     decorated by the associated event. Ex: @event_callback('GO'). | ||
|  | 
 | ||
|  |     Target test sends requests for dispatching next test. It reads | ||
|  |     tests from the intermediate data file and sends test function | ||
|  |     identifier, dependency identifiers, expression identifiers and | ||
|  |     the test data in binary form. Target test checks dependencies | ||
|  |     , evaluate integer constant expressions and dispatches the test | ||
|  |     function with received test parameters. After test function is | ||
|  |     finished, target sends the result. This class handles the result | ||
|  |     event and prints verdict in the form that Greentea understands. | ||
|  | 
 | ||
|  |     """
 | ||
|  |     # status/error codes from suites/helpers.function | ||
|  |     DEPENDENCY_SUPPORTED = 0 | ||
|  |     KEY_VALUE_MAPPING_FOUND = DEPENDENCY_SUPPORTED | ||
|  |     DISPATCH_TEST_SUCCESS = DEPENDENCY_SUPPORTED | ||
|  | 
 | ||
|  |     KEY_VALUE_MAPPING_NOT_FOUND = -1    # Expression Id not found. | ||
|  |     DEPENDENCY_NOT_SUPPORTED = -2       # Dependency not supported. | ||
|  |     DISPATCH_TEST_FN_NOT_FOUND = -3     # Test function not found. | ||
|  |     DISPATCH_INVALID_TEST_DATA = -4     # Invalid parameter type. | ||
|  |     DISPATCH_UNSUPPORTED_SUITE = -5     # Test suite not supported/enabled. | ||
|  | 
 | ||
|  |     def __init__(self): | ||
|  |         """
 | ||
|  |         Constructor initialises test index to 0. | ||
|  |         """
 | ||
|  |         super(MbedTlsTest, self).__init__() | ||
|  |         self.tests = [] | ||
|  |         self.test_index = -1 | ||
|  |         self.dep_index = 0 | ||
|  |         self.suite_passed = True | ||
|  |         self.error_str = dict() | ||
|  |         self.error_str[self.DEPENDENCY_SUPPORTED] = \ | ||
|  |             'DEPENDENCY_SUPPORTED' | ||
|  |         self.error_str[self.KEY_VALUE_MAPPING_NOT_FOUND] = \ | ||
|  |             'KEY_VALUE_MAPPING_NOT_FOUND' | ||
|  |         self.error_str[self.DEPENDENCY_NOT_SUPPORTED] = \ | ||
|  |             'DEPENDENCY_NOT_SUPPORTED' | ||
|  |         self.error_str[self.DISPATCH_TEST_FN_NOT_FOUND] = \ | ||
|  |             'DISPATCH_TEST_FN_NOT_FOUND' | ||
|  |         self.error_str[self.DISPATCH_INVALID_TEST_DATA] = \ | ||
|  |             'DISPATCH_INVALID_TEST_DATA' | ||
|  |         self.error_str[self.DISPATCH_UNSUPPORTED_SUITE] = \ | ||
|  |             'DISPATCH_UNSUPPORTED_SUITE' | ||
|  | 
 | ||
|  |     def setup(self): | ||
|  |         """
 | ||
|  |         Setup hook implementation. Reads test suite data file and parses out | ||
|  |         tests. | ||
|  |         """
 | ||
|  |         binary_path = self.get_config_item('image_path') | ||
|  |         script_dir = os.path.split(os.path.abspath(__file__))[0] | ||
|  |         suite_name = os.path.splitext(os.path.basename(binary_path))[0] | ||
|  |         data_file = ".".join((suite_name, 'datax')) | ||
|  |         data_file = os.path.join(script_dir, '..', 'mbedtls', | ||
|  |                                  suite_name, data_file) | ||
|  |         if os.path.exists(data_file): | ||
|  |             self.log("Running tests from %s" % data_file) | ||
|  |             parser = TestDataParser() | ||
|  |             parser.parse(data_file) | ||
|  |             self.tests = parser.get_test_data() | ||
|  |             self.print_test_info() | ||
|  |         else: | ||
|  |             self.log("Data file not found: %s" % data_file) | ||
|  |             self.notify_complete(False) | ||
|  | 
 | ||
|  |     def print_test_info(self): | ||
|  |         """
 | ||
|  |         Prints test summary read by Greentea to detect test cases. | ||
|  |         """
 | ||
|  |         self.log('{{__testcase_count;%d}}' % len(self.tests)) | ||
|  |         for name, _, _, _ in self.tests: | ||
|  |             self.log('{{__testcase_name;%s}}' % name) | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def align_32bit(data_bytes): | ||
|  |         """
 | ||
|  |         4 byte aligns input byte array. | ||
|  | 
 | ||
|  |         :return: | ||
|  |         """
 | ||
|  |         data_bytes += bytearray((4 - (len(data_bytes))) % 4) | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def hex_str_bytes(hex_str): | ||
|  |         """
 | ||
|  |         Converts Hex string representation to byte array | ||
|  | 
 | ||
|  |         :param hex_str: Hex in string format. | ||
|  |         :return: Output Byte array | ||
|  |         """
 | ||
|  |         if hex_str[0] != '"' or hex_str[len(hex_str) - 1] != '"': | ||
|  |             raise TestDataParserError("HEX test parameter missing '\"':" | ||
|  |                                       " %s" % hex_str) | ||
|  |         hex_str = hex_str.strip('"') | ||
|  |         if len(hex_str) % 2 != 0: | ||
|  |             raise TestDataParserError("HEX parameter len should be mod of " | ||
|  |                                       "2: %s" % hex_str) | ||
|  | 
 | ||
|  |         data_bytes = binascii.unhexlify(hex_str) | ||
|  |         return data_bytes | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def int32_to_big_endian_bytes(i): | ||
|  |         """
 | ||
|  |         Coverts i to byte array in big endian format. | ||
|  | 
 | ||
|  |         :param i: Input integer | ||
|  |         :return: Output bytes array in big endian or network order | ||
|  |         """
 | ||
|  |         data_bytes = bytearray([((i >> x) & 0xff) for x in [24, 16, 8, 0]]) | ||
|  |         return data_bytes | ||
|  | 
 | ||
|  |     def test_vector_to_bytes(self, function_id, dependencies, parameters): | ||
|  |         """
 | ||
|  |         Converts test vector into a byte array that can be sent to the target. | ||
|  | 
 | ||
|  |         :param function_id: Test Function Identifier | ||
|  |         :param dependencies: Dependency list | ||
|  |         :param parameters: Test function input parameters | ||
|  |         :return: Byte array and its length | ||
|  |         """
 | ||
|  |         data_bytes = bytearray([len(dependencies)]) | ||
|  |         if dependencies: | ||
|  |             data_bytes += bytearray(dependencies) | ||
|  |         data_bytes += bytearray([function_id, len(parameters)]) | ||
|  |         for typ, param in parameters: | ||
|  |             if typ in ('int', 'exp'): | ||
|  |                 i = int(param, 0) | ||
|  |                 data_bytes += b'I' if typ == 'int' else b'E' | ||
|  |                 self.align_32bit(data_bytes) | ||
|  |                 data_bytes += self.int32_to_big_endian_bytes(i) | ||
|  |             elif typ == 'char*': | ||
|  |                 param = param.strip('"') | ||
|  |                 i = len(param) + 1  # + 1 for null termination | ||
|  |                 data_bytes += b'S' | ||
|  |                 self.align_32bit(data_bytes) | ||
|  |                 data_bytes += self.int32_to_big_endian_bytes(i) | ||
|  |                 data_bytes += bytearray(param, encoding='ascii') | ||
|  |                 data_bytes += b'\0'   # Null terminate | ||
|  |             elif typ == 'hex': | ||
|  |                 binary_data = self.hex_str_bytes(param) | ||
|  |                 data_bytes += b'H' | ||
|  |                 self.align_32bit(data_bytes) | ||
|  |                 i = len(binary_data) | ||
|  |                 data_bytes += self.int32_to_big_endian_bytes(i) | ||
|  |                 data_bytes += binary_data | ||
|  |         length = self.int32_to_big_endian_bytes(len(data_bytes)) | ||
|  |         return data_bytes, length | ||
|  | 
 | ||
|  |     def run_next_test(self): | ||
|  |         """
 | ||
|  |         Fetch next test information and execute the test. | ||
|  | 
 | ||
|  |         """
 | ||
|  |         self.test_index += 1 | ||
|  |         self.dep_index = 0 | ||
|  |         if self.test_index < len(self.tests): | ||
|  |             name, function_id, dependencies, args = self.tests[self.test_index] | ||
|  |             self.run_test(name, function_id, dependencies, args) | ||
|  |         else: | ||
|  |             self.notify_complete(self.suite_passed) | ||
|  | 
 | ||
|  |     def run_test(self, name, function_id, dependencies, args): | ||
|  |         """
 | ||
|  |         Execute the test on target by sending next test information. | ||
|  | 
 | ||
|  |         :param name: Test name | ||
|  |         :param function_id: function identifier | ||
|  |         :param dependencies: Dependencies list | ||
|  |         :param args: test parameters | ||
|  |         :return: | ||
|  |         """
 | ||
|  |         self.log("Running: %s" % name) | ||
|  | 
 | ||
|  |         param_bytes, length = self.test_vector_to_bytes(function_id, | ||
|  |                                                         dependencies, args) | ||
|  |         self.send_kv( | ||
|  |             ''.join('{:02x}'.format(x) for x in length), | ||
|  |             ''.join('{:02x}'.format(x) for x in param_bytes) | ||
|  |         ) | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def get_result(value): | ||
|  |         """
 | ||
|  |         Converts result from string type to integer | ||
|  |         :param value: Result code in string | ||
|  |         :return: Integer result code. Value is from the test status | ||
|  |                  constants defined under the MbedTlsTest class. | ||
|  |         """
 | ||
|  |         try: | ||
|  |             return int(value) | ||
|  |         except ValueError: | ||
|  |             ValueError("Result should return error number. " | ||
|  |                        "Instead received %s" % value) | ||
|  | 
 | ||
|  |     @event_callback('GO') | ||
|  |     def on_go(self, _key, _value, _timestamp): | ||
|  |         """
 | ||
|  |         Sent by the target to start first test. | ||
|  | 
 | ||
|  |         :param _key: Event key | ||
|  |         :param _value: Value. ignored | ||
|  |         :param _timestamp: Timestamp ignored. | ||
|  |         :return: | ||
|  |         """
 | ||
|  |         self.run_next_test() | ||
|  | 
 | ||
|  |     @event_callback("R") | ||
|  |     def on_result(self, _key, value, _timestamp): | ||
|  |         """
 | ||
|  |         Handle result. Prints test start, finish required by Greentea | ||
|  |         to detect test execution. | ||
|  | 
 | ||
|  |         :param _key: Event key | ||
|  |         :param value: Value. ignored | ||
|  |         :param _timestamp: Timestamp ignored. | ||
|  |         :return: | ||
|  |         """
 | ||
|  |         int_val = self.get_result(value) | ||
|  |         name, _, _, _ = self.tests[self.test_index] | ||
|  |         self.log('{{__testcase_start;%s}}' % name) | ||
|  |         self.log('{{__testcase_finish;%s;%d;%d}}' % (name, int_val == 0, | ||
|  |                                                      int_val != 0)) | ||
|  |         if int_val != 0: | ||
|  |             self.suite_passed = False | ||
|  |         self.run_next_test() | ||
|  | 
 | ||
|  |     @event_callback("F") | ||
|  |     def on_failure(self, _key, value, _timestamp): | ||
|  |         """
 | ||
|  |         Handles test execution failure. That means dependency not supported or | ||
|  |         Test function not supported. Hence marking test as skipped. | ||
|  | 
 | ||
|  |         :param _key: Event key | ||
|  |         :param value: Value. ignored | ||
|  |         :param _timestamp: Timestamp ignored. | ||
|  |         :return: | ||
|  |         """
 | ||
|  |         int_val = self.get_result(value) | ||
|  |         if int_val in self.error_str: | ||
|  |             err = self.error_str[int_val] | ||
|  |         else: | ||
|  |             err = 'Unknown error' | ||
|  |         # For skip status, do not write {{__testcase_finish;...}} | ||
|  |         self.log("Error: %s" % err) | ||
|  |         self.run_next_test() |