78 lines
3.2 KiB
Python
78 lines
3.2 KiB
Python
|
"""Test runner for data-flow analysis test cases."""
|
||
|
|
||
|
from __future__ import annotations
|
||
|
|
||
|
import os.path
|
||
|
|
||
|
from mypy.errors import CompileError
|
||
|
from mypy.test.config import test_temp_dir
|
||
|
from mypy.test.data import DataDrivenTestCase
|
||
|
from mypyc.analysis import dataflow
|
||
|
from mypyc.common import TOP_LEVEL_NAME
|
||
|
from mypyc.ir.func_ir import all_values
|
||
|
from mypyc.ir.ops import Value
|
||
|
from mypyc.ir.pprint import format_func, generate_names_for_ir
|
||
|
from mypyc.test.testutil import (
|
||
|
ICODE_GEN_BUILTINS,
|
||
|
MypycDataSuite,
|
||
|
assert_test_output,
|
||
|
build_ir_for_single_file,
|
||
|
use_custom_builtins,
|
||
|
)
|
||
|
from mypyc.transform import exceptions
|
||
|
|
||
|
files = ["analysis.test"]
|
||
|
|
||
|
|
||
|
class TestAnalysis(MypycDataSuite):
|
||
|
files = files
|
||
|
base_path = test_temp_dir
|
||
|
optional_out = True
|
||
|
|
||
|
def run_case(self, testcase: DataDrivenTestCase) -> None:
|
||
|
"""Perform a data-flow analysis test case."""
|
||
|
|
||
|
with use_custom_builtins(os.path.join(self.data_prefix, ICODE_GEN_BUILTINS), testcase):
|
||
|
try:
|
||
|
ir = build_ir_for_single_file(testcase.input)
|
||
|
except CompileError as e:
|
||
|
actual = e.messages
|
||
|
else:
|
||
|
actual = []
|
||
|
for fn in ir:
|
||
|
if fn.name == TOP_LEVEL_NAME and not testcase.name.endswith("_toplevel"):
|
||
|
continue
|
||
|
exceptions.insert_exception_handling(fn)
|
||
|
actual.extend(format_func(fn))
|
||
|
cfg = dataflow.get_cfg(fn.blocks)
|
||
|
args: set[Value] = set(fn.arg_regs)
|
||
|
name = testcase.name
|
||
|
if name.endswith("_MaybeDefined"):
|
||
|
# Forward, maybe
|
||
|
analysis_result = dataflow.analyze_maybe_defined_regs(fn.blocks, cfg, args)
|
||
|
elif name.endswith("_Liveness"):
|
||
|
# Backward, maybe
|
||
|
analysis_result = dataflow.analyze_live_regs(fn.blocks, cfg)
|
||
|
elif name.endswith("_MustDefined"):
|
||
|
# Forward, must
|
||
|
analysis_result = dataflow.analyze_must_defined_regs(
|
||
|
fn.blocks, cfg, args, regs=all_values(fn.arg_regs, fn.blocks)
|
||
|
)
|
||
|
elif name.endswith("_BorrowedArgument"):
|
||
|
# Forward, must
|
||
|
analysis_result = dataflow.analyze_borrowed_arguments(fn.blocks, cfg, args)
|
||
|
else:
|
||
|
assert False, "No recognized _AnalysisName suffix in test case"
|
||
|
|
||
|
names = generate_names_for_ir(fn.arg_regs, fn.blocks)
|
||
|
|
||
|
for key in sorted(
|
||
|
analysis_result.before.keys(), key=lambda x: (x[0].label, x[1])
|
||
|
):
|
||
|
pre = ", ".join(sorted(names[reg] for reg in analysis_result.before[key]))
|
||
|
post = ", ".join(sorted(names[reg] for reg in analysis_result.after[key]))
|
||
|
actual.append(
|
||
|
"%-8s %-23s %s" % ((key[0].label, key[1]), "{%s}" % pre, "{%s}" % post)
|
||
|
)
|
||
|
assert_test_output(testcase, actual, "Invalid source code output")
|