xref: /linux/tools/testing/selftests/x86/bugs/its_permutations.py (revision 6f5bf947bab06f37ff931c359fd5770c4d9cbf87)
1*7a9b709eSPawan Gupta#!/usr/bin/env python3
2*7a9b709eSPawan Gupta# SPDX-License-Identifier: GPL-2.0
3*7a9b709eSPawan Gupta#
4*7a9b709eSPawan Gupta# Copyright (c) 2025 Intel Corporation
5*7a9b709eSPawan Gupta#
6*7a9b709eSPawan Gupta# Test for indirect target selection (ITS) cmdline permutations with other bugs
7*7a9b709eSPawan Gupta# like spectre_v2 and retbleed.
8*7a9b709eSPawan Gupta
9*7a9b709eSPawan Guptaimport os, sys, subprocess, itertools, re, shutil
10*7a9b709eSPawan Gupta
11*7a9b709eSPawan Guptatest_dir = os.path.dirname(os.path.realpath(__file__))
12*7a9b709eSPawan Guptasys.path.insert(0, test_dir + '/../../kselftest')
13*7a9b709eSPawan Guptaimport ksft
14*7a9b709eSPawan Guptaimport common as c
15*7a9b709eSPawan Gupta
16*7a9b709eSPawan Guptabug = "indirect_target_selection"
17*7a9b709eSPawan Guptamitigation = c.get_sysfs(bug)
18*7a9b709eSPawan Gupta
19*7a9b709eSPawan Guptaif not mitigation or "Not affected" in mitigation:
20*7a9b709eSPawan Gupta    ksft.test_result_skip("Skipping its_permutations.py: not applicable")
21*7a9b709eSPawan Gupta    ksft.finished()
22*7a9b709eSPawan Gupta
23*7a9b709eSPawan Guptaif shutil.which('vng') is None:
24*7a9b709eSPawan Gupta    ksft.test_result_skip("Skipping its_permutations.py: virtme-ng ('vng') not found in PATH.")
25*7a9b709eSPawan Gupta    ksft.finished()
26*7a9b709eSPawan Gupta
27*7a9b709eSPawan GuptaTEST = f"{test_dir}/its_sysfs.py"
28*7a9b709eSPawan Guptadefault_kparam = ['clearcpuid=hypervisor', 'panic=5', 'panic_on_warn=1', 'oops=panic', 'nmi_watchdog=1', 'hung_task_panic=1']
29*7a9b709eSPawan Gupta
30*7a9b709eSPawan GuptaDEBUG = " -v "
31*7a9b709eSPawan Gupta
32*7a9b709eSPawan Gupta# Install dependencies
33*7a9b709eSPawan Gupta# https://github.com/arighi/virtme-ng
34*7a9b709eSPawan Gupta# apt install virtme-ng
35*7a9b709eSPawan GuptaBOOT_CMD = f"vng --run {test_dir}/../../../../../arch/x86/boot/bzImage "
36*7a9b709eSPawan Gupta#BOOT_CMD += DEBUG
37*7a9b709eSPawan Gupta
38*7a9b709eSPawan Guptabug = "indirect_target_selection"
39*7a9b709eSPawan Gupta
40*7a9b709eSPawan Guptainput_options = {
41*7a9b709eSPawan Gupta    'indirect_target_selection'     : ['off', 'on', 'stuff', 'vmexit'],
42*7a9b709eSPawan Gupta    'retbleed'                      : ['off', 'stuff', 'auto'],
43*7a9b709eSPawan Gupta    'spectre_v2'                    : ['off', 'on', 'eibrs', 'retpoline', 'ibrs', 'eibrs,retpoline'],
44*7a9b709eSPawan Gupta}
45*7a9b709eSPawan Gupta
46*7a9b709eSPawan Guptadef pretty_print(output):
47*7a9b709eSPawan Gupta    OKBLUE = '\033[94m'
48*7a9b709eSPawan Gupta    OKGREEN = '\033[92m'
49*7a9b709eSPawan Gupta    WARNING = '\033[93m'
50*7a9b709eSPawan Gupta    FAIL = '\033[91m'
51*7a9b709eSPawan Gupta    ENDC = '\033[0m'
52*7a9b709eSPawan Gupta    BOLD = '\033[1m'
53*7a9b709eSPawan Gupta
54*7a9b709eSPawan Gupta    # Define patterns and their corresponding colors
55*7a9b709eSPawan Gupta    patterns = {
56*7a9b709eSPawan Gupta        r"^ok \d+": OKGREEN,
57*7a9b709eSPawan Gupta        r"^not ok \d+": FAIL,
58*7a9b709eSPawan Gupta        r"^# Testing .*": OKBLUE,
59*7a9b709eSPawan Gupta        r"^# Found: .*": WARNING,
60*7a9b709eSPawan Gupta        r"^# Totals: .*": BOLD,
61*7a9b709eSPawan Gupta        r"pass:([1-9]\d*)": OKGREEN,
62*7a9b709eSPawan Gupta        r"fail:([1-9]\d*)": FAIL,
63*7a9b709eSPawan Gupta        r"skip:([1-9]\d*)": WARNING,
64*7a9b709eSPawan Gupta    }
65*7a9b709eSPawan Gupta
66*7a9b709eSPawan Gupta    # Apply colors based on patterns
67*7a9b709eSPawan Gupta    for pattern, color in patterns.items():
68*7a9b709eSPawan Gupta        output = re.sub(pattern, lambda match: f"{color}{match.group(0)}{ENDC}", output, flags=re.MULTILINE)
69*7a9b709eSPawan Gupta
70*7a9b709eSPawan Gupta    print(output)
71*7a9b709eSPawan Gupta
72*7a9b709eSPawan Guptacombinations = list(itertools.product(*input_options.values()))
73*7a9b709eSPawan Guptaksft.print_header()
74*7a9b709eSPawan Guptaksft.set_plan(len(combinations))
75*7a9b709eSPawan Gupta
76*7a9b709eSPawan Guptalogs = ""
77*7a9b709eSPawan Gupta
78*7a9b709eSPawan Guptafor combination in combinations:
79*7a9b709eSPawan Gupta    append = ""
80*7a9b709eSPawan Gupta    log = ""
81*7a9b709eSPawan Gupta    for p in default_kparam:
82*7a9b709eSPawan Gupta        append += f' --append={p}'
83*7a9b709eSPawan Gupta    command = BOOT_CMD + append
84*7a9b709eSPawan Gupta    test_params = ""
85*7a9b709eSPawan Gupta    for i, key in enumerate(input_options.keys()):
86*7a9b709eSPawan Gupta        param = f'{key}={combination[i]}'
87*7a9b709eSPawan Gupta        test_params += f' {param}'
88*7a9b709eSPawan Gupta        command += f" --append={param}"
89*7a9b709eSPawan Gupta    command += f" -- {TEST}"
90*7a9b709eSPawan Gupta    test_name = f"{bug} {test_params}"
91*7a9b709eSPawan Gupta    pretty_print(f'# Testing {test_name}')
92*7a9b709eSPawan Gupta    t =  subprocess.Popen(command, shell=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
93*7a9b709eSPawan Gupta    t.wait()
94*7a9b709eSPawan Gupta    output, _ = t.communicate()
95*7a9b709eSPawan Gupta    if t.returncode == 0:
96*7a9b709eSPawan Gupta        ksft.test_result_pass(test_name)
97*7a9b709eSPawan Gupta    else:
98*7a9b709eSPawan Gupta        ksft.test_result_fail(test_name)
99*7a9b709eSPawan Gupta    output = output.decode()
100*7a9b709eSPawan Gupta    log += f" {output}"
101*7a9b709eSPawan Gupta    pretty_print(log)
102*7a9b709eSPawan Gupta    logs += output + "\n"
103*7a9b709eSPawan Gupta
104*7a9b709eSPawan Gupta# Optionally use tappy to parse the output
105*7a9b709eSPawan Gupta# apt install python3-tappy
106*7a9b709eSPawan Guptawith open("logs.txt", "w") as f:
107*7a9b709eSPawan Gupta    f.write(logs)
108*7a9b709eSPawan Gupta
109*7a9b709eSPawan Guptaksft.finished()
110