| import os |
| import platform |
| import re |
| import subprocess |
| import sys |
| |
| import lit.formats |
| import lit.util |
| |
| from lit.llvm import llvm_config |
| from lit.llvm.subst import ToolSubst |
| |
| # Configuration file for the 'lit' test runner. |
| |
| # name: The name of this test suite. |
| config.name = "cross-project-tests" |
| |
| # testFormat: The test format to use to interpret tests. |
| config.test_format = lit.formats.ShTest(not llvm_config.use_lit_shell) |
| |
| # suffixes: A list of file extensions to treat as test files. |
| config.suffixes = [".c", ".cl", ".cpp", ".m", ".test"] |
| |
| # excludes: A list of directories to exclude from the testsuite. The 'Inputs' |
| # subdirectories contain auxiliary inputs for various tests in their parent |
| # directories. |
| config.excludes = ["Inputs"] |
| |
| # test_source_root: The root path where tests are located. |
| config.test_source_root = config.cross_project_tests_src_root |
| |
| # test_exec_root: The root path where tests should be run. |
| config.test_exec_root = config.cross_project_tests_obj_root |
| |
| llvm_config.use_default_substitutions() |
| |
| lldb_python_path = os.path.join( |
| config.llvm_libs_dir, |
| f"python{sys.version_info.major}.{sys.version_info.minor}", |
| "site-packages", |
| ) |
| python_exec_path = sys.executable |
| tools = [ |
| ToolSubst( |
| "%test_debuginfo", |
| command="PYTHON_EXEC_PATH=" |
| + python_exec_path |
| + " LLDB_PYTHON_PATH=" |
| + lldb_python_path |
| + " " |
| + os.path.join( |
| config.cross_project_tests_src_root, |
| "debuginfo-tests", |
| "llgdb-tests", |
| "test_debuginfo.pl", |
| ), |
| ), |
| ToolSubst("%llvm_src_root", config.llvm_src_root), |
| ToolSubst("%llvm_tools_dir", config.llvm_tools_dir), |
| ] |
| |
| |
| def get_required_attr(config, attr_name): |
| attr_value = getattr(config, attr_name, None) |
| if attr_value is None: |
| lit_config.fatal( |
| "No attribute %r in test configuration! You may need to run " |
| "tests from your build directory or add this attribute " |
| "to lit.site.cfg " % attr_name |
| ) |
| return attr_value |
| |
| |
| # If this is an MSVC environment, the tests at the root of the tree are |
| # unsupported. The local win_cdb test suite, however, is supported. |
| is_msvc = get_required_attr(config, "is_msvc") |
| if is_msvc: |
| config.available_features.add("msvc") |
| # FIXME: We should add some llvm lit utility code to find the Windows SDK |
| # and set up the environment appopriately. |
| win_sdk = "C:/Program Files (x86)/Windows Kits/10/" |
| arch = "x64" |
| llvm_config.with_system_environment(["LIB", "LIBPATH", "INCLUDE"]) |
| # Clear _NT_SYMBOL_PATH to prevent cdb from attempting to load symbols from |
| # the network. |
| llvm_config.with_environment("_NT_SYMBOL_PATH", "") |
| tools.append( |
| ToolSubst("%cdb", '"%s"' % os.path.join(win_sdk, "Debuggers", arch, "cdb.exe")) |
| ) |
| |
| # clang_src_dir and lld_src_dir are not used by these tests, but are required by |
| # use_clang() and use_lld() respectively, so set them to "", if needed. |
| if not hasattr(config, "clang_src_dir"): |
| config.clang_src_dir = "" |
| llvm_config.use_clang(required=("clang" in config.llvm_enabled_projects)) |
| |
| if not hasattr(config, "lld_src_dir"): |
| config.lld_src_dir = "" |
| llvm_config.use_lld(required=("lld" in config.llvm_enabled_projects)) |
| |
| if "compiler-rt" in config.llvm_enabled_projects: |
| config.available_features.add("compiler-rt") |
| |
| # Check which debuggers are available: |
| lldb_dap_path = llvm_config.use_llvm_tool("lldb-dap") |
| if lldb_dap_path is not None: |
| config.available_features.add("lldb") |
| |
| if llvm_config.use_llvm_tool("llvm-ar"): |
| config.available_features.add("llvm-ar") |
| |
| |
| def configure_dexter_substitutions(): |
| """Configure substitutions for host platform and return list of dependencies""" |
| # Produce dexter path, lldb path, and combine into the %dexter substitution |
| # for running a test. |
| dexter_path = os.path.join( |
| config.cross_project_tests_src_root, "debuginfo-tests", "dexter", "dexter.py" |
| ) |
| tools.append(ToolSubst("%dexter", f'"{sys.executable}" "{dexter_path}" test -v')) |
| if lldb_dap_path is not None: |
| tools.append( |
| ToolSubst( |
| "%dexter_lldb_args", |
| f'--lldb-executable "{lldb_dap_path}" --debugger lldb-dap --dap-message-log=-e', |
| ) |
| ) |
| |
| # For testing other bits of dexter that aren't under the "test" subcommand, |
| # have a %dexter_base substitution. |
| dexter_base_cmd = '"{}" "{}"'.format(sys.executable, dexter_path) |
| tools.append(ToolSubst("%dexter_base", dexter_base_cmd)) |
| |
| # Set up commands for DexTer regression tests. |
| # Builder, debugger, optimisation level and several other flags differ |
| # depending on whether we're running a unix like or windows os. |
| if platform.system() == "Windows": |
| # The Windows builder script uses lld. |
| dependencies = ["clang", "lld-link"] |
| dexter_regression_test_c_builder = "clang-cl" |
| dexter_regression_test_cxx_builder = "clang-cl" |
| dexter_regression_test_debugger = "dbgeng" |
| dexter_regression_test_c_flags = "/Zi /Od" |
| dexter_regression_test_cxx_flags = "/Zi /Od" |
| dexter_regression_test_additional_flags = "" |
| else: |
| # Use lldb as the debugger on non-Windows platforms. |
| dependencies = ["clang", "lldb"] |
| dexter_regression_test_c_builder = "clang" |
| dexter_regression_test_cxx_builder = "clang++" |
| dexter_regression_test_debugger = "lldb-dap" |
| dexter_regression_test_additional_flags = ( |
| f'--lldb-executable "{lldb_dap_path}" --dap-message-log=-e' |
| ) |
| dexter_regression_test_c_flags = "-O0 -glldb -std=gnu11" |
| dexter_regression_test_cxx_flags = "-O0 -glldb -std=gnu++11" |
| |
| tools.append( |
| ToolSubst( |
| "%dexter_regression_test_debugger_args", |
| f"--debugger {dexter_regression_test_debugger} {dexter_regression_test_additional_flags}", |
| ) |
| ) |
| |
| # Typical command would take the form: |
| # ./path_to_py/python.exe ./path_to_dex/dexter.py test --fail-lt 1.0 -w --binary %t --debugger lldb --cflags '-O0 -g' |
| dexter_regression_test_run = " ".join( |
| # "python", "dexter.py", test, fail_mode, builder, debugger, cflags, ldflags |
| [ |
| '"{}"'.format(sys.executable), |
| '"{}"'.format(dexter_path), |
| "test", |
| "--fail-lt 1.0 -w -v", |
| "--debugger", |
| dexter_regression_test_debugger, |
| dexter_regression_test_additional_flags, |
| ] |
| ) |
| tools.append(ToolSubst("%dexter_regression_test_run", dexter_regression_test_run)) |
| |
| # Include build flags for %dexter_regression_test. |
| dexter_regression_test_c_build = " ".join( |
| [ |
| dexter_regression_test_c_builder, |
| dexter_regression_test_c_flags, |
| ] |
| ) |
| dexter_regression_test_cxx_build = " ".join( |
| [ |
| dexter_regression_test_cxx_builder, |
| dexter_regression_test_cxx_flags, |
| ] |
| ) |
| tools.append( |
| ToolSubst("%dexter_regression_test_c_build", dexter_regression_test_c_build) |
| ) |
| tools.append( |
| ToolSubst("%dexter_regression_test_cxx_build", dexter_regression_test_cxx_build) |
| ) |
| return dependencies |
| |
| |
| def add_host_triple(clang): |
| return "{} --target={}".format(clang, config.host_triple) |
| |
| |
| # The set of arches we can build. |
| targets = set(config.targets_to_build) |
| # Add aliases to the target set. |
| if "AArch64" in targets: |
| targets.add("arm64") |
| if "ARM" in config.targets_to_build: |
| targets.add("thumbv7") |
| |
| |
| def can_target_host(): |
| # Check if the targets set contains anything that looks like our host arch. |
| # The arch name in the triple and targets set may be spelled differently |
| # (e.g. x86 vs X86). |
| return any(config.host_triple.lower().startswith(x.lower()) for x in targets) |
| |
| |
| # Dexter tests run on the host machine. If the host arch is supported add |
| # 'dexter' as an available feature and force the dexter tests to use the host |
| # triple. |
| if can_target_host(): |
| if config.host_triple != config.target_triple: |
| print("Forcing dexter tests to use host triple {}.".format(config.host_triple)) |
| dependencies = configure_dexter_substitutions() |
| if all(d in config.available_features for d in dependencies): |
| config.available_features.add("dexter") |
| else: |
| print( |
| "Host triple {} not supported. Skipping dexter tests in the " |
| "debuginfo-tests project.".format(config.host_triple) |
| ) |
| |
| tool_dirs = [config.llvm_tools_dir] |
| |
| llvm_config.add_tool_substitutions(tools, tool_dirs) |
| |
| lit.util.usePlatformSdkOnDarwin(config, lit_config) |
| |
| |
| def get_gdb_version_string(): |
| """Return gdb's version string, or None if gdb cannot be found or the |
| --version output is formatted unexpectedly. |
| """ |
| # See if we can get a gdb version, e.g. |
| # $ gdb --version |
| # GNU gdb (GDB) 10.2 |
| # ...More stuff... |
| try: |
| gdb_vers_lines = ( |
| subprocess.check_output(["gdb", "--version"]).decode().splitlines() |
| ) |
| except: |
| return None # We couldn't find gdb or something went wrong running it. |
| if len(gdb_vers_lines) < 1: |
| print("Unknown GDB version format (too few lines)", file=sys.stderr) |
| return None |
| match = re.search(r"GNU gdb \(.*?\) ((\d|\.)+)", gdb_vers_lines[0].strip()) |
| if match is None: |
| print(f"Unknown GDB version format: {gdb_vers_lines[0]}", file=sys.stderr) |
| return None |
| return match.group(1) |
| |
| |
| def get_clang_default_dwarf_version_string(triple): |
| """Return the default dwarf version string for clang on this (host) platform |
| or None if we can't work it out. |
| """ |
| # Get the flags passed by the driver and look for -dwarf-version. |
| cmd = f'{llvm_config.use_llvm_tool("clang")} -g -xc -c - -v -### --target={triple}' |
| stderr = subprocess.run(cmd.split(), stderr=subprocess.PIPE).stderr.decode() |
| match = re.search(r"-dwarf-version=(\d+)", stderr) |
| if match is None: |
| print("Cannot determine default dwarf version", file=sys.stderr) |
| return None |
| return match.group(1) |
| |
| |
| def get_lldb_version_string(): |
| """Return LLDB's version string, or None if lldb cannot be found or the |
| --version output is formatted unexpectedly. |
| """ |
| try: |
| if platform.system() == "Darwin": |
| # On Darwin, use system lldb which has Apple-specific versioning. |
| cmd = ["xcrun", "lldb", "--version"] |
| else: |
| # On non-Darwin, use the locally-built lldb from llvm_tools_dir. |
| lldb_path = os.path.join(config.llvm_tools_dir, "lldb") |
| if not os.path.exists(lldb_path): |
| print(f"LLDB not found at {lldb_path}", file=sys.stderr) |
| return None |
| cmd = [lldb_path, "--version"] |
| |
| lldb_vers_lines = subprocess.check_output(cmd).decode().splitlines() |
| except: |
| return None |
| if len(lldb_vers_lines) < 1: |
| print("Unknown LLDB version format (too few lines)", file=sys.stderr) |
| return None |
| match = re.search(r"lldb.*?[ -]((\d|\.)+)", lldb_vers_lines[0].strip()) |
| if match is None: |
| print(f"Unknown LLDB version format: {lldb_vers_lines[0]}", file=sys.stderr) |
| return None |
| return match.group(1) |
| |
| |
| def set_lldb_formatters_compatibility_feature(): |
| current_lldb_version = get_lldb_version_string() |
| if current_lldb_version: |
| print( |
| f"Found LLDB version '{current_lldb_version}'", |
| file=sys.stderr, |
| ) |
| else: |
| print( |
| "No LLDB found on host. Skipping tests that require LLDB.", |
| file=sys.stderr, |
| ) |
| return |
| |
| if platform.system() == "Darwin": |
| # The Apple LLDB version doesn't follow the LLVM release versioning. |
| min_required_lldb_version = "1700" |
| else: |
| # Minimum version required for SBType::FindDirectNestedType API |
| # which some LLVM data formatters depend on. |
| min_required_lldb_version = "19.0.0" |
| |
| try: |
| from packaging import version |
| except: |
| lit_config.fatal("Running lldb tests requires the packaging package") |
| return |
| |
| if version.parse(current_lldb_version) < version.parse(min_required_lldb_version): |
| raise ValueError( |
| f"using version {current_lldb_version} whereas a version >= {min_required_lldb_version} is required" |
| ) |
| |
| config.available_features.add("lldb-formatters-compatibility") |
| |
| |
| def set_apple_lldb_pre_1000_feature(): |
| apple_lldb_vers = get_lldb_version_string() |
| if not apple_lldb_vers: |
| return |
| |
| try: |
| from packaging import version |
| except: |
| lit_config.fatal("Running lldb tests requires the packaging package") |
| return |
| |
| if version.parse(apple_lldb_vers) < version.parse("1000"): |
| config.available_features.add("apple-lldb-pre-1000") |
| |
| |
| # Some cross-project-tests use gdb, but not all versions of gdb are compatible |
| # with clang's dwarf. Add feature `gdb-clang-incompatibility` to signal that |
| # there's an incompatibility between clang's default dwarf version for this |
| # platform and the installed gdb version. |
| dwarf_version_string = get_clang_default_dwarf_version_string(config.host_triple) |
| gdb_version_string = get_gdb_version_string() |
| |
| if gdb_version_string: |
| config.available_features.add("has-gdb") |
| print( |
| f"Found GDB version '{gdb_version_string}'", |
| file=sys.stderr, |
| ) |
| else: |
| print( |
| "No GDB found on host. Skipping tests that require GDB.", |
| file=sys.stderr, |
| ) |
| |
| if dwarf_version_string and gdb_version_string: |
| if int(dwarf_version_string) >= 5: |
| try: |
| from packaging import version |
| except: |
| lit_config.fatal("Running gdb tests requires the packaging package") |
| if version.parse(gdb_version_string) < version.parse("10.1"): |
| # Example for llgdb-tests, which use lldb on darwin but gdb elsewhere: |
| # XFAIL: !system-darwin && gdb-clang-incompatibility |
| config.available_features.add("gdb-clang-incompatibility") |
| print( |
| "XFAIL some tests: use gdb version >= 10.1 to restore test coverage", |
| file=sys.stderr, |
| ) |
| |
| try: |
| set_lldb_formatters_compatibility_feature() |
| except ValueError as e: |
| print( |
| f"Marking some LLDB LLVM data-formatter tests as unsupported: {e}", |
| file=sys.stderr, |
| ) |
| |
| if platform.system() == "Darwin": |
| set_apple_lldb_pre_1000_feature() |
| |
| llvm_config.feature_config([("--build-mode", {"Debug|RelWithDebInfo": "debug-info"})]) |
| |
| # Allow 'REQUIRES: XXX-registered-target' in tests. |
| for arch in config.targets_to_build: |
| config.available_features.add(arch.lower() + "-registered-target") |
| |
| |
| def find_dbgeng(): |
| if platform.system() != "Windows": |
| return None |
| |
| for path in os.environ.get("PATH", "").split(os.pathsep): |
| p = os.path.join(path, "dbgeng.dll") |
| if os.path.exists(p) and not os.path.isdir(p): |
| return os.path.abspath(p) |
| |
| return None |
| |
| |
| def get_dbgeng_version(): |
| dbgeng = find_dbgeng() |
| if not dbgeng: |
| return None |
| |
| try: |
| import win32api |
| except: |
| return None |
| |
| info = win32api.GetFileVersionInfo(dbgeng, "\\") |
| ms = info["FileVersionMS"] |
| ls = info["FileVersionLS"] |
| return ( |
| win32api.HIWORD(ms), |
| win32api.LOWORD(ms), |
| win32api.HIWORD(ls), |
| win32api.LOWORD(ls), |
| ) |
| |
| |
| dbgeng_version = get_dbgeng_version() |
| if dbgeng_version and dbgeng_version >= (10, 0, 19041, 0): |
| config.available_features.add("dbgeng-10-19041") |