| """ This module contains functions used by the test cases to hide the | 
 | architecture and/or the platform dependent nature of the tests. """ | 
 |  | 
 | from __future__ import absolute_import | 
 |  | 
 | # System modules | 
 | import itertools | 
 | import re | 
 | import subprocess | 
 | import sys | 
 | import os | 
 |  | 
 | # Third-party modules | 
 | import six | 
 | from six.moves.urllib import parse as urlparse | 
 |  | 
 | # LLDB modules | 
 | from . import configuration | 
 | import use_lldb_suite | 
 | import lldb | 
 |  | 
 |  | 
 | def check_first_register_readable(test_case): | 
 |     arch = test_case.getArchitecture() | 
 |  | 
 |     if arch in ['x86_64', 'i386']: | 
 |         test_case.expect("register read eax", substrs=['eax = 0x']) | 
 |     elif arch in ['arm', 'armv7', 'armv7k']: | 
 |         test_case.expect("register read r0", substrs=['r0 = 0x']) | 
 |     elif arch in ['aarch64', 'arm64']: | 
 |         test_case.expect("register read x0", substrs=['x0 = 0x']) | 
 |     elif re.match("mips", arch): | 
 |         test_case.expect("register read zero", substrs=['zero = 0x']) | 
 |     elif arch in ['s390x']: | 
 |         test_case.expect("register read r0", substrs=['r0 = 0x']) | 
 |     elif arch in ['powerpc64le']: | 
 |         test_case.expect("register read r0", substrs=['r0 = 0x']) | 
 |     else: | 
 |         # TODO: Add check for other architectures | 
 |         test_case.fail( | 
 |             "Unsupported architecture for test case (arch: %s)" % | 
 |             test_case.getArchitecture()) | 
 |  | 
 |  | 
 | def _run_adb_command(cmd, device_id): | 
 |     device_id_args = [] | 
 |     if device_id: | 
 |         device_id_args = ["-s", device_id] | 
 |     full_cmd = ["adb"] + device_id_args + cmd | 
 |     p = subprocess.Popen( | 
 |         full_cmd, | 
 |         stdout=subprocess.PIPE, | 
 |         stderr=subprocess.PIPE) | 
 |     stdout, stderr = p.communicate() | 
 |     return p.returncode, stdout, stderr | 
 |  | 
 |  | 
 | def target_is_android(): | 
 |     if not hasattr(target_is_android, 'result'): | 
 |         triple = lldb.DBG.GetSelectedPlatform().GetTriple() | 
 |         match = re.match(".*-.*-.*-android", triple) | 
 |         target_is_android.result = match is not None | 
 |     return target_is_android.result | 
 |  | 
 |  | 
 | def android_device_api(): | 
 |     if not hasattr(android_device_api, 'result'): | 
 |         assert configuration.lldb_platform_url is not None | 
 |         device_id = None | 
 |         parsed_url = urlparse.urlparse(configuration.lldb_platform_url) | 
 |         host_name = parsed_url.netloc.split(":")[0] | 
 |         if host_name != 'localhost': | 
 |             device_id = host_name | 
 |             if device_id.startswith('[') and device_id.endswith(']'): | 
 |                 device_id = device_id[1:-1] | 
 |         retcode, stdout, stderr = _run_adb_command( | 
 |             ["shell", "getprop", "ro.build.version.sdk"], device_id) | 
 |         if retcode == 0: | 
 |             android_device_api.result = int(stdout) | 
 |         else: | 
 |             raise LookupError( | 
 |                 ">>> Unable to determine the API level of the Android device.\n" | 
 |                 ">>> stdout:\n%s\n" | 
 |                 ">>> stderr:\n%s\n" % | 
 |                 (stdout, stderr)) | 
 |     return android_device_api.result | 
 |  | 
 |  | 
 | def match_android_device(device_arch, valid_archs=None, valid_api_levels=None): | 
 |     if not target_is_android(): | 
 |         return False | 
 |     if valid_archs is not None and device_arch not in valid_archs: | 
 |         return False | 
 |     if valid_api_levels is not None and android_device_api() not in valid_api_levels: | 
 |         return False | 
 |  | 
 |     return True | 
 |  | 
 |  | 
 | def finalize_build_dictionary(dictionary): | 
 |     if target_is_android(): | 
 |         if dictionary is None: | 
 |             dictionary = {} | 
 |         dictionary["OS"] = "Android" | 
 |         dictionary["PIE"] = 1 | 
 |     return dictionary | 
 |  | 
 |  | 
 | def getHostPlatform(): | 
 |     """Returns the host platform running the test suite.""" | 
 |     # Attempts to return a platform name matching a target Triple platform. | 
 |     if sys.platform.startswith('linux'): | 
 |         return 'linux' | 
 |     elif sys.platform.startswith('win32') or sys.platform.startswith('cygwin'): | 
 |         return 'windows' | 
 |     elif sys.platform.startswith('darwin'): | 
 |         return 'darwin' | 
 |     elif sys.platform.startswith('freebsd'): | 
 |         return 'freebsd' | 
 |     elif sys.platform.startswith('netbsd'): | 
 |         return 'netbsd' | 
 |     else: | 
 |         return sys.platform | 
 |  | 
 |  | 
 | def getDarwinOSTriples(): | 
 |     return ['darwin', 'macosx', 'ios', 'watchos', 'tvos', 'bridgeos'] | 
 |  | 
 |  | 
 | def getPlatform(): | 
 |     """Returns the target platform which the tests are running on.""" | 
 |     platform = lldb.DBG.GetSelectedPlatform().GetTriple().split('-')[2] | 
 |     if platform.startswith('freebsd'): | 
 |         platform = 'freebsd' | 
 |     elif platform.startswith('netbsd'): | 
 |         platform = 'netbsd' | 
 |     return platform | 
 |  | 
 |  | 
 | def platformIsDarwin(): | 
 |     """Returns true if the OS triple for the selected platform is any valid apple OS""" | 
 |     return getPlatform() in getDarwinOSTriples() | 
 |  | 
 |  | 
 | def findMainThreadCheckerDylib(): | 
 |     if not platformIsDarwin(): | 
 |         return "" | 
 |  | 
 |     with os.popen('xcode-select -p') as output: | 
 |         xcode_developer_path = output.read().strip() | 
 |         mtc_dylib_path = '%s/usr/lib/libMainThreadChecker.dylib' % xcode_developer_path | 
 |         if os.path.isfile(mtc_dylib_path): | 
 |             return mtc_dylib_path | 
 |  | 
 |     return "" | 
 |  | 
 |  | 
 | class _PlatformContext(object): | 
 |     """Value object class which contains platform-specific options.""" | 
 |  | 
 |     def __init__(self, shlib_environment_var, shlib_prefix, shlib_extension): | 
 |         self.shlib_environment_var = shlib_environment_var | 
 |         self.shlib_prefix = shlib_prefix | 
 |         self.shlib_extension = shlib_extension | 
 |  | 
 |  | 
 | def createPlatformContext(): | 
 |     if platformIsDarwin(): | 
 |         return _PlatformContext('DYLD_LIBRARY_PATH', 'lib', 'dylib') | 
 |     elif getPlatform() in ("freebsd", "linux", "netbsd"): | 
 |         return _PlatformContext('LD_LIBRARY_PATH', 'lib', 'so') | 
 |     else: | 
 |         return None | 
 |  | 
 |  | 
 | def hasChattyStderr(test_case): | 
 |     """Some targets produce garbage on the standard error output. This utility function | 
 |     determines whether the tests can be strict about the expected stderr contents.""" | 
 |     if match_android_device(test_case.getArchitecture(), ['aarch64'], range(22, 25+1)): | 
 |         return True  # The dynamic linker on the device will complain about unknown DT entries | 
 |     return False |