mirror of
				https://github.com/Mbed-TLS/mbedtls.git
				synced 2025-10-24 13:32:59 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			291 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			291 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
| #!/usr/bin/env python3
 | |
| 
 | |
| """Edit test cases to use PSA dependencies instead of classic dependencies.
 | |
| """
 | |
| 
 | |
| # Copyright The Mbed TLS Contributors
 | |
| # SPDX-License-Identifier: Apache-2.0 OR GPL-2.0-or-later
 | |
| 
 | |
| import os
 | |
| import re
 | |
| import sys
 | |
| 
 | |
| CLASSIC_DEPENDENCIES = frozenset([
 | |
|     # This list is manually filtered from config.h.
 | |
| 
 | |
|     # Mbed TLS feature support.
 | |
|     # Only features that affect what can be done are listed here.
 | |
|     # Options that control optimizations or alternative implementations
 | |
|     # are omitted.
 | |
|     'MBEDTLS_CIPHER_MODE_CBC',
 | |
|     'MBEDTLS_CIPHER_MODE_CFB',
 | |
|     'MBEDTLS_CIPHER_MODE_CTR',
 | |
|     'MBEDTLS_CIPHER_MODE_OFB',
 | |
|     'MBEDTLS_CIPHER_MODE_XTS',
 | |
|     'MBEDTLS_CIPHER_NULL_CIPHER',
 | |
|     'MBEDTLS_CIPHER_PADDING_PKCS7',
 | |
|     'MBEDTLS_CIPHER_PADDING_ONE_AND_ZEROS',
 | |
|     'MBEDTLS_CIPHER_PADDING_ZEROS_AND_LEN',
 | |
|     'MBEDTLS_CIPHER_PADDING_ZEROS',
 | |
|     #curve#'MBEDTLS_ECP_DP_SECP192R1_ENABLED',
 | |
|     #curve#'MBEDTLS_ECP_DP_SECP224R1_ENABLED',
 | |
|     #curve#'MBEDTLS_ECP_DP_SECP256R1_ENABLED',
 | |
|     #curve#'MBEDTLS_ECP_DP_SECP384R1_ENABLED',
 | |
|     #curve#'MBEDTLS_ECP_DP_SECP521R1_ENABLED',
 | |
|     #curve#'MBEDTLS_ECP_DP_SECP192K1_ENABLED',
 | |
|     #curve#'MBEDTLS_ECP_DP_SECP224K1_ENABLED',
 | |
|     #curve#'MBEDTLS_ECP_DP_SECP256K1_ENABLED',
 | |
|     #curve#'MBEDTLS_ECP_DP_BP256R1_ENABLED',
 | |
|     #curve#'MBEDTLS_ECP_DP_BP384R1_ENABLED',
 | |
|     #curve#'MBEDTLS_ECP_DP_BP512R1_ENABLED',
 | |
|     #curve#'MBEDTLS_ECP_DP_CURVE25519_ENABLED',
 | |
|     #curve#'MBEDTLS_ECP_DP_CURVE448_ENABLED',
 | |
|     'MBEDTLS_ECDSA_DETERMINISTIC',
 | |
|     #'MBEDTLS_GENPRIME', #needed for RSA key generation
 | |
|     'MBEDTLS_PKCS1_V15',
 | |
|     'MBEDTLS_PKCS1_V21',
 | |
|     'MBEDTLS_SHA512_NO_SHA384',
 | |
| 
 | |
|     # Mbed TLS modules.
 | |
|     # Only modules that provide cryptographic mechanisms are listed here.
 | |
|     # Platform, data formatting, X.509 or TLS modules are omitted.
 | |
|     'MBEDTLS_AES_C',
 | |
|     'MBEDTLS_ARC4_C',
 | |
|     'MBEDTLS_BIGNUM_C',
 | |
|     #cipher#'MBEDTLS_BLOWFISH_C',
 | |
|     'MBEDTLS_CAMELLIA_C',
 | |
|     'MBEDTLS_ARIA_C',
 | |
|     'MBEDTLS_CCM_C',
 | |
|     'MBEDTLS_CHACHA20_C',
 | |
|     'MBEDTLS_CHACHAPOLY_C',
 | |
|     'MBEDTLS_CMAC_C',
 | |
|     'MBEDTLS_CTR_DRBG_C',
 | |
|     'MBEDTLS_DES_C',
 | |
|     'MBEDTLS_DHM_C',
 | |
|     'MBEDTLS_ECDH_C',
 | |
|     'MBEDTLS_ECDSA_C',
 | |
|     'MBEDTLS_ECJPAKE_C',
 | |
|     'MBEDTLS_ECP_C',
 | |
|     'MBEDTLS_ENTROPY_C',
 | |
|     'MBEDTLS_GCM_C',
 | |
|     'MBEDTLS_HKDF_C',
 | |
|     'MBEDTLS_HMAC_DRBG_C',
 | |
|     'MBEDTLS_NIST_KW_C',
 | |
|     'MBEDTLS_MD2_C',
 | |
|     'MBEDTLS_MD4_C',
 | |
|     'MBEDTLS_MD5_C',
 | |
|     'MBEDTLS_PKCS5_C',
 | |
|     'MBEDTLS_PKCS12_C',
 | |
|     'MBEDTLS_POLY1305_C',
 | |
|     'MBEDTLS_RIPEMD160_C',
 | |
|     'MBEDTLS_RSA_C',
 | |
|     'MBEDTLS_SHA1_C',
 | |
|     'MBEDTLS_SHA256_C',
 | |
|     'MBEDTLS_SHA512_C',
 | |
|     'MBEDTLS_XTEA_C',
 | |
| ])
 | |
| 
 | |
| def is_classic_dependency(dep):
 | |
|     """Whether dep is a classic dependency that PSA test cases should not use."""
 | |
|     if dep.startswith('!'):
 | |
|         dep = dep[1:]
 | |
|     return dep in CLASSIC_DEPENDENCIES
 | |
| 
 | |
| def is_systematic_dependency(dep):
 | |
|     """Whether dep is a PSA dependency which is determined systematically."""
 | |
|     if dep.startswith('PSA_WANT_ECC_'):
 | |
|         return False
 | |
|     return dep.startswith('PSA_WANT_')
 | |
| 
 | |
| WITHOUT_SYSTEMATIC_DEPENDENCIES = frozenset([
 | |
|     'PSA_ALG_AEAD_WITH_SHORTENED_TAG', # only a modifier
 | |
|     'PSA_ALG_ANY_HASH', # only meaningful in policies
 | |
|     'PSA_ALG_KEY_AGREEMENT', # only a way to combine algorithms
 | |
|     'PSA_ALG_TRUNCATED_MAC', # only a modifier
 | |
|     'PSA_KEY_TYPE_NONE', # not a real key type
 | |
|     'PSA_KEY_TYPE_DERIVE', # always supported, don't list it to reduce noise
 | |
|     'PSA_KEY_TYPE_RAW_DATA', # always supported, don't list it to reduce noise
 | |
|     'PSA_ALG_AT_LEAST_THIS_LENGTH_MAC', #only a modifier
 | |
|     'PSA_ALG_AEAD_WITH_AT_LEAST_THIS_LENGTH_TAG', #only a modifier
 | |
| ])
 | |
| 
 | |
| SPECIAL_SYSTEMATIC_DEPENDENCIES = {
 | |
|     'PSA_ALG_ECDSA_ANY': frozenset(['PSA_WANT_ALG_ECDSA']),
 | |
|     'PSA_ALG_RSA_PKCS1V15_SIGN_RAW': frozenset(['PSA_WANT_ALG_RSA_PKCS1V15_SIGN']),
 | |
| }
 | |
| 
 | |
| def dependencies_of_symbol(symbol):
 | |
|     """Return the dependencies for a symbol that designates a cryptographic mechanism."""
 | |
|     if symbol in WITHOUT_SYSTEMATIC_DEPENDENCIES:
 | |
|         return frozenset()
 | |
|     if symbol in SPECIAL_SYSTEMATIC_DEPENDENCIES:
 | |
|         return SPECIAL_SYSTEMATIC_DEPENDENCIES[symbol]
 | |
|     if symbol.startswith('PSA_ALG_CATEGORY_') or \
 | |
|        symbol.startswith('PSA_KEY_TYPE_CATEGORY_'):
 | |
|         # Categories are used in test data when an unsupported but plausible
 | |
|         # mechanism number needed. They have no associated dependency.
 | |
|         return frozenset()
 | |
|     return {symbol.replace('_', '_WANT_', 1)}
 | |
| 
 | |
| def systematic_dependencies(file_name, function_name, arguments):
 | |
|     """List the systematically determined dependency for a test case."""
 | |
|     deps = set()
 | |
| 
 | |
|     # Run key policy negative tests even if the algorithm to attempt performing
 | |
|     # is not supported but in the case where the test is to check an
 | |
|     # incompatibility between a requested algorithm for a cryptographic
 | |
|     # operation and a key policy. In the latter, we want to filter out the
 | |
|     # cases # where PSA_ERROR_NOT_SUPPORTED is returned instead of
 | |
|     # PSA_ERROR_NOT_PERMITTED.
 | |
|     if function_name.endswith('_key_policy') and \
 | |
|        arguments[-1].startswith('PSA_ERROR_') and \
 | |
|        arguments[-1] != ('PSA_ERROR_NOT_PERMITTED'):
 | |
|         arguments[-2] = ''
 | |
|     if function_name == 'copy_fail' and \
 | |
|        arguments[-1].startswith('PSA_ERROR_'):
 | |
|         arguments[-2] = ''
 | |
|         arguments[-3] = ''
 | |
| 
 | |
|     # Storage format tests that only look at how the file is structured and
 | |
|     # don't care about the format of the key material don't depend on any
 | |
|     # cryptographic mechanisms.
 | |
|     if os.path.basename(file_name) == 'test_suite_psa_crypto_persistent_key.data' and \
 | |
|        function_name in {'format_storage_data_check',
 | |
|                          'parse_storage_data_check'}:
 | |
|         return []
 | |
| 
 | |
|     for arg in arguments:
 | |
|         for symbol in re.findall(r'PSA_(?:ALG|KEY_TYPE)_\w+', arg):
 | |
|             deps.update(dependencies_of_symbol(symbol))
 | |
|     return sorted(deps)
 | |
| 
 | |
| def updated_dependencies(file_name, function_name, arguments, dependencies):
 | |
|     """Rework the list of dependencies into PSA_WANT_xxx.
 | |
| 
 | |
|     Remove classic crypto dependencies such as MBEDTLS_RSA_C,
 | |
|     MBEDTLS_PKCS1_V15, etc.
 | |
| 
 | |
|     Add systematic PSA_WANT_xxx dependencies based on the called function and
 | |
|     its arguments, replacing existing PSA_WANT_xxx dependencies.
 | |
|     """
 | |
|     automatic = systematic_dependencies(file_name, function_name, arguments)
 | |
|     manual = [dep for dep in dependencies
 | |
|               if not (is_systematic_dependency(dep) or
 | |
|                       is_classic_dependency(dep))]
 | |
|     return automatic + manual
 | |
| 
 | |
| def keep_manual_dependencies(file_name, function_name, arguments):
 | |
|     #pylint: disable=unused-argument
 | |
|     """Declare test functions with unusual dependencies here."""
 | |
|     # If there are no arguments, we can't do any useful work. Assume that if
 | |
|     # there are dependencies, they are warranted.
 | |
|     if not arguments:
 | |
|         return True
 | |
|     # When PSA_ERROR_NOT_SUPPORTED is expected, usually, at least one of the
 | |
|     # constants mentioned in the test should not be supported. It isn't
 | |
|     # possible to determine which one in a systematic way. So let the programmer
 | |
|     # decide.
 | |
|     if arguments[-1] == 'PSA_ERROR_NOT_SUPPORTED':
 | |
|         return True
 | |
|     return False
 | |
| 
 | |
| def process_data_stanza(stanza, file_name, test_case_number):
 | |
|     """Update PSA crypto dependencies in one Mbed TLS test case.
 | |
| 
 | |
|     stanza is the test case text (including the description, the dependencies,
 | |
|     the line with the function and arguments, and optionally comments). Return
 | |
|     a new stanza with an updated dependency line, preserving everything else
 | |
|     (description, comments, arguments, etc.).
 | |
|     """
 | |
|     if not stanza.lstrip('\n'):
 | |
|         # Just blank lines
 | |
|         return stanza
 | |
|     # Expect 2 or 3 non-comment lines: description, optional dependencies,
 | |
|     # function-and-arguments.
 | |
|     content_matches = list(re.finditer(r'^[\t ]*([^\t #].*)$', stanza, re.M))
 | |
|     if len(content_matches) < 2:
 | |
|         raise Exception('Not enough content lines in paragraph {} in {}'
 | |
|                         .format(test_case_number, file_name))
 | |
|     if len(content_matches) > 3:
 | |
|         raise Exception('Too many content lines in paragraph {} in {}'
 | |
|                         .format(test_case_number, file_name))
 | |
|     arguments = content_matches[-1].group(0).split(':')
 | |
|     function_name = arguments.pop(0)
 | |
|     if keep_manual_dependencies(file_name, function_name, arguments):
 | |
|         return stanza
 | |
|     if len(content_matches) == 2:
 | |
|         # Insert a line for the dependencies. If it turns out that there are
 | |
|         # no dependencies, we'll remove that empty line below.
 | |
|         dependencies_location = content_matches[-1].start()
 | |
|         text_before = stanza[:dependencies_location]
 | |
|         text_after = '\n' + stanza[dependencies_location:]
 | |
|         old_dependencies = []
 | |
|         dependencies_leader = 'depends_on:'
 | |
|     else:
 | |
|         dependencies_match = content_matches[-2]
 | |
|         text_before = stanza[:dependencies_match.start()]
 | |
|         text_after = stanza[dependencies_match.end():]
 | |
|         old_dependencies = dependencies_match.group(0).split(':')
 | |
|         dependencies_leader = old_dependencies.pop(0) + ':'
 | |
|         if dependencies_leader != 'depends_on:':
 | |
|             raise Exception('Next-to-last line does not start with "depends_on:"'
 | |
|                             ' in paragraph {} in {}'
 | |
|                             .format(test_case_number, file_name))
 | |
|     new_dependencies = updated_dependencies(file_name, function_name, arguments,
 | |
|                                             old_dependencies)
 | |
|     if new_dependencies:
 | |
|         stanza = (text_before +
 | |
|                   dependencies_leader + ':'.join(new_dependencies) +
 | |
|                   text_after)
 | |
|     else:
 | |
|         # The dependencies have become empty. Remove the depends_on: line.
 | |
|         assert text_after[0] == '\n'
 | |
|         stanza = text_before + text_after[1:]
 | |
|     return stanza
 | |
| 
 | |
| def process_data_file(file_name, old_content):
 | |
|     """Update PSA crypto dependencies in an Mbed TLS test suite data file.
 | |
| 
 | |
|     Process old_content (the old content of the file) and return the new content.
 | |
|     """
 | |
|     old_stanzas = old_content.split('\n\n')
 | |
|     new_stanzas = [process_data_stanza(stanza, file_name, n)
 | |
|                    for n, stanza in enumerate(old_stanzas, start=1)]
 | |
|     return '\n\n'.join(new_stanzas)
 | |
| 
 | |
| def update_file(file_name, old_content, new_content):
 | |
|     """Update the given file with the given new content.
 | |
| 
 | |
|     Replace the existing file. The previous version is renamed to *.bak.
 | |
|     Don't modify the file if the content was unchanged.
 | |
|     """
 | |
|     if new_content == old_content:
 | |
|         return
 | |
|     backup = file_name + '.bak'
 | |
|     tmp = file_name + '.tmp'
 | |
|     with open(tmp, 'w', encoding='utf-8') as new_file:
 | |
|         new_file.write(new_content)
 | |
|     os.replace(file_name, backup)
 | |
|     os.replace(tmp, file_name)
 | |
| 
 | |
| def process_file(file_name):
 | |
|     """Update PSA crypto dependencies in an Mbed TLS test suite data file.
 | |
| 
 | |
|     Replace the existing file. The previous version is renamed to *.bak.
 | |
|     Don't modify the file if the content was unchanged.
 | |
|     """
 | |
|     old_content = open(file_name, encoding='utf-8').read()
 | |
|     if file_name.endswith('.data'):
 | |
|         new_content = process_data_file(file_name, old_content)
 | |
|     else:
 | |
|         raise Exception('File type not recognized: {}'
 | |
|                         .format(file_name))
 | |
|     update_file(file_name, old_content, new_content)
 | |
| 
 | |
| def main(args):
 | |
|     for file_name in args:
 | |
|         process_file(file_name)
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|     main(sys.argv[1:])
 |