| Gilles Peskine | fdb7223 | 2023-06-19 20:46:47 +0200 | [diff] [blame] | 1 | """Collect information about PSA cryptographic mechanisms. | 
|  | 2 | """ | 
|  | 3 |  | 
|  | 4 | # Copyright The Mbed TLS Contributors | 
| Dave Rodgman | 16799db | 2023-11-02 19:47:20 +0000 | [diff] [blame] | 5 | # SPDX-License-Identifier: Apache-2.0 OR GPL-2.0-or-later | 
| Gilles Peskine | fdb7223 | 2023-06-19 20:46:47 +0200 | [diff] [blame] | 6 | # | 
| Gilles Peskine | fdb7223 | 2023-06-19 20:46:47 +0200 | [diff] [blame] | 7 |  | 
|  | 8 | import re | 
| Yanray Wang | 6b190d4 | 2023-11-01 13:44:14 +0800 | [diff] [blame] | 9 | from collections import OrderedDict | 
|  | 10 | from typing import FrozenSet, List, Optional | 
| Gilles Peskine | fdb7223 | 2023-06-19 20:46:47 +0200 | [diff] [blame] | 11 |  | 
|  | 12 | from . import macro_collector | 
|  | 13 |  | 
|  | 14 |  | 
|  | 15 | class Information: | 
|  | 16 | """Gather information about PSA constructors.""" | 
|  | 17 |  | 
|  | 18 | def __init__(self) -> None: | 
|  | 19 | self.constructors = self.read_psa_interface() | 
|  | 20 |  | 
|  | 21 | @staticmethod | 
|  | 22 | def remove_unwanted_macros( | 
|  | 23 | constructors: macro_collector.PSAMacroEnumerator | 
|  | 24 | ) -> None: | 
|  | 25 | # Mbed TLS does not support finite-field DSA. | 
|  | 26 | # Don't attempt to generate any related test case. | 
|  | 27 | constructors.key_types.discard('PSA_KEY_TYPE_DSA_KEY_PAIR') | 
|  | 28 | constructors.key_types.discard('PSA_KEY_TYPE_DSA_PUBLIC_KEY') | 
|  | 29 |  | 
|  | 30 | def read_psa_interface(self) -> macro_collector.PSAMacroEnumerator: | 
|  | 31 | """Return the list of known key types, algorithms, etc.""" | 
|  | 32 | constructors = macro_collector.InputsForTest() | 
|  | 33 | header_file_names = ['include/psa/crypto_values.h', | 
|  | 34 | 'include/psa/crypto_extra.h'] | 
|  | 35 | test_suites = ['tests/suites/test_suite_psa_crypto_metadata.data'] | 
|  | 36 | for header_file_name in header_file_names: | 
|  | 37 | constructors.parse_header(header_file_name) | 
|  | 38 | for test_cases in test_suites: | 
|  | 39 | constructors.parse_test_cases(test_cases) | 
|  | 40 | self.remove_unwanted_macros(constructors) | 
|  | 41 | constructors.gather_arguments() | 
|  | 42 | return constructors | 
|  | 43 |  | 
|  | 44 |  | 
|  | 45 | def psa_want_symbol(name: str) -> str: | 
|  | 46 | """Return the PSA_WANT_xxx symbol associated with a PSA crypto feature.""" | 
|  | 47 | if name.startswith('PSA_'): | 
|  | 48 | return name[:4] + 'WANT_' + name[4:] | 
|  | 49 | else: | 
|  | 50 | raise ValueError('Unable to determine the PSA_WANT_ symbol for ' + name) | 
|  | 51 |  | 
|  | 52 | def finish_family_dependency(dep: str, bits: int) -> str: | 
|  | 53 | """Finish dep if it's a family dependency symbol prefix. | 
|  | 54 |  | 
|  | 55 | A family dependency symbol prefix is a PSA_WANT_ symbol that needs to be | 
|  | 56 | qualified by the key size. If dep is such a symbol, finish it by adjusting | 
|  | 57 | the prefix and appending the key size. Other symbols are left unchanged. | 
|  | 58 | """ | 
|  | 59 | return re.sub(r'_FAMILY_(.*)', r'_\1_' + str(bits), dep) | 
|  | 60 |  | 
|  | 61 | def finish_family_dependencies(dependencies: List[str], bits: int) -> List[str]: | 
|  | 62 | """Finish any family dependency symbol prefixes. | 
|  | 63 |  | 
|  | 64 | Apply `finish_family_dependency` to each element of `dependencies`. | 
|  | 65 | """ | 
|  | 66 | return [finish_family_dependency(dep, bits) for dep in dependencies] | 
|  | 67 |  | 
|  | 68 | SYMBOLS_WITHOUT_DEPENDENCY = frozenset([ | 
|  | 69 | 'PSA_ALG_AEAD_WITH_AT_LEAST_THIS_LENGTH_TAG', # modifier, only in policies | 
|  | 70 | 'PSA_ALG_AEAD_WITH_SHORTENED_TAG', # modifier | 
|  | 71 | 'PSA_ALG_ANY_HASH', # only in policies | 
|  | 72 | 'PSA_ALG_AT_LEAST_THIS_LENGTH_MAC', # modifier, only in policies | 
|  | 73 | 'PSA_ALG_KEY_AGREEMENT', # chaining | 
|  | 74 | 'PSA_ALG_TRUNCATED_MAC', # modifier | 
|  | 75 | ]) | 
|  | 76 | def automatic_dependencies(*expressions: str) -> List[str]: | 
|  | 77 | """Infer dependencies of a test case by looking for PSA_xxx symbols. | 
|  | 78 |  | 
|  | 79 | The arguments are strings which should be C expressions. Do not use | 
|  | 80 | string literals or comments as this function is not smart enough to | 
|  | 81 | skip them. | 
|  | 82 | """ | 
|  | 83 | used = set() | 
|  | 84 | for expr in expressions: | 
|  | 85 | used.update(re.findall(r'PSA_(?:ALG|ECC_FAMILY|KEY_TYPE)_\w+', expr)) | 
|  | 86 | used.difference_update(SYMBOLS_WITHOUT_DEPENDENCY) | 
|  | 87 | return sorted(psa_want_symbol(name) for name in used) | 
|  | 88 |  | 
|  | 89 | # Define set of regular expressions and dependencies to optionally append | 
| Yanray Wang | 6b190d4 | 2023-11-01 13:44:14 +0800 | [diff] [blame] | 90 | # extra dependencies for test case based on key description. | 
|  | 91 |  | 
|  | 92 | # Skip AES test cases which require 192- or 256-bit key | 
|  | 93 | # if MBEDTLS_AES_ONLY_128_BIT_KEY_LENGTH defined | 
| Yanray Wang | 70743b0 | 2023-11-09 16:13:53 +0800 | [diff] [blame] | 94 | AES_128BIT_ONLY_DEP_REGEX = re.compile(r'AES\s(192|256)') | 
| Yanray Wang | 6b190d4 | 2023-11-01 13:44:14 +0800 | [diff] [blame] | 95 | AES_128BIT_ONLY_DEP = ['!MBEDTLS_AES_ONLY_128_BIT_KEY_LENGTH'] | 
|  | 96 | # Skip AES/ARIA/CAMELLIA test cases which require decrypt operation in ECB mode | 
|  | 97 | # if MBEDTLS_BLOCK_CIPHER_NO_DECRYPT enabled. | 
| Yanray Wang | 70743b0 | 2023-11-09 16:13:53 +0800 | [diff] [blame] | 98 | ECB_NO_PADDING_DEP_REGEX = re.compile(r'(AES|ARIA|CAMELLIA).*ECB_NO_PADDING') | 
| Yanray Wang | 6b190d4 | 2023-11-01 13:44:14 +0800 | [diff] [blame] | 99 | ECB_NO_PADDING_DEP = ['!MBEDTLS_BLOCK_CIPHER_NO_DECRYPT'] | 
| Gilles Peskine | fdb7223 | 2023-06-19 20:46:47 +0200 | [diff] [blame] | 100 |  | 
| Yanray Wang | 6b190d4 | 2023-11-01 13:44:14 +0800 | [diff] [blame] | 101 | DEPENDENCY_FROM_DESCRIPTION = OrderedDict() | 
|  | 102 | DEPENDENCY_FROM_DESCRIPTION[AES_128BIT_ONLY_DEP_REGEX] = AES_128BIT_ONLY_DEP | 
|  | 103 | DEPENDENCY_FROM_DESCRIPTION[ECB_NO_PADDING_DEP_REGEX] = ECB_NO_PADDING_DEP | 
| Yanray Wang | 19583e4 | 2023-11-13 17:39:32 +0800 | [diff] [blame] | 104 | def generate_deps_from_description( | 
| Yanray Wang | 6b190d4 | 2023-11-01 13:44:14 +0800 | [diff] [blame] | 105 | description: str | 
|  | 106 | ) -> List[str]: | 
|  | 107 | """Return additional dependencies based on test case description and REGEX. | 
| Gilles Peskine | fdb7223 | 2023-06-19 20:46:47 +0200 | [diff] [blame] | 108 | """ | 
| Yanray Wang | 19583e4 | 2023-11-13 17:39:32 +0800 | [diff] [blame] | 109 | dep_list = [] | 
| Yanray Wang | 6b190d4 | 2023-11-01 13:44:14 +0800 | [diff] [blame] | 110 | for regex, deps in DEPENDENCY_FROM_DESCRIPTION.items(): | 
| Gilles Peskine | fdb7223 | 2023-06-19 20:46:47 +0200 | [diff] [blame] | 111 | if re.search(regex, description): | 
| Yanray Wang | 6b190d4 | 2023-11-01 13:44:14 +0800 | [diff] [blame] | 112 | dep_list += deps | 
| Gilles Peskine | fdb7223 | 2023-06-19 20:46:47 +0200 | [diff] [blame] | 113 |  | 
| Yanray Wang | 6b190d4 | 2023-11-01 13:44:14 +0800 | [diff] [blame] | 114 | return dep_list | 
| Gilles Peskine | fdb7223 | 2023-06-19 20:46:47 +0200 | [diff] [blame] | 115 |  | 
|  | 116 | # A temporary hack: at the time of writing, not all dependency symbols | 
|  | 117 | # are implemented yet. Skip test cases for which the dependency symbols are | 
|  | 118 | # not available. Once all dependency symbols are available, this hack must | 
|  | 119 | # be removed so that a bug in the dependency symbols properly leads to a test | 
|  | 120 | # failure. | 
|  | 121 | def read_implemented_dependencies(filename: str) -> FrozenSet[str]: | 
|  | 122 | return frozenset(symbol | 
|  | 123 | for line in open(filename) | 
|  | 124 | for symbol in re.findall(r'\bPSA_WANT_\w+\b', line)) | 
|  | 125 | _implemented_dependencies = None #type: Optional[FrozenSet[str]] #pylint: disable=invalid-name | 
|  | 126 | def hack_dependencies_not_implemented(dependencies: List[str]) -> None: | 
|  | 127 | global _implemented_dependencies #pylint: disable=global-statement,invalid-name | 
|  | 128 | if _implemented_dependencies is None: | 
|  | 129 | _implemented_dependencies = \ | 
|  | 130 | read_implemented_dependencies('include/psa/crypto_config.h') | 
|  | 131 | if not all((dep.lstrip('!') in _implemented_dependencies or | 
|  | 132 | not dep.lstrip('!').startswith('PSA_WANT')) | 
|  | 133 | for dep in dependencies): | 
|  | 134 | dependencies.append('DEPENDENCY_NOT_IMPLEMENTED_YET') | 
|  | 135 |  | 
|  | 136 | def tweak_key_pair_dependency(dep: str, usage: str): | 
|  | 137 | """ | 
|  | 138 | This helper function add the proper suffix to PSA_WANT_KEY_TYPE_xxx_KEY_PAIR | 
|  | 139 | symbols according to the required usage. | 
|  | 140 | """ | 
|  | 141 | ret_list = list() | 
|  | 142 | if dep.endswith('KEY_PAIR'): | 
|  | 143 | if usage == "BASIC": | 
|  | 144 | # BASIC automatically includes IMPORT and EXPORT for test purposes (see | 
|  | 145 | # config_psa.h). | 
|  | 146 | ret_list.append(re.sub(r'KEY_PAIR', r'KEY_PAIR_BASIC', dep)) | 
|  | 147 | ret_list.append(re.sub(r'KEY_PAIR', r'KEY_PAIR_IMPORT', dep)) | 
|  | 148 | ret_list.append(re.sub(r'KEY_PAIR', r'KEY_PAIR_EXPORT', dep)) | 
|  | 149 | elif usage == "GENERATE": | 
|  | 150 | ret_list.append(re.sub(r'KEY_PAIR', r'KEY_PAIR_GENERATE', dep)) | 
|  | 151 | else: | 
|  | 152 | # No replacement to do in this case | 
|  | 153 | ret_list.append(dep) | 
|  | 154 | return ret_list | 
|  | 155 |  | 
|  | 156 | def fix_key_pair_dependencies(dep_list: List[str], usage: str): | 
|  | 157 | new_list = [new_deps | 
|  | 158 | for dep in dep_list | 
|  | 159 | for new_deps in tweak_key_pair_dependency(dep, usage)] | 
|  | 160 |  | 
|  | 161 | return new_list |