| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 1 | #!/usr/bin/env python3 | 
|  | 2 |  | 
| Andrzej Kurek | ffbc8f5 | 2022-10-17 08:34:40 -0400 | [diff] [blame] | 3 | # Copyright (c) 2022, Arm Limited, All Rights Reserved. | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 4 | # SPDX-License-Identifier: Apache-2.0 | 
|  | 5 | # | 
|  | 6 | # Licensed under the Apache License, Version 2.0 (the "License"); you may | 
|  | 7 | # not use this file except in compliance with the License. | 
|  | 8 | # You may obtain a copy of the License at | 
|  | 9 | # | 
|  | 10 | # http://www.apache.org/licenses/LICENSE-2.0 | 
|  | 11 | # | 
|  | 12 | # Unless required by applicable law or agreed to in writing, software | 
|  | 13 | # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT | 
|  | 14 | # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | 15 | # See the License for the specific language governing permissions and | 
|  | 16 | # limitations under the License. | 
|  | 17 | # | 
|  | 18 | # This file is part of Mbed TLS (https://tls.mbed.org) | 
|  | 19 |  | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 20 | """ | 
| Andrzej Kurek | 110fc48 | 2022-10-09 05:29:44 -0400 | [diff] [blame] | 21 | Test Mbed TLS with a subset of algorithms. | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 22 |  | 
| Andrzej Kurek | 110fc48 | 2022-10-09 05:29:44 -0400 | [diff] [blame] | 23 | This script can be divided into several steps: | 
|  | 24 |  | 
| Andrzej Kurek | 467a0f2 | 2022-10-20 06:15:06 -0400 | [diff] [blame] | 25 | First, include/mbedtls/config.h or a different config file passed | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 26 | in the arguments is parsed to extract any configuration options (using config.py). | 
| Andrzej Kurek | 110fc48 | 2022-10-09 05:29:44 -0400 | [diff] [blame] | 27 |  | 
|  | 28 | Then, test domains (groups of jobs, tests) are built based on predefined data | 
|  | 29 | collected in the DomainData class. Here, each domain has five major traits: | 
| Andrzej Kurek | ffbc8f5 | 2022-10-17 08:34:40 -0400 | [diff] [blame] | 30 | - domain name, can be used to run only specific tests via command-line; | 
| Andrzej Kurek | 110fc48 | 2022-10-09 05:29:44 -0400 | [diff] [blame] | 31 | - configuration building method, described in detail below; | 
|  | 32 | - list of symbols passed to the configuration building method; | 
|  | 33 | - commands to be run on each job (only build, build and test, or any other custom); | 
|  | 34 | - optional list of symbols to be excluded from testing. | 
|  | 35 |  | 
|  | 36 | The configuration building method can be one of the three following: | 
|  | 37 |  | 
|  | 38 | - ComplementaryDomain - build a job for each passed symbol by disabling a single | 
|  | 39 | symbol and its reverse dependencies (defined in REVERSE_DEPENDENCIES); | 
|  | 40 |  | 
|  | 41 | - ExclusiveDomain - build a job where, for each passed symbol, only this particular | 
|  | 42 | one is defined and other symbols from the list are unset. For each job look for | 
|  | 43 | any non-standard symbols to set/unset in EXCLUSIVE_GROUPS. These are usually not | 
|  | 44 | direct dependencies, but rather non-trivial results of other configs missing. Then | 
|  | 45 | look for any unset symbols and handle their reverse dependencies. | 
| Andrzej Kurek | 110fc48 | 2022-10-09 05:29:44 -0400 | [diff] [blame] | 46 |  | 
|  | 47 | - DualDomain - combination of the two above - both complementary and exclusive domain | 
|  | 48 | job generation code will be run. Currently only used for hashes. | 
|  | 49 |  | 
|  | 50 | Lastly, the collected jobs are executed and (optionally) tested, with | 
|  | 51 | error reporting and coloring as configured in options. Each test starts with | 
|  | 52 | a full config without a couple of slowing down or unnecessary options | 
|  | 53 | (see set_reference_config), then the specific job config is derived. | 
|  | 54 | """ | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 55 | import argparse | 
|  | 56 | import os | 
|  | 57 | import re | 
|  | 58 | import shutil | 
|  | 59 | import subprocess | 
|  | 60 | import sys | 
|  | 61 | import traceback | 
| Andrzej Kurek | 2432dc2 | 2023-01-24 07:40:42 -0500 | [diff] [blame] | 62 | from typing import Union | 
|  | 63 |  | 
| Andrzej Kurek | 2e1aeb1 | 2023-01-23 07:19:22 -0500 | [diff] [blame] | 64 | # Add the Mbed TLS Python library directory to the module search path | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 65 | import scripts_path # pylint: disable=unused-import | 
|  | 66 | import config | 
|  | 67 |  | 
| Andrzej Kurek | b95ba9a | 2022-10-04 15:02:41 -0400 | [diff] [blame] | 68 | class Colors: # pylint: disable=too-few-public-methods | 
| Gilles Peskine | fd1d69c | 2019-01-29 18:48:48 +0100 | [diff] [blame] | 69 | """Minimalistic support for colored output. | 
|  | 70 | Each field of an object of this class is either None if colored output | 
|  | 71 | is not possible or not desired, or a pair of strings (start, stop) such | 
|  | 72 | that outputting start switches the text color to the desired color and | 
|  | 73 | stop switches the text color back to the default.""" | 
|  | 74 | red = None | 
|  | 75 | green = None | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 76 | cyan = None | 
| Gilles Peskine | fd1d69c | 2019-01-29 18:48:48 +0100 | [diff] [blame] | 77 | bold_red = None | 
|  | 78 | bold_green = None | 
|  | 79 | def __init__(self, options=None): | 
| Andrzej Kurek | b95ba9a | 2022-10-04 15:02:41 -0400 | [diff] [blame] | 80 | """Initialize color profile according to passed options.""" | 
| Gilles Peskine | fd1d69c | 2019-01-29 18:48:48 +0100 | [diff] [blame] | 81 | if not options or options.color in ['no', 'never']: | 
|  | 82 | want_color = False | 
|  | 83 | elif options.color in ['yes', 'always']: | 
|  | 84 | want_color = True | 
|  | 85 | else: | 
|  | 86 | want_color = sys.stderr.isatty() | 
|  | 87 | if want_color: | 
|  | 88 | # Assume ANSI compatible terminal | 
|  | 89 | normal = '\033[0m' | 
|  | 90 | self.red = ('\033[31m', normal) | 
|  | 91 | self.green = ('\033[32m', normal) | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 92 | self.cyan = ('\033[36m', normal) | 
| Gilles Peskine | fd1d69c | 2019-01-29 18:48:48 +0100 | [diff] [blame] | 93 | self.bold_red = ('\033[1;31m', normal) | 
|  | 94 | self.bold_green = ('\033[1;32m', normal) | 
|  | 95 | NO_COLORS = Colors(None) | 
|  | 96 |  | 
|  | 97 | def log_line(text, prefix='depends.py:', suffix='', color=None): | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 98 | """Print a status message.""" | 
| Andrzej Kurek | b95ba9a | 2022-10-04 15:02:41 -0400 | [diff] [blame] | 99 | if color is not None: | 
| Gilles Peskine | fd1d69c | 2019-01-29 18:48:48 +0100 | [diff] [blame] | 100 | prefix = color[0] + prefix | 
|  | 101 | suffix = suffix + color[1] | 
|  | 102 | sys.stderr.write(prefix + ' ' + text + suffix + '\n') | 
| Gilles Peskine | e6a60db | 2019-01-29 18:42:55 +0100 | [diff] [blame] | 103 | sys.stderr.flush() | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 104 |  | 
| Gilles Peskine | d43ce2b | 2019-01-29 18:46:34 +0100 | [diff] [blame] | 105 | def log_command(cmd): | 
|  | 106 | """Print a trace of the specified command. | 
|  | 107 | cmd is a list of strings: a command name and its arguments.""" | 
|  | 108 | log_line(' '.join(cmd), prefix='+') | 
|  | 109 |  | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 110 | def backup_config(options): | 
| Andrzej Kurek | 467a0f2 | 2022-10-20 06:15:06 -0400 | [diff] [blame] | 111 | """Back up the library configuration file (config.h). | 
| Gilles Peskine | 88e8dd6 | 2019-01-29 18:52:16 +0100 | [diff] [blame] | 112 | If the backup file already exists, it is presumed to be the desired backup, | 
|  | 113 | so don't make another backup.""" | 
|  | 114 | if os.path.exists(options.config_backup): | 
|  | 115 | options.own_backup = False | 
|  | 116 | else: | 
|  | 117 | options.own_backup = True | 
|  | 118 | shutil.copy(options.config, options.config_backup) | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 119 |  | 
| Gilles Peskine | 88e8dd6 | 2019-01-29 18:52:16 +0100 | [diff] [blame] | 120 | def restore_config(options): | 
| Andrzej Kurek | 467a0f2 | 2022-10-20 06:15:06 -0400 | [diff] [blame] | 121 | """Restore the library configuration file (config.h). | 
| Gilles Peskine | 88e8dd6 | 2019-01-29 18:52:16 +0100 | [diff] [blame] | 122 | Remove the backup file if it was saved earlier.""" | 
|  | 123 | if options.own_backup: | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 124 | shutil.move(options.config_backup, options.config) | 
|  | 125 | else: | 
|  | 126 | shutil.copy(options.config_backup, options.config) | 
| Gilles Peskine | 88e8dd6 | 2019-01-29 18:52:16 +0100 | [diff] [blame] | 127 |  | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 128 | def option_exists(conf, option): | 
| Andrzej Kurek | 3ebe7d6 | 2023-02-06 10:48:43 +0100 | [diff] [blame] | 129 | return option in conf.settings | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 130 |  | 
| Andrzej Kurek | 2432dc2 | 2023-01-24 07:40:42 -0500 | [diff] [blame] | 131 | def set_config_option_value(conf, option, colors, value: Union[bool, str]): | 
|  | 132 | """Set/unset a configuration option, optionally specifying a value. | 
|  | 133 | value can be either True/False (set/unset config option), or a string, | 
|  | 134 | which will make a symbol defined with a certain value.""" | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 135 | if not option_exists(conf, option): | 
|  | 136 | log_line('Symbol {} was not found in {}'.format(option, conf.filename), color=colors.red) | 
|  | 137 | return False | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 138 |  | 
| Andrzej Kurek | 2e1aeb1 | 2023-01-23 07:19:22 -0500 | [diff] [blame] | 139 | if value is False: | 
|  | 140 | log_command(['config.py', 'unset', option]) | 
|  | 141 | conf.unset(option) | 
| Andrzej Kurek | 3e7666b | 2023-02-06 10:49:46 +0100 | [diff] [blame] | 142 | elif value is True: | 
|  | 143 | log_command(['config.py', 'set', option]) | 
|  | 144 | conf.set(option) | 
| Andrzej Kurek | 2e1aeb1 | 2023-01-23 07:19:22 -0500 | [diff] [blame] | 145 | else: | 
| Andrzej Kurek | 3e7666b | 2023-02-06 10:49:46 +0100 | [diff] [blame] | 146 | log_command(['config.py', 'set', option, value]) | 
|  | 147 | conf.set(option, value) | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 148 | return True | 
|  | 149 |  | 
|  | 150 | def set_reference_config(conf, options, colors): | 
| Andrzej Kurek | 467a0f2 | 2022-10-20 06:15:06 -0400 | [diff] [blame] | 151 | """Change the library configuration file (config.h) to the reference state. | 
| Andrzej Kurek | b95ba9a | 2022-10-04 15:02:41 -0400 | [diff] [blame] | 152 | The reference state is the one from which the tested configurations are | 
|  | 153 | derived.""" | 
| Andrzej Kurek | 8b7a157 | 2022-10-14 07:06:43 -0400 | [diff] [blame] | 154 | # Turn off options that are not relevant to the tests and slow them down. | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 155 | log_command(['config.py', 'full']) | 
|  | 156 | conf.adapt(config.full_adapter) | 
| Andrzej Kurek | 2e1aeb1 | 2023-01-23 07:19:22 -0500 | [diff] [blame] | 157 | set_config_option_value(conf, 'MBEDTLS_TEST_HOOKS', colors, False) | 
| Andrzej Kurek | 2b44a92 | 2022-10-24 10:41:20 -0400 | [diff] [blame] | 158 | if options.unset_use_psa: | 
| Andrzej Kurek | 2e1aeb1 | 2023-01-23 07:19:22 -0500 | [diff] [blame] | 159 | set_config_option_value(conf, 'MBEDTLS_USE_PSA_CRYPTO', colors, False) | 
| Andrzej Kurek | b95ba9a | 2022-10-04 15:02:41 -0400 | [diff] [blame] | 160 |  | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 161 | class Job: | 
|  | 162 | """A job builds the library in a specific configuration and runs some tests.""" | 
|  | 163 | def __init__(self, name, config_settings, commands): | 
|  | 164 | """Build a job object. | 
|  | 165 | The job uses the configuration described by config_settings. This is a | 
|  | 166 | dictionary where the keys are preprocessor symbols and the values are | 
|  | 167 | booleans or strings. A boolean indicates whether or not to #define the | 
|  | 168 | symbol. With a string, the symbol is #define'd to that value. | 
|  | 169 | After setting the configuration, the job runs the programs specified by | 
|  | 170 | commands. This is a list of lists of strings; each list of string is a | 
|  | 171 | command name and its arguments and is passed to subprocess.call with | 
|  | 172 | shell=False.""" | 
|  | 173 | self.name = name | 
|  | 174 | self.config_settings = config_settings | 
|  | 175 | self.commands = commands | 
|  | 176 |  | 
| Gilles Peskine | fd1d69c | 2019-01-29 18:48:48 +0100 | [diff] [blame] | 177 | def announce(self, colors, what): | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 178 | '''Announce the start or completion of a job. | 
|  | 179 | If what is None, announce the start of the job. | 
|  | 180 | If what is True, announce that the job has passed. | 
|  | 181 | If what is False, announce that the job has failed.''' | 
|  | 182 | if what is True: | 
| Gilles Peskine | fd1d69c | 2019-01-29 18:48:48 +0100 | [diff] [blame] | 183 | log_line(self.name + ' PASSED', color=colors.green) | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 184 | elif what is False: | 
| Gilles Peskine | fd1d69c | 2019-01-29 18:48:48 +0100 | [diff] [blame] | 185 | log_line(self.name + ' FAILED', color=colors.red) | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 186 | else: | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 187 | log_line('starting ' + self.name, color=colors.cyan) | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 188 |  | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 189 | def configure(self, conf, options, colors): | 
| Andrzej Kurek | 8b7a157 | 2022-10-14 07:06:43 -0400 | [diff] [blame] | 190 | '''Set library configuration options as required for the job.''' | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 191 | set_reference_config(conf, options, colors) | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 192 | for key, value in sorted(self.config_settings.items()): | 
| Andrzej Kurek | 2e1aeb1 | 2023-01-23 07:19:22 -0500 | [diff] [blame] | 193 | ret = set_config_option_value(conf, key, colors, value) | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 194 | if ret is False: | 
|  | 195 | return False | 
|  | 196 | return True | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 197 |  | 
|  | 198 | def test(self, options): | 
|  | 199 | '''Run the job's build and test commands. | 
|  | 200 | Return True if all the commands succeed and False otherwise. | 
|  | 201 | If options.keep_going is false, stop as soon as one command fails. Otherwise | 
|  | 202 | run all the commands, except that if the first command fails, none of the | 
|  | 203 | other commands are run (typically, the first command is a build command | 
|  | 204 | and subsequent commands are tests that cannot run if the build failed).''' | 
|  | 205 | built = False | 
|  | 206 | success = True | 
|  | 207 | for command in self.commands: | 
| Gilles Peskine | d43ce2b | 2019-01-29 18:46:34 +0100 | [diff] [blame] | 208 | log_command(command) | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 209 | ret = subprocess.call(command) | 
|  | 210 | if ret != 0: | 
|  | 211 | if command[0] not in ['make', options.make_command]: | 
|  | 212 | log_line('*** [{}] Error {}'.format(' '.join(command), ret)) | 
|  | 213 | if not options.keep_going or not built: | 
|  | 214 | return False | 
|  | 215 | success = False | 
|  | 216 | built = True | 
|  | 217 | return success | 
|  | 218 |  | 
|  | 219 | # SSL/TLS versions up to 1.1 and corresponding options. These require | 
|  | 220 | # both MD5 and SHA-1. | 
| Andrzej Kurek | fb3e27e | 2022-10-04 16:22:22 -0400 | [diff] [blame] | 221 | SSL_PRE_1_2_DEPENDENCIES = ['MBEDTLS_SSL_CBC_RECORD_SPLITTING', | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 222 | 'MBEDTLS_SSL_PROTO_SSL3', | 
|  | 223 | 'MBEDTLS_SSL_PROTO_TLS1', | 
|  | 224 | 'MBEDTLS_SSL_PROTO_TLS1_1'] | 
|  | 225 |  | 
|  | 226 | # If the configuration option A requires B, make sure that | 
| Andrzej Kurek | fb3e27e | 2022-10-04 16:22:22 -0400 | [diff] [blame] | 227 | # B in REVERSE_DEPENDENCIES[A]. | 
| Gilles Peskine | b81f406 | 2019-01-29 19:30:40 +0100 | [diff] [blame] | 228 | # All the information here should be contained in check_config.h. This | 
|  | 229 | # file includes a copy because it changes rarely and it would be a pain | 
|  | 230 | # to extract automatically. | 
| Andrzej Kurek | fb3e27e | 2022-10-04 16:22:22 -0400 | [diff] [blame] | 231 | REVERSE_DEPENDENCIES = { | 
| Gilles Peskine | 3ce0e32 | 2019-01-29 23:12:28 +0100 | [diff] [blame] | 232 | 'MBEDTLS_AES_C': ['MBEDTLS_CTR_DRBG_C', | 
| Andrzej Kurek | 9068625 | 2022-09-28 03:17:56 -0400 | [diff] [blame] | 233 | 'MBEDTLS_NIST_KW_C'], | 
| Gilles Peskine | 3ce0e32 | 2019-01-29 23:12:28 +0100 | [diff] [blame] | 234 | 'MBEDTLS_CHACHA20_C': ['MBEDTLS_CHACHAPOLY_C'], | 
| Andrzej Kurek | 9068625 | 2022-09-28 03:17:56 -0400 | [diff] [blame] | 235 | 'MBEDTLS_ECDSA_C': ['MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED', | 
|  | 236 | 'MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA_ENABLED'], | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 237 | 'MBEDTLS_ECP_C': ['MBEDTLS_ECDSA_C', | 
|  | 238 | 'MBEDTLS_ECDH_C', | 
|  | 239 | 'MBEDTLS_ECJPAKE_C', | 
| Manuel Pégourié-Gonnard | 3dc7f23 | 2022-12-06 13:20:06 +0100 | [diff] [blame] | 240 | 'MBEDTLS_ECP_RESTARTABLE', | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 241 | 'MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA_ENABLED', | 
|  | 242 | 'MBEDTLS_KEY_EXCHANGE_ECDH_RSA_ENABLED', | 
|  | 243 | 'MBEDTLS_KEY_EXCHANGE_ECDHE_PSK_ENABLED', | 
|  | 244 | 'MBEDTLS_KEY_EXCHANGE_ECDHE_RSA_ENABLED', | 
| Andrzej Kurek | 9068625 | 2022-09-28 03:17:56 -0400 | [diff] [blame] | 245 | 'MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED', | 
|  | 246 | 'MBEDTLS_KEY_EXCHANGE_ECJPAKE_ENABLED'], | 
| Gilles Peskine | b81f406 | 2019-01-29 19:30:40 +0100 | [diff] [blame] | 247 | 'MBEDTLS_ECP_DP_SECP256R1_ENABLED': ['MBEDTLS_KEY_EXCHANGE_ECJPAKE_ENABLED'], | 
| Andrzej Kurek | fb3e27e | 2022-10-04 16:22:22 -0400 | [diff] [blame] | 248 | 'MBEDTLS_MD5_C': SSL_PRE_1_2_DEPENDENCIES, | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 249 | 'MBEDTLS_PKCS1_V21': ['MBEDTLS_X509_RSASSA_PSS_SUPPORT'], | 
|  | 250 | 'MBEDTLS_PKCS1_V15': ['MBEDTLS_KEY_EXCHANGE_DHE_RSA_ENABLED', | 
|  | 251 | 'MBEDTLS_KEY_EXCHANGE_ECDHE_RSA_ENABLED', | 
|  | 252 | 'MBEDTLS_KEY_EXCHANGE_RSA_PSK_ENABLED', | 
|  | 253 | 'MBEDTLS_KEY_EXCHANGE_RSA_ENABLED'], | 
|  | 254 | 'MBEDTLS_RSA_C': ['MBEDTLS_X509_RSASSA_PSS_SUPPORT', | 
|  | 255 | 'MBEDTLS_KEY_EXCHANGE_DHE_RSA_ENABLED', | 
|  | 256 | 'MBEDTLS_KEY_EXCHANGE_ECDHE_RSA_ENABLED', | 
|  | 257 | 'MBEDTLS_KEY_EXCHANGE_RSA_PSK_ENABLED', | 
| Andrzej Kurek | 9068625 | 2022-09-28 03:17:56 -0400 | [diff] [blame] | 258 | 'MBEDTLS_KEY_EXCHANGE_RSA_ENABLED', | 
|  | 259 | 'MBEDTLS_KEY_EXCHANGE_ECDH_RSA_ENABLED'], | 
| Andrzej Kurek | fb3e27e | 2022-10-04 16:22:22 -0400 | [diff] [blame] | 260 | 'MBEDTLS_SHA1_C': SSL_PRE_1_2_DEPENDENCIES, | 
| Gilles Peskine | b81f406 | 2019-01-29 19:30:40 +0100 | [diff] [blame] | 261 | 'MBEDTLS_SHA256_C': ['MBEDTLS_KEY_EXCHANGE_ECJPAKE_ENABLED', | 
| Andrzej Kurek | b790c93 | 2023-02-15 15:19:37 -0500 | [diff] [blame] | 262 | 'MBEDTLS_ENTROPY_FORCE_SHA256'], | 
| Andrzej Kurek | 9068625 | 2022-09-28 03:17:56 -0400 | [diff] [blame] | 263 | 'MBEDTLS_X509_RSASSA_PSS_SUPPORT': [] | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 264 | } | 
|  | 265 |  | 
| Andrzej Kurek | 9068625 | 2022-09-28 03:17:56 -0400 | [diff] [blame] | 266 | # If an option is tested in an exclusive test, alter the following defines. | 
| Andrzej Kurek | 110fc48 | 2022-10-09 05:29:44 -0400 | [diff] [blame] | 267 | # These are not necessarily dependencies, but just minimal required changes | 
| Andrzej Kurek | 9068625 | 2022-09-28 03:17:56 -0400 | [diff] [blame] | 268 | # if a given define is the only one enabled from an exclusive group. | 
| Andrzej Kurek | fb3e27e | 2022-10-04 16:22:22 -0400 | [diff] [blame] | 269 | EXCLUSIVE_GROUPS = { | 
| Andrzej Kurek | 467a0f2 | 2022-10-20 06:15:06 -0400 | [diff] [blame] | 270 | 'MBEDTLS_SHA512_C': ['-MBEDTLS_SSL_PROTO_TLS1_3_EXPERIMENTAL'], | 
| Andrzej Kurek | f53d0ba | 2022-11-23 05:54:46 -0500 | [diff] [blame] | 271 | 'MBEDTLS_SHA512_NO_SHA384': ['+MBEDTLS_SHA512_C', | 
|  | 272 | '-MBEDTLS_SSL_PROTO_TLS1_2', | 
|  | 273 | '-MBEDTLS_SSL_PROTO_DTLS', | 
|  | 274 | '-MBEDTLS_SSL_TLS_C', | 
|  | 275 | '-MBEDTLS_SSL_CLI_C', | 
|  | 276 | '-MBEDTLS_SSL_SRV_C', | 
|  | 277 | '-MBEDTLS_SSL_DTLS_HELLO_VERIFY', | 
|  | 278 | '-MBEDTLS_SSL_DTLS_ANTI_REPLAY', | 
|  | 279 | '-MBEDTLS_SSL_DTLS_CONNECTION_ID', | 
|  | 280 | '-MBEDTLS_SSL_DTLS_BADMAC_LIMIT', | 
|  | 281 | '-MBEDTLS_SSL_ENCRYPT_THEN_MAC', | 
|  | 282 | '-MBEDTLS_SSL_EXTENDED_MASTER_SECRET', | 
|  | 283 | '-MBEDTLS_SSL_DTLS_SRTP', | 
|  | 284 | '-MBEDTLS_SSL_DTLS_CLIENT_PORT_REUSE'], | 
| Andrzej Kurek | 9cbdf10 | 2022-10-14 08:09:16 -0400 | [diff] [blame] | 285 | 'MBEDTLS_ECP_DP_CURVE448_ENABLED': ['-MBEDTLS_ECDSA_C', | 
|  | 286 | '-MBEDTLS_ECDSA_DETERMINISTIC', | 
|  | 287 | '-MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED', | 
|  | 288 | '-MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA_ENABLED', | 
|  | 289 | '-MBEDTLS_ECJPAKE_C', | 
|  | 290 | '-MBEDTLS_KEY_EXCHANGE_ECJPAKE_ENABLED'], | 
|  | 291 | 'MBEDTLS_ECP_DP_CURVE25519_ENABLED': ['-MBEDTLS_ECDSA_C', | 
|  | 292 | '-MBEDTLS_ECDSA_DETERMINISTIC', | 
|  | 293 | '-MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED', | 
|  | 294 | '-MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA_ENABLED', | 
|  | 295 | '-MBEDTLS_ECJPAKE_C', | 
|  | 296 | '-MBEDTLS_KEY_EXCHANGE_ECJPAKE_ENABLED'], | 
|  | 297 | 'MBEDTLS_ARIA_C': ['-MBEDTLS_CMAC_C'], | 
| Andrzej Kurek | 467a0f2 | 2022-10-20 06:15:06 -0400 | [diff] [blame] | 298 | 'MBEDTLS_ARC4_C': ['-MBEDTLS_CMAC_C', | 
|  | 299 | '-MBEDTLS_CCM_C', | 
|  | 300 | '-MBEDTLS_SSL_TICKET_C', | 
|  | 301 | '-MBEDTLS_SSL_CONTEXT_SERIALIZATION', | 
|  | 302 | '-MBEDTLS_GCM_C'], | 
|  | 303 | 'MBEDTLS_BLOWFISH_C': ['-MBEDTLS_CMAC_C', | 
|  | 304 | '-MBEDTLS_CCM_C', | 
|  | 305 | '-MBEDTLS_SSL_TICKET_C', | 
|  | 306 | '-MBEDTLS_SSL_CONTEXT_SERIALIZATION', | 
|  | 307 | '-MBEDTLS_GCM_C'], | 
| Andrzej Kurek | 9cbdf10 | 2022-10-14 08:09:16 -0400 | [diff] [blame] | 308 | 'MBEDTLS_CAMELLIA_C': ['-MBEDTLS_CMAC_C'], | 
|  | 309 | 'MBEDTLS_CHACHA20_C': ['-MBEDTLS_CMAC_C', '-MBEDTLS_CCM_C', '-MBEDTLS_GCM_C'], | 
|  | 310 | 'MBEDTLS_DES_C': ['-MBEDTLS_CCM_C', | 
|  | 311 | '-MBEDTLS_GCM_C', | 
|  | 312 | '-MBEDTLS_SSL_TICKET_C', | 
|  | 313 | '-MBEDTLS_SSL_CONTEXT_SERIALIZATION'], | 
| Andrzej Kurek | 9068625 | 2022-09-28 03:17:56 -0400 | [diff] [blame] | 314 | } | 
|  | 315 | def handle_exclusive_groups(config_settings, symbol): | 
|  | 316 | """For every symbol tested in an exclusive group check if there are other | 
|  | 317 | defines to be altered. """ | 
| Andrzej Kurek | fb3e27e | 2022-10-04 16:22:22 -0400 | [diff] [blame] | 318 | for dep in EXCLUSIVE_GROUPS.get(symbol, []): | 
| Andrzej Kurek | 9cbdf10 | 2022-10-14 08:09:16 -0400 | [diff] [blame] | 319 | unset = dep.startswith('-') | 
|  | 320 | dep = dep[1:] | 
| Andrzej Kurek | 9068625 | 2022-09-28 03:17:56 -0400 | [diff] [blame] | 321 | config_settings[dep] = not unset | 
|  | 322 |  | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 323 | def turn_off_dependencies(config_settings): | 
|  | 324 | """For every option turned off config_settings, also turn off what depends on it. | 
|  | 325 | An option O is turned off if config_settings[O] is False.""" | 
|  | 326 | for key, value in sorted(config_settings.items()): | 
|  | 327 | if value is not False: | 
|  | 328 | continue | 
| Andrzej Kurek | fb3e27e | 2022-10-04 16:22:22 -0400 | [diff] [blame] | 329 | for dep in REVERSE_DEPENDENCIES.get(key, []): | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 330 | config_settings[dep] = False | 
|  | 331 |  | 
| Andrzej Kurek | 068a73f | 2022-10-06 18:52:44 -0400 | [diff] [blame] | 332 | class BaseDomain: # pylint: disable=too-few-public-methods, unused-argument | 
|  | 333 | """A base class for all domains.""" | 
|  | 334 | def __init__(self, symbols, commands, exclude): | 
|  | 335 | """Initialize the jobs container""" | 
|  | 336 | self.jobs = [] | 
|  | 337 |  | 
|  | 338 | class ExclusiveDomain(BaseDomain): # pylint: disable=too-few-public-methods | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 339 | """A domain consisting of a set of conceptually-equivalent settings. | 
|  | 340 | Establish a list of configuration symbols. For each symbol, run a test job | 
| Andrzej Kurek | 2e105b5 | 2022-10-06 16:57:38 -0400 | [diff] [blame] | 341 | with this symbol set and the others unset.""" | 
| Gilles Peskine | 3dd0dab | 2019-01-29 18:56:03 +0100 | [diff] [blame] | 342 | def __init__(self, symbols, commands, exclude=None): | 
|  | 343 | """Build a domain for the specified list of configuration symbols. | 
| Andrzej Kurek | 2e105b5 | 2022-10-06 16:57:38 -0400 | [diff] [blame] | 344 | The domain contains a set of jobs that enable one of the elements | 
|  | 345 | of symbols and disable the others. | 
| Gilles Peskine | 3dd0dab | 2019-01-29 18:56:03 +0100 | [diff] [blame] | 346 | Each job runs the specified commands. | 
|  | 347 | If exclude is a regular expression, skip generated jobs whose description | 
|  | 348 | would match this regular expression.""" | 
| Andrzej Kurek | 068a73f | 2022-10-06 18:52:44 -0400 | [diff] [blame] | 349 | super().__init__(symbols, commands, exclude) | 
| Andrzej Kurek | 2e105b5 | 2022-10-06 16:57:38 -0400 | [diff] [blame] | 350 | base_config_settings = {} | 
|  | 351 | for symbol in symbols: | 
|  | 352 | base_config_settings[symbol] = False | 
|  | 353 | for symbol in symbols: | 
|  | 354 | description = symbol | 
|  | 355 | if exclude and re.match(exclude, description): | 
|  | 356 | continue | 
|  | 357 | config_settings = base_config_settings.copy() | 
|  | 358 | config_settings[symbol] = True | 
|  | 359 | handle_exclusive_groups(config_settings, symbol) | 
|  | 360 | turn_off_dependencies(config_settings) | 
|  | 361 | job = Job(description, config_settings, commands) | 
|  | 362 | self.jobs.append(job) | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 363 |  | 
| Andrzej Kurek | 068a73f | 2022-10-06 18:52:44 -0400 | [diff] [blame] | 364 | class ComplementaryDomain(BaseDomain): # pylint: disable=too-few-public-methods | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 365 | """A domain consisting of a set of loosely-related settings. | 
|  | 366 | Establish a list of configuration symbols. For each symbol, run a test job | 
| Andrzej Kurek | 8b7a157 | 2022-10-14 07:06:43 -0400 | [diff] [blame] | 367 | with this symbol unset. | 
|  | 368 | If exclude is a regular expression, skip generated jobs whose description | 
|  | 369 | would match this regular expression.""" | 
| Andrzej Kurek | 068a73f | 2022-10-06 18:52:44 -0400 | [diff] [blame] | 370 | def __init__(self, symbols, commands, exclude=None): | 
| Gilles Peskine | 3dd0dab | 2019-01-29 18:56:03 +0100 | [diff] [blame] | 371 | """Build a domain for the specified list of configuration symbols. | 
|  | 372 | Each job in the domain disables one of the specified symbols. | 
|  | 373 | Each job runs the specified commands.""" | 
| Andrzej Kurek | 068a73f | 2022-10-06 18:52:44 -0400 | [diff] [blame] | 374 | super().__init__(symbols, commands, exclude) | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 375 | for symbol in symbols: | 
|  | 376 | description = '!' + symbol | 
| Andrzej Kurek | 068a73f | 2022-10-06 18:52:44 -0400 | [diff] [blame] | 377 | if exclude and re.match(exclude, description): | 
|  | 378 | continue | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 379 | config_settings = {symbol: False} | 
|  | 380 | turn_off_dependencies(config_settings) | 
|  | 381 | job = Job(description, config_settings, commands) | 
|  | 382 | self.jobs.append(job) | 
|  | 383 |  | 
| Andrzej Kurek | 068a73f | 2022-10-06 18:52:44 -0400 | [diff] [blame] | 384 | class DualDomain(ExclusiveDomain, ComplementaryDomain): # pylint: disable=too-few-public-methods | 
| Andrzej Kurek | 8b7a157 | 2022-10-14 07:06:43 -0400 | [diff] [blame] | 385 | """A domain that contains both the ExclusiveDomain and BaseDomain tests. | 
| Andrzej Kurek | 0325ced | 2022-10-18 09:37:59 -0400 | [diff] [blame] | 386 | Both parent class __init__ calls are performed in any order and | 
| Andrzej Kurek | 8b7a157 | 2022-10-14 07:06:43 -0400 | [diff] [blame] | 387 | each call adds respective jobs. The job array initialization is done once in | 
|  | 388 | BaseDomain, before the parent __init__ calls.""" | 
| Andrzej Kurek | 068a73f | 2022-10-06 18:52:44 -0400 | [diff] [blame] | 389 |  | 
| Andrzej Kurek | b95ba9a | 2022-10-04 15:02:41 -0400 | [diff] [blame] | 390 | class CipherInfo: # pylint: disable=too-few-public-methods | 
| Gilles Peskine | 3ce0e32 | 2019-01-29 23:12:28 +0100 | [diff] [blame] | 391 | """Collect data about cipher.h.""" | 
| Andrzej Kurek | b95ba9a | 2022-10-04 15:02:41 -0400 | [diff] [blame] | 392 | def __init__(self): | 
| Gilles Peskine | 3ce0e32 | 2019-01-29 23:12:28 +0100 | [diff] [blame] | 393 | self.base_symbols = set() | 
| Andrzej Kurek | b95ba9a | 2022-10-04 15:02:41 -0400 | [diff] [blame] | 394 | with open('include/mbedtls/cipher.h', encoding="utf-8") as fh: | 
| Gilles Peskine | 3ce0e32 | 2019-01-29 23:12:28 +0100 | [diff] [blame] | 395 | for line in fh: | 
|  | 396 | m = re.match(r' *MBEDTLS_CIPHER_ID_(\w+),', line) | 
|  | 397 | if m and m.group(1) not in ['NONE', 'NULL', '3DES']: | 
|  | 398 | self.base_symbols.add('MBEDTLS_' + m.group(1) + '_C') | 
|  | 399 |  | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 400 | class DomainData: | 
| Andrzej Kurek | b95ba9a | 2022-10-04 15:02:41 -0400 | [diff] [blame] | 401 | """A container for domains and jobs, used to structurize testing.""" | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 402 | def config_symbols_matching(self, regexp): | 
| Andrzej Kurek | 467a0f2 | 2022-10-20 06:15:06 -0400 | [diff] [blame] | 403 | """List the config.h settings matching regexp.""" | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 404 | return [symbol for symbol in self.all_config_symbols | 
|  | 405 | if re.match(regexp, symbol)] | 
|  | 406 |  | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 407 | def __init__(self, options, conf): | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 408 | """Gather data about the library and establish a list of domains to test.""" | 
|  | 409 | build_command = [options.make_command, 'CFLAGS=-Werror'] | 
|  | 410 | build_and_test = [build_command, [options.make_command, 'test']] | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 411 | self.all_config_symbols = set(conf.settings.keys()) | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 412 | # Find hash modules by name. | 
|  | 413 | hash_symbols = self.config_symbols_matching(r'MBEDTLS_(MD|RIPEMD|SHA)[0-9]+_C\Z') | 
| Andrzej Kurek | aa11281 | 2022-11-22 08:13:45 -0500 | [diff] [blame] | 414 | hash_symbols.append("MBEDTLS_SHA512_NO_SHA384") | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 415 | # Find elliptic curve enabling macros by name. | 
|  | 416 | curve_symbols = self.config_symbols_matching(r'MBEDTLS_ECP_DP_\w+_ENABLED\Z') | 
|  | 417 | # Find key exchange enabling macros by name. | 
|  | 418 | key_exchange_symbols = self.config_symbols_matching(r'MBEDTLS_KEY_EXCHANGE_\w+_ENABLED\Z') | 
| Gilles Peskine | 3ce0e32 | 2019-01-29 23:12:28 +0100 | [diff] [blame] | 419 | # Find cipher IDs (block permutations and stream ciphers --- chaining | 
|  | 420 | # and padding modes are exercised separately) information by parsing | 
| Andrzej Kurek | 467a0f2 | 2022-10-20 06:15:06 -0400 | [diff] [blame] | 421 | # cipher.h, as the information is not readily available in config.h. | 
| Andrzej Kurek | b95ba9a | 2022-10-04 15:02:41 -0400 | [diff] [blame] | 422 | cipher_info = CipherInfo() | 
| Gilles Peskine | 3ce0e32 | 2019-01-29 23:12:28 +0100 | [diff] [blame] | 423 | # Find block cipher chaining and padding mode enabling macros by name. | 
|  | 424 | cipher_chaining_symbols = self.config_symbols_matching(r'MBEDTLS_CIPHER_MODE_\w+\Z') | 
|  | 425 | cipher_padding_symbols = self.config_symbols_matching(r'MBEDTLS_CIPHER_PADDING_\w+\Z') | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 426 | self.domains = { | 
| Gilles Peskine | 3ce0e32 | 2019-01-29 23:12:28 +0100 | [diff] [blame] | 427 | # Cipher IDs, chaining modes and padding modes. Run the test suites. | 
|  | 428 | 'cipher_id': ExclusiveDomain(cipher_info.base_symbols, | 
|  | 429 | build_and_test), | 
|  | 430 | 'cipher_chaining': ExclusiveDomain(cipher_chaining_symbols, | 
|  | 431 | build_and_test), | 
|  | 432 | 'cipher_padding': ExclusiveDomain(cipher_padding_symbols, | 
|  | 433 | build_and_test), | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 434 | # Elliptic curves. Run the test suites. | 
|  | 435 | 'curves': ExclusiveDomain(curve_symbols, build_and_test), | 
| Andrzej Kurek | 467a0f2 | 2022-10-20 06:15:06 -0400 | [diff] [blame] | 436 | # Hash algorithms. Exclude exclusive domain of MD, RIPEMD, SHA1 (obsolete) | 
| Andrzej Kurek | 068a73f | 2022-10-06 18:52:44 -0400 | [diff] [blame] | 437 | 'hashes': DualDomain(hash_symbols, build_and_test, | 
| Andrzej Kurek | aa11281 | 2022-11-22 08:13:45 -0500 | [diff] [blame] | 438 | exclude=r'MBEDTLS_(MD|RIPEMD|SHA1_)'\ | 
|  | 439 | '|!MBEDTLS_*_NO_SHA'), | 
| Andrzej Kurek | ddf6260 | 2023-01-23 06:19:14 -0500 | [diff] [blame] | 440 | # Key exchange types. | 
| Andrzej Kurek | de416fc | 2022-11-02 04:50:16 -0400 | [diff] [blame] | 441 | 'kex': ExclusiveDomain(key_exchange_symbols, build_and_test), | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 442 | 'pkalgs': ComplementaryDomain(['MBEDTLS_ECDSA_C', | 
|  | 443 | 'MBEDTLS_ECP_C', | 
|  | 444 | 'MBEDTLS_PKCS1_V21', | 
|  | 445 | 'MBEDTLS_PKCS1_V15', | 
|  | 446 | 'MBEDTLS_RSA_C', | 
|  | 447 | 'MBEDTLS_X509_RSASSA_PSS_SUPPORT'], | 
|  | 448 | build_and_test), | 
|  | 449 | } | 
|  | 450 | self.jobs = {} | 
|  | 451 | for domain in self.domains.values(): | 
|  | 452 | for job in domain.jobs: | 
|  | 453 | self.jobs[job.name] = job | 
|  | 454 |  | 
|  | 455 | def get_jobs(self, name): | 
|  | 456 | """Return the list of jobs identified by the given name. | 
|  | 457 | A name can either be the name of a domain or the name of one specific job.""" | 
|  | 458 | if name in self.domains: | 
|  | 459 | return sorted(self.domains[name].jobs, key=lambda job: job.name) | 
|  | 460 | else: | 
|  | 461 | return [self.jobs[name]] | 
|  | 462 |  | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 463 | def run(options, job, conf, colors=NO_COLORS): | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 464 | """Run the specified job (a Job instance).""" | 
|  | 465 | subprocess.check_call([options.make_command, 'clean']) | 
| Gilles Peskine | fd1d69c | 2019-01-29 18:48:48 +0100 | [diff] [blame] | 466 | job.announce(colors, None) | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 467 | if not job.configure(conf, options, colors): | 
|  | 468 | job.announce(colors, False) | 
|  | 469 | return False | 
|  | 470 | conf.write() | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 471 | success = job.test(options) | 
| Gilles Peskine | fd1d69c | 2019-01-29 18:48:48 +0100 | [diff] [blame] | 472 | job.announce(colors, success) | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 473 | return success | 
|  | 474 |  | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 475 | def run_tests(options, domain_data, conf): | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 476 | """Run the desired jobs. | 
|  | 477 | domain_data should be a DomainData instance that describes the available | 
|  | 478 | domains and jobs. | 
| Andrzej Kurek | 113952d | 2022-10-17 08:39:09 -0400 | [diff] [blame] | 479 | Run the jobs listed in options.tasks.""" | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 480 | if not hasattr(options, 'config_backup'): | 
|  | 481 | options.config_backup = options.config + '.bak' | 
| Gilles Peskine | fd1d69c | 2019-01-29 18:48:48 +0100 | [diff] [blame] | 482 | colors = Colors(options) | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 483 | jobs = [] | 
|  | 484 | failures = [] | 
|  | 485 | successes = [] | 
| Andrzej Kurek | 113952d | 2022-10-17 08:39:09 -0400 | [diff] [blame] | 486 | for name in options.tasks: | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 487 | jobs += domain_data.get_jobs(name) | 
|  | 488 | backup_config(options) | 
|  | 489 | try: | 
|  | 490 | for job in jobs: | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 491 | success = run(options, job, conf, colors=colors) | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 492 | if not success: | 
|  | 493 | if options.keep_going: | 
|  | 494 | failures.append(job.name) | 
|  | 495 | else: | 
|  | 496 | return False | 
|  | 497 | else: | 
|  | 498 | successes.append(job.name) | 
| Gilles Peskine | 88e8dd6 | 2019-01-29 18:52:16 +0100 | [diff] [blame] | 499 | restore_config(options) | 
|  | 500 | except: | 
|  | 501 | # Restore the configuration, except in stop-on-error mode if there | 
|  | 502 | # was an error, where we leave the failing configuration up for | 
|  | 503 | # developer convenience. | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 504 | if options.keep_going: | 
| Gilles Peskine | 88e8dd6 | 2019-01-29 18:52:16 +0100 | [diff] [blame] | 505 | restore_config(options) | 
|  | 506 | raise | 
| Gilles Peskine | dc68f61 | 2019-01-29 18:50:03 +0100 | [diff] [blame] | 507 | if successes: | 
|  | 508 | log_line('{} passed'.format(' '.join(successes)), color=colors.bold_green) | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 509 | if failures: | 
| Gilles Peskine | dc68f61 | 2019-01-29 18:50:03 +0100 | [diff] [blame] | 510 | log_line('{} FAILED'.format(' '.join(failures)), color=colors.bold_red) | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 511 | return False | 
|  | 512 | else: | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 513 | return True | 
|  | 514 |  | 
| Andrzej Kurek | b95ba9a | 2022-10-04 15:02:41 -0400 | [diff] [blame] | 515 | def main(): | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 516 | try: | 
| Andrzej Kurek | 110fc48 | 2022-10-09 05:29:44 -0400 | [diff] [blame] | 517 | parser = argparse.ArgumentParser( | 
|  | 518 | formatter_class=argparse.RawDescriptionHelpFormatter, | 
|  | 519 | description= | 
|  | 520 | "Test Mbed TLS with a subset of algorithms.\n\n" | 
|  | 521 | "Example usage:\n" | 
| Andrzej Kurek | ffbc8f5 | 2022-10-17 08:34:40 -0400 | [diff] [blame] | 522 | r"./tests/scripts/depends.py \!MBEDTLS_SHA1_C MBEDTLS_SHA256_C""\n" | 
| Andrzej Kurek | 110fc48 | 2022-10-09 05:29:44 -0400 | [diff] [blame] | 523 | "./tests/scripts/depends.py MBEDTLS_AES_C hashes\n" | 
|  | 524 | "./tests/scripts/depends.py cipher_id cipher_chaining\n") | 
| Gilles Peskine | fd1d69c | 2019-01-29 18:48:48 +0100 | [diff] [blame] | 525 | parser.add_argument('--color', metavar='WHEN', | 
|  | 526 | help='Colorize the output (always/auto/never)', | 
|  | 527 | choices=['always', 'auto', 'never'], default='auto') | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 528 | parser.add_argument('-c', '--config', metavar='FILE', | 
|  | 529 | help='Configuration file to modify', | 
| Andrzej Kurek | 467a0f2 | 2022-10-20 06:15:06 -0400 | [diff] [blame] | 530 | default='include/mbedtls/config.h') | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 531 | parser.add_argument('-C', '--directory', metavar='DIR', | 
|  | 532 | help='Change to this directory before anything else', | 
|  | 533 | default='.') | 
|  | 534 | parser.add_argument('-k', '--keep-going', | 
|  | 535 | help='Try all configurations even if some fail (default)', | 
|  | 536 | action='store_true', dest='keep_going', default=True) | 
|  | 537 | parser.add_argument('-e', '--no-keep-going', | 
|  | 538 | help='Stop as soon as a configuration fails', | 
|  | 539 | action='store_false', dest='keep_going') | 
|  | 540 | parser.add_argument('--list-jobs', | 
|  | 541 | help='List supported jobs and exit', | 
|  | 542 | action='append_const', dest='list', const='jobs') | 
|  | 543 | parser.add_argument('--list-domains', | 
|  | 544 | help='List supported domains and exit', | 
|  | 545 | action='append_const', dest='list', const='domains') | 
|  | 546 | parser.add_argument('--make-command', metavar='CMD', | 
|  | 547 | help='Command to run instead of make (e.g. gmake)', | 
|  | 548 | action='store', default='make') | 
| Andrzej Kurek | 2b44a92 | 2022-10-24 10:41:20 -0400 | [diff] [blame] | 549 | parser.add_argument('--unset-use-psa', | 
|  | 550 | help='Unset MBEDTLS_USE_PSA_CRYPTO before any test', | 
|  | 551 | action='store_true', dest='unset_use_psa') | 
| Andrzej Kurek | 113952d | 2022-10-17 08:39:09 -0400 | [diff] [blame] | 552 | parser.add_argument('tasks', metavar='TASKS', nargs='*', | 
|  | 553 | help='The domain(s) or job(s) to test (default: all).', | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 554 | default=True) | 
|  | 555 | options = parser.parse_args() | 
|  | 556 | os.chdir(options.directory) | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 557 | conf = config.ConfigFile(options.config) | 
|  | 558 | domain_data = DomainData(options, conf) | 
|  | 559 |  | 
| Andrzej Kurek | 113952d | 2022-10-17 08:39:09 -0400 | [diff] [blame] | 560 | if options.tasks is True: | 
|  | 561 | options.tasks = sorted(domain_data.domains.keys()) | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 562 | if options.list: | 
| Andrzej Kurek | b95ba9a | 2022-10-04 15:02:41 -0400 | [diff] [blame] | 563 | for arg in options.list: | 
|  | 564 | for domain_name in sorted(getattr(domain_data, arg).keys()): | 
|  | 565 | print(domain_name) | 
|  | 566 | sys.exit(0) | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 567 | else: | 
| Andrzej Kurek | cf39406 | 2023-02-15 05:42:02 -0500 | [diff] [blame] | 568 | sys.exit(0 if run_tests(options, domain_data, conf) else 1) | 
| Andrzej Kurek | b95ba9a | 2022-10-04 15:02:41 -0400 | [diff] [blame] | 569 | except Exception: # pylint: disable=broad-except | 
| Gilles Peskine | f5ea197 | 2019-01-29 08:50:20 +0100 | [diff] [blame] | 570 | traceback.print_exc() | 
| Andrzej Kurek | b95ba9a | 2022-10-04 15:02:41 -0400 | [diff] [blame] | 571 | sys.exit(3) | 
|  | 572 |  | 
|  | 573 | if __name__ == '__main__': | 
|  | 574 | main() |