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