1*4d495c6eSApple OSS Distributions#!/usr/bin/python3 2*4d495c6eSApple OSS Distributions 3*4d495c6eSApple OSS Distributions# format_vm_parameter_validation.py 4*4d495c6eSApple OSS Distributions# Pretty-print the output of tests/vm/vm_parameter_validation.c 5*4d495c6eSApple OSS Distributions# 6*4d495c6eSApple OSS Distributions# usage: 7*4d495c6eSApple OSS Distributions# vm_parameter_validation | format_vm_parameter_validation.py 8*4d495c6eSApple OSS Distributions 9*4d495c6eSApple OSS Distributionsimport re 10*4d495c6eSApple OSS Distributionsimport sys 11*4d495c6eSApple OSS Distributionsimport copy 12*4d495c6eSApple OSS Distributionsimport itertools 13*4d495c6eSApple OSS Distributions 14*4d495c6eSApple OSS Distributions# magic return values used for in-band signalling 15*4d495c6eSApple OSS Distributions# fixme duplicated in vm_parameter_validation.c 16*4d495c6eSApple OSS Distributions# fixme also duplicated in other_return_values below 17*4d495c6eSApple OSS DistributionsRESULT_SUCCESS = 0 18*4d495c6eSApple OSS DistributionsRESULT_BUSTED = -99 19*4d495c6eSApple OSS DistributionsRESULT_IGNORED = -98 20*4d495c6eSApple OSS DistributionsRESULT_ZEROSIZE = -97 21*4d495c6eSApple OSS DistributionsRESULT_PANIC = -96 22*4d495c6eSApple OSS DistributionsRESULT_GUARD = -95 23*4d495c6eSApple OSS DistributionsRESULT_MISMATCH = -94 24*4d495c6eSApple OSS DistributionsRESULT_OUT_PARAM_BAD = -93 25*4d495c6eSApple OSS Distributions# Some Mach errors use their normal integer values, 26*4d495c6eSApple OSS Distributions# but we handle them specially here because those 27*4d495c6eSApple OSS Distributions# integers are too long to fit in the grid output. 28*4d495c6eSApple OSS DistributionsRESULT_MACH_SEND_INVALID_MEMORY = 0x1000000c 29*4d495c6eSApple OSS DistributionsRESULT_MACH_SEND_INVALID_DEST = 0x10000003 30*4d495c6eSApple OSS Distributions 31*4d495c6eSApple OSS Distributions# output formatting 32*4d495c6eSApple OSS Distributionsformat_result = { 33*4d495c6eSApple OSS Distributions RESULT_SUCCESS : ' .', 34*4d495c6eSApple OSS Distributions RESULT_BUSTED : ' **', 35*4d495c6eSApple OSS Distributions RESULT_MISMATCH : ' ##', 36*4d495c6eSApple OSS Distributions RESULT_IGNORED : ' ', 37*4d495c6eSApple OSS Distributions RESULT_ZEROSIZE : ' o', 38*4d495c6eSApple OSS Distributions RESULT_PANIC : ' pp', 39*4d495c6eSApple OSS Distributions RESULT_GUARD : ' gg', 40*4d495c6eSApple OSS Distributions RESULT_OUT_PARAM_BAD : ' ot', 41*4d495c6eSApple OSS Distributions RESULT_MACH_SEND_INVALID_MEMORY : ' mi', 42*4d495c6eSApple OSS Distributions RESULT_MACH_SEND_INVALID_DEST : ' md', 43*4d495c6eSApple OSS Distributions} 44*4d495c6eSApple OSS Distributions 45*4d495c6eSApple OSS Distributions# same as format_result, but for functions 46*4d495c6eSApple OSS Distributions# where 0=failure and 1=success 47*4d495c6eSApple OSS Distributionsformat_bool_result = format_result.copy() 48*4d495c6eSApple OSS Distributionsformat_bool_result.update({ 49*4d495c6eSApple OSS Distributions 0 : ' x', 50*4d495c6eSApple OSS Distributions 1 : format_result[RESULT_SUCCESS], 51*4d495c6eSApple OSS Distributions}) 52*4d495c6eSApple OSS Distributions 53*4d495c6eSApple OSS Distributionsdef formatter_for_testname(testname): 54*4d495c6eSApple OSS Distributions if (error_code_values_for_testname(testname) == bool_return_values): 55*4d495c6eSApple OSS Distributions return format_bool_result 56*4d495c6eSApple OSS Distributions return format_result 57*4d495c6eSApple OSS Distributions 58*4d495c6eSApple OSS Distributionsformat_default = '%3d' 59*4d495c6eSApple OSS Distributionsformat_col_width = 3 60*4d495c6eSApple OSS Distributionsformat_empty_col = format_col_width * ' ' 61*4d495c6eSApple OSS Distributionsformat_indent_width = 4 62*4d495c6eSApple OSS Distributionsformat_indent = format_indent_width * ' ' 63*4d495c6eSApple OSS Distributions 64*4d495c6eSApple OSS Distributions 65*4d495c6eSApple OSS Distributions# record the result of one trial: 66*4d495c6eSApple OSS Distributions# ret: the return value from the tested function 67*4d495c6eSApple OSS Distributions# parameters: array of the input parameter names for that trial 68*4d495c6eSApple OSS Distributions# (for example ["start PGSZ-2", "size -1"]) 69*4d495c6eSApple OSS Distributionsclass Result: 70*4d495c6eSApple OSS Distributions def __init__(self, new_ret, new_parameters): 71*4d495c6eSApple OSS Distributions self.ret = new_ret 72*4d495c6eSApple OSS Distributions self.parameters = new_parameters 73*4d495c6eSApple OSS Distributions def __repr__(self): 74*4d495c6eSApple OSS Distributions return str(self.ret) + " = " + str(self.parameters) 75*4d495c6eSApple OSS Distributions 76*4d495c6eSApple OSS Distributions# record the results of all trials in one test 77*4d495c6eSApple OSS Distributions# testname: the name of the test (including the function being tested) 78*4d495c6eSApple OSS Distributions# config: a string describing OS, CPU, etc 79*4d495c6eSApple OSS Distributions# compat: code for error compatibility 80*4d495c6eSApple OSS Distributions# results: an array of Result, one per trial 81*4d495c6eSApple OSS Distributionsclass Test: 82*4d495c6eSApple OSS Distributions def __init__(self, new_name, new_config, new_compat, new_results = []): 83*4d495c6eSApple OSS Distributions self.testname = new_name 84*4d495c6eSApple OSS Distributions self.config = new_config 85*4d495c6eSApple OSS Distributions self.compat = new_compat 86*4d495c6eSApple OSS Distributions self.results = new_results 87*4d495c6eSApple OSS Distributions 88*4d495c6eSApple OSS Distributions# print column labels under some output 89*4d495c6eSApple OSS Distributions# example output given indent=2 col_width=4 labels=[foo,bar,baz,qux]: 90*4d495c6eSApple OSS Distributions# | | | | 91*4d495c6eSApple OSS Distributions# | | | qux 92*4d495c6eSApple OSS Distributions# | | baz 93*4d495c6eSApple OSS Distributions# | bar 94*4d495c6eSApple OSS Distributions# foo 95*4d495c6eSApple OSS Distributionsdef print_column_labels(labels, indent_width, col_width): 96*4d495c6eSApple OSS Distributions indent = indent_width * ' ' 97*4d495c6eSApple OSS Distributions empty_column = '|' + (col_width-1) * ' ' 98*4d495c6eSApple OSS Distributions 99*4d495c6eSApple OSS Distributions unprinted = len(labels) 100*4d495c6eSApple OSS Distributions print(indent + unprinted*empty_column) 101*4d495c6eSApple OSS Distributions 102*4d495c6eSApple OSS Distributions for label in reversed(labels): 103*4d495c6eSApple OSS Distributions unprinted -= 1 104*4d495c6eSApple OSS Distributions print(indent + unprinted*empty_column + label) 105*4d495c6eSApple OSS Distributions 106*4d495c6eSApple OSS Distributions# pretty-print one function return code 107*4d495c6eSApple OSS Distributionsdef print_one_result(ret, formatter): 108*4d495c6eSApple OSS Distributions if ret in formatter: 109*4d495c6eSApple OSS Distributions print(formatter[ret], end='') 110*4d495c6eSApple OSS Distributions else: 111*4d495c6eSApple OSS Distributions print(format_default % (ret), end='') 112*4d495c6eSApple OSS Distributions 113*4d495c6eSApple OSS Distributions# choose the appropriate error code table for a test 114*4d495c6eSApple OSS Distributions# (either errno_return_values, bool_return_values, or kern_return_values) 115*4d495c6eSApple OSS Distributionsdef error_code_values_for_testname(testname): 116*4d495c6eSApple OSS Distributions errno_fns = ['mprotect', 'msync', 'minherit', 'mincore', 'mlock', 'munlock', 117*4d495c6eSApple OSS Distributions 'mmap', 'munmap', 'mremap_encrypted', 'vslock', 'vsunlock', 118*4d495c6eSApple OSS Distributions 'madvise'] 119*4d495c6eSApple OSS Distributions bool_fns = ['useracc', 'task_find_region_details'] 120*4d495c6eSApple OSS Distributions for fn in errno_fns: 121*4d495c6eSApple OSS Distributions if testname.startswith(fn): 122*4d495c6eSApple OSS Distributions return errno_return_values 123*4d495c6eSApple OSS Distributions for fn in bool_fns: 124*4d495c6eSApple OSS Distributions if testname.startswith(fn): 125*4d495c6eSApple OSS Distributions return bool_return_values 126*4d495c6eSApple OSS Distributions return kern_return_values 127*4d495c6eSApple OSS Distributions 128*4d495c6eSApple OSS Distributions# print a helpful description of the return values seen in results 129*4d495c6eSApple OSS Distributions# fixme these won't include RESULT_MISMATCH 130*4d495c6eSApple OSS Distributionsdef print_legend(test): 131*4d495c6eSApple OSS Distributions # find all error codes represented in the results 132*4d495c6eSApple OSS Distributions codes = {} 133*4d495c6eSApple OSS Distributions for result in test.results: 134*4d495c6eSApple OSS Distributions codes[result.ret] = True 135*4d495c6eSApple OSS Distributions 136*4d495c6eSApple OSS Distributions known_return_values = error_code_values_for_testname(test.testname) 137*4d495c6eSApple OSS Distributions 138*4d495c6eSApple OSS Distributions # print the names of the detected error codes 139*4d495c6eSApple OSS Distributions output = [] 140*4d495c6eSApple OSS Distributions for code in sorted(codes.keys()): 141*4d495c6eSApple OSS Distributions if code in known_return_values: 142*4d495c6eSApple OSS Distributions output.append(known_return_values[code]) 143*4d495c6eSApple OSS Distributions elif code in other_return_values: 144*4d495c6eSApple OSS Distributions output.append(other_return_values[code]) 145*4d495c6eSApple OSS Distributions elif code != 0: 146*4d495c6eSApple OSS Distributions output.append(str(code) + ': ????') 147*4d495c6eSApple OSS Distributions 148*4d495c6eSApple OSS Distributions print(format_indent + '(' + ', '.join(output) + ')') 149*4d495c6eSApple OSS Distributions 150*4d495c6eSApple OSS Distributions# display names for error codes returned in errno 151*4d495c6eSApple OSS Distributionserrno_return_values = { 152*4d495c6eSApple OSS Distributions 1: 'EPERM', 153*4d495c6eSApple OSS Distributions 9: 'EBADF', 154*4d495c6eSApple OSS Distributions 12: 'ENOMEM', 155*4d495c6eSApple OSS Distributions 13: 'EACCES', 156*4d495c6eSApple OSS Distributions 14: 'EFAULT', 157*4d495c6eSApple OSS Distributions 22: 'EINVAL', 158*4d495c6eSApple OSS Distributions 45: 'ENOTSUP', 159*4d495c6eSApple OSS Distributions} 160*4d495c6eSApple OSS Distributionsfor k, v in errno_return_values.items(): 161*4d495c6eSApple OSS Distributions errno_return_values[k] = str(k) + ': ' + v 162*4d495c6eSApple OSS Distributions 163*4d495c6eSApple OSS Distributions# display names for error codes returned in kern_return_t 164*4d495c6eSApple OSS Distributionskern_return_values = { 165*4d495c6eSApple OSS Distributions 1: 'KERN_INVALID_ADDRESS', 166*4d495c6eSApple OSS Distributions 2: 'KERN_PROTECTION_FAILURE', 167*4d495c6eSApple OSS Distributions 3: 'KERN_NO_SPACE', 168*4d495c6eSApple OSS Distributions 4: 'KERN_INVALID_ARGUMENT', 169*4d495c6eSApple OSS Distributions 5: 'KERN_FAILURE', 170*4d495c6eSApple OSS Distributions 6: 'KERN_RESOURCE_SHORTAGE', 171*4d495c6eSApple OSS Distributions 7: 'KERN_NOT_RECEIVER', 172*4d495c6eSApple OSS Distributions 8: 'KERN_NO_ACCESS', 173*4d495c6eSApple OSS Distributions 9: 'KERN_MEMORY_FAILURE', 174*4d495c6eSApple OSS Distributions 10: 'KERN_MEMORY_ERROR', 175*4d495c6eSApple OSS Distributions 11: 'KERN_ALREADY_IN_SET', 176*4d495c6eSApple OSS Distributions 12: 'KERN_NOT_IN_SET', 177*4d495c6eSApple OSS Distributions 13: 'KERN_NAME_EXISTS', 178*4d495c6eSApple OSS Distributions 14: 'KERN_ABORTED', 179*4d495c6eSApple OSS Distributions 15: 'KERN_INVALID_NAME', 180*4d495c6eSApple OSS Distributions 16: 'KERN_INVALID_TASK', 181*4d495c6eSApple OSS Distributions 17: 'KERN_INVALID_RIGHT', 182*4d495c6eSApple OSS Distributions 18: 'KERN_INVALID_VALUE', 183*4d495c6eSApple OSS Distributions 19: 'KERN_UREFS_OVERFLOW', 184*4d495c6eSApple OSS Distributions 20: 'KERN_INVALID_CAPABILITY', 185*4d495c6eSApple OSS Distributions 21: 'KERN_RIGHT_EXISTS', 186*4d495c6eSApple OSS Distributions 22: 'KERN_INVALID_HOST', 187*4d495c6eSApple OSS Distributions 23: 'KERN_MEMORY_PRESENT', 188*4d495c6eSApple OSS Distributions 24: 'KERN_MEMORY_DATA_MOVED', 189*4d495c6eSApple OSS Distributions 25: 'KERN_MEMORY_RESTART_COPY', 190*4d495c6eSApple OSS Distributions 26: 'KERN_INVALID_PROCESSOR_SET', 191*4d495c6eSApple OSS Distributions 27: 'KERN_POLICY_LIMIT', 192*4d495c6eSApple OSS Distributions 28: 'KERN_INVALID_POLICY', 193*4d495c6eSApple OSS Distributions 29: 'KERN_INVALID_OBJECT', 194*4d495c6eSApple OSS Distributions 30: 'KERN_ALREADY_WAITING', 195*4d495c6eSApple OSS Distributions 31: 'KERN_DEFAULT_SET', 196*4d495c6eSApple OSS Distributions 32: 'KERN_EXCEPTION_PROTECTED', 197*4d495c6eSApple OSS Distributions 33: 'KERN_INVALID_LEDGER', 198*4d495c6eSApple OSS Distributions 34: 'KERN_INVALID_MEMORY_CONTROL', 199*4d495c6eSApple OSS Distributions 35: 'KERN_INVALID_SECURITY', 200*4d495c6eSApple OSS Distributions 36: 'KERN_NOT_DEPRESSED', 201*4d495c6eSApple OSS Distributions 37: 'KERN_TERMINATED', 202*4d495c6eSApple OSS Distributions 38: 'KERN_LOCK_SET_DESTROYED', 203*4d495c6eSApple OSS Distributions 39: 'KERN_LOCK_UNSTABLE', 204*4d495c6eSApple OSS Distributions 40: 'KERN_LOCK_OWNED', 205*4d495c6eSApple OSS Distributions 41: 'KERN_LOCK_OWNED_SELF', 206*4d495c6eSApple OSS Distributions 42: 'KERN_SEMAPHORE_DESTROYED', 207*4d495c6eSApple OSS Distributions 43: 'KERN_RPC_SERVER_TERMINATED', 208*4d495c6eSApple OSS Distributions 44: 'KERN_RPC_TERMINATE_ORPHAN', 209*4d495c6eSApple OSS Distributions 45: 'KERN_RPC_CONTINUE_ORPHAN', 210*4d495c6eSApple OSS Distributions 46: 'KERN_NOT_SUPPORTED', 211*4d495c6eSApple OSS Distributions 47: 'KERN_NODE_DOWN', 212*4d495c6eSApple OSS Distributions 48: 'KERN_NOT_WAITING', 213*4d495c6eSApple OSS Distributions 49: 'KERN_OPERATION_TIMED_OUT', 214*4d495c6eSApple OSS Distributions 50: 'KERN_CODESIGN_ERROR', 215*4d495c6eSApple OSS Distributions 51: 'KERN_POLICY_STATIC', 216*4d495c6eSApple OSS Distributions 52: 'KERN_INSUFFICIENT_BUFFER_SIZE', 217*4d495c6eSApple OSS Distributions 53: 'KERN_DENIED', 218*4d495c6eSApple OSS Distributions 54: 'KERN_MISSING_KC', 219*4d495c6eSApple OSS Distributions 55: 'KERN_INVALID_KC', 220*4d495c6eSApple OSS Distributions 56: 'KERN_NOT_FOUND', 221*4d495c6eSApple OSS Distributions 100: 'KERN_RETURN_MAX', 222*4d495c6eSApple OSS Distributions -304: 'MIG_BAD_ARGUMENTS (server type check failure)', 223*4d495c6eSApple OSS Distributions # MACH_SEND_INVALID_MEMORY and other Mach errors with large integer values 224*4d495c6eSApple OSS Distributions # are not handled here. They use format_result and other_return_values instead. 225*4d495c6eSApple OSS Distributions} 226*4d495c6eSApple OSS Distributionsfor k, v in kern_return_values.items(): 227*4d495c6eSApple OSS Distributions kern_return_values[k] = str(k) + ': ' + v 228*4d495c6eSApple OSS Distributions 229*4d495c6eSApple OSS Distributions# display names for error codes return by a boolean function 230*4d495c6eSApple OSS Distributions# where 0=failure and 1=success 231*4d495c6eSApple OSS Distributionsbool_return_values = { 232*4d495c6eSApple OSS Distributions 0: format_bool_result[0].lstrip() + ': false/failure', 233*4d495c6eSApple OSS Distributions 1: format_bool_result[1].lstrip() + ': true/success', 234*4d495c6eSApple OSS Distributions} 235*4d495c6eSApple OSS Distributions 236*4d495c6eSApple OSS Distributions# display names for the special return values used by the test machinery 237*4d495c6eSApple OSS Distributionsother_return_values = { 238*4d495c6eSApple OSS Distributions RESULT_BUSTED: format_result[RESULT_BUSTED].lstrip() + ': trial broken, not performed', 239*4d495c6eSApple OSS Distributions RESULT_IGNORED: '<empty> trial ignored, not performed', 240*4d495c6eSApple OSS Distributions RESULT_ZEROSIZE: format_result[RESULT_ZEROSIZE].lstrip() + ': size == 0', 241*4d495c6eSApple OSS Distributions RESULT_PANIC: format_result[RESULT_PANIC].lstrip() + ': trial is believed to panic, not performed', 242*4d495c6eSApple OSS Distributions RESULT_GUARD: format_result[RESULT_GUARD].lstrip() + ': trial is believed to throw EXC_GUARD, not performed', 243*4d495c6eSApple OSS Distributions RESULT_OUT_PARAM_BAD: format_result[RESULT_OUT_PARAM_BAD].lstrip() + ': trial set incorrect values to out parameters', 244*4d495c6eSApple OSS Distributions RESULT_MACH_SEND_INVALID_MEMORY: format_result[RESULT_MACH_SEND_INVALID_MEMORY].lstrip() + ': MACH_SEND_INVALID_MEMORY', 245*4d495c6eSApple OSS Distributions RESULT_MACH_SEND_INVALID_DEST: format_result[RESULT_MACH_SEND_INVALID_DEST].lstrip() + ': MACH_SEND_INVALID_DEST', 246*4d495c6eSApple OSS Distributions} 247*4d495c6eSApple OSS Distributions 248*4d495c6eSApple OSS Distributions# inside line, replace 'return 123' with 'return ERR_CODE_NAME' 249*4d495c6eSApple OSS Distributionsdef replace_error_code_return(test, line): 250*4d495c6eSApple OSS Distributions known_return_values = error_code_values_for_testname(test.testname) 251*4d495c6eSApple OSS Distributions for code, name in known_return_values.items(): 252*4d495c6eSApple OSS Distributions line = line.replace('return ' + str(code) + ';', 'return ' + name + ';') 253*4d495c6eSApple OSS Distributions return line 254*4d495c6eSApple OSS Distributions 255*4d495c6eSApple OSS Distributionsdef dimensions(results): 256*4d495c6eSApple OSS Distributions if len(results) == 0: 257*4d495c6eSApple OSS Distributions return 0 258*4d495c6eSApple OSS Distributions return len(results[0].parameters) 259*4d495c6eSApple OSS Distributions 260*4d495c6eSApple OSS Distributions# given one k-dimensional results 261*4d495c6eSApple OSS Distributions# return a list of k counts that is the size of each dimension 262*4d495c6eSApple OSS Distributionsdef count_each_dimension(results): 263*4d495c6eSApple OSS Distributions if len(results) == 0: 264*4d495c6eSApple OSS Distributions return [] 265*4d495c6eSApple OSS Distributions first = results[0].parameters 266*4d495c6eSApple OSS Distributions k = dimensions(results) 267*4d495c6eSApple OSS Distributions counts = [] 268*4d495c6eSApple OSS Distributions step = 1 269*4d495c6eSApple OSS Distributions for dim in range(k-1, -1, -1): 270*4d495c6eSApple OSS Distributions count = round(len(results) / step) 271*4d495c6eSApple OSS Distributions for i in range(0, len(results), step): 272*4d495c6eSApple OSS Distributions cur = results[i].parameters 273*4d495c6eSApple OSS Distributions if i != 0 and cur[dim] == first[dim]: 274*4d495c6eSApple OSS Distributions count = round(i / step) 275*4d495c6eSApple OSS Distributions break; 276*4d495c6eSApple OSS Distributions step *= count 277*4d495c6eSApple OSS Distributions counts.append(count) 278*4d495c6eSApple OSS Distributions 279*4d495c6eSApple OSS Distributions counts.reverse() 280*4d495c6eSApple OSS Distributions return counts; 281*4d495c6eSApple OSS Distributions 282*4d495c6eSApple OSS Distributions# Reduce one k-dimensional results to many (k-1) dimensional results 283*4d495c6eSApple OSS Distributions# Yields a sequence of [results, name] pairs 284*4d495c6eSApple OSS Distributions# where results has k-1 dimensions 285*4d495c6eSApple OSS Distributions# and name is the parameter name from the removed dimension 286*4d495c6eSApple OSS Distributionsdef iterate_dimension(results, dim = 0): 287*4d495c6eSApple OSS Distributions if len(results) == 0: 288*4d495c6eSApple OSS Distributions return 289*4d495c6eSApple OSS Distributions 290*4d495c6eSApple OSS Distributions k = dimensions(results) 291*4d495c6eSApple OSS Distributions dim_counts = count_each_dimension(results) 292*4d495c6eSApple OSS Distributions 293*4d495c6eSApple OSS Distributions inner_count = 1 294*4d495c6eSApple OSS Distributions for d in range(dim+1, k): 295*4d495c6eSApple OSS Distributions inner_count *= dim_counts[d] 296*4d495c6eSApple OSS Distributions 297*4d495c6eSApple OSS Distributions outer_step = len(results) 298*4d495c6eSApple OSS Distributions for d in range(0, dim): 299*4d495c6eSApple OSS Distributions outer_step = int(outer_step / dim_counts[d]) 300*4d495c6eSApple OSS Distributions 301*4d495c6eSApple OSS Distributions for r in range(dim_counts[dim]): 302*4d495c6eSApple OSS Distributions start = r * inner_count 303*4d495c6eSApple OSS Distributions name = results[start].parameters[dim] 304*4d495c6eSApple OSS Distributions new_results = [] 305*4d495c6eSApple OSS Distributions for i in range(start, len(results), outer_step): 306*4d495c6eSApple OSS Distributions for j in range(inner_count): 307*4d495c6eSApple OSS Distributions new_result = copy.deepcopy(results[i+j]) 308*4d495c6eSApple OSS Distributions del new_result.parameters[dim] 309*4d495c6eSApple OSS Distributions new_results.append(new_result) 310*4d495c6eSApple OSS Distributions yield [new_results, name] 311*4d495c6eSApple OSS Distributions 312*4d495c6eSApple OSS Distributions# Print the results of a test that has two parameters (for example a test of start/size) 313*4d495c6eSApple OSS Distributions# If overrides!=None, use any non-SUCCESS return values from override in place of the other results. 314*4d495c6eSApple OSS Distributionsdef print_results_2D(results, formatter, overrides=None): 315*4d495c6eSApple OSS Distributions # complain if results and override have different dimensions 316*4d495c6eSApple OSS Distributions if overrides: 317*4d495c6eSApple OSS Distributions if len(overrides) != len(results): 318*4d495c6eSApple OSS Distributions print("WARNING: override results have a different height; overrides ignored") 319*4d495c6eSApple OSS Distributions for i, result in enumerate(results): 320*4d495c6eSApple OSS Distributions if len(overrides[i].parameters) != len(result.parameters): 321*4d495c6eSApple OSS Distributions print("WARNING: override results have a different width; overrides ignored") 322*4d495c6eSApple OSS Distributions 323*4d495c6eSApple OSS Distributions columns = [] 324*4d495c6eSApple OSS Distributions prev_row_label = '' 325*4d495c6eSApple OSS Distributions first_row_label = '' 326*4d495c6eSApple OSS Distributions for i, result in enumerate(results): 327*4d495c6eSApple OSS Distributions if overrides: override = overrides[i].ret 328*4d495c6eSApple OSS Distributions 329*4d495c6eSApple OSS Distributions if first_row_label == '': 330*4d495c6eSApple OSS Distributions # record first row's name so we can use it to find columns 331*4d495c6eSApple OSS Distributions # (assumes every row has the same column labels) 332*4d495c6eSApple OSS Distributions first_row_label = result.parameters[0] 333*4d495c6eSApple OSS Distributions 334*4d495c6eSApple OSS Distributions if result.parameters[0] == first_row_label: 335*4d495c6eSApple OSS Distributions # record column names in the first row 336*4d495c6eSApple OSS Distributions columns.append(result.parameters[1]) 337*4d495c6eSApple OSS Distributions 338*4d495c6eSApple OSS Distributions if result.parameters[0] != prev_row_label: 339*4d495c6eSApple OSS Distributions # new row 340*4d495c6eSApple OSS Distributions if prev_row_label != '': print(format_indent + prev_row_label) 341*4d495c6eSApple OSS Distributions print(format_indent, end='') 342*4d495c6eSApple OSS Distributions prev_row_label = result.parameters[0] 343*4d495c6eSApple OSS Distributions 344*4d495c6eSApple OSS Distributions if overrides and override != RESULT_SUCCESS: 345*4d495c6eSApple OSS Distributions print_one_result(override, formatter) 346*4d495c6eSApple OSS Distributions else: 347*4d495c6eSApple OSS Distributions print_one_result(result.ret, formatter) 348*4d495c6eSApple OSS Distributions 349*4d495c6eSApple OSS Distributions if prev_row_label: print(format_indent + prev_row_label) 350*4d495c6eSApple OSS Distributions print_column_labels(columns, format_indent_width + format_col_width - 1, format_col_width) 351*4d495c6eSApple OSS Distributions 352*4d495c6eSApple OSS Distributionsdef print_results_2D_try_condensed(results, formatter): 353*4d495c6eSApple OSS Distributions if 0 == len(results): 354*4d495c6eSApple OSS Distributions return 355*4d495c6eSApple OSS Distributions singleton = results[0].ret 356*4d495c6eSApple OSS Distributions if any([result.ret != singleton for result in results]): 357*4d495c6eSApple OSS Distributions print_results_2D(results, formatter) 358*4d495c6eSApple OSS Distributions return 359*4d495c6eSApple OSS Distributions # will print as condensed 360*4d495c6eSApple OSS Distributions rows = set() 361*4d495c6eSApple OSS Distributions cols = set() 362*4d495c6eSApple OSS Distributions for result in results: 363*4d495c6eSApple OSS Distributions rows.add(result.parameters[0].split()[1]) 364*4d495c6eSApple OSS Distributions cols.add(result.parameters[1].split()[1]) 365*4d495c6eSApple OSS Distributions print_one_result(result.ret, formatter) 366*4d495c6eSApple OSS Distributions print(" for all pairs") 367*4d495c6eSApple OSS Distributions 368*4d495c6eSApple OSS Distributionsdef print_results_3D(results, formatter, testname): 369*4d495c6eSApple OSS Distributions # foreach parameter[1], print 2D table of parameter[0] and parameter[2] 370*4d495c6eSApple OSS Distributions for results2D, name in iterate_dimension(results, 1): 371*4d495c6eSApple OSS Distributions print(testname + ': ' + name) 372*4d495c6eSApple OSS Distributions print_results_2D(results2D, formatter) 373*4d495c6eSApple OSS Distributions 374*4d495c6eSApple OSS Distributions # foreach parameter[0], print 2D table of parameter[1] and parameter[2] 375*4d495c6eSApple OSS Distributions # This is redundant but can be useful for human readers. 376*4d495c6eSApple OSS Distributions for results2D, name in iterate_dimension(results, 0): 377*4d495c6eSApple OSS Distributions print(testname + ': ' + name) 378*4d495c6eSApple OSS Distributions print_results_2D(results2D, formatter) 379*4d495c6eSApple OSS Distributions 380*4d495c6eSApple OSS Distributionsdef print_results_4D(results, formatter): 381*4d495c6eSApple OSS Distributions x, y, z = '', '', '' 382*4d495c6eSApple OSS Distributions # Make a map[{3rd_param, 4th_param, ...}] = {all options} 383*4d495c6eSApple OSS Distributions # For now, we print 2d tables of 1st, 2nd param for each possible combination of remaining values 384*4d495c6eSApple OSS Distributions 385*4d495c6eSApple OSS Distributions map_of_results = {} 386*4d495c6eSApple OSS Distributions for _, result in enumerate(results): 387*4d495c6eSApple OSS Distributions k = tuple(result.parameters[2:]) 388*4d495c6eSApple OSS Distributions 389*4d495c6eSApple OSS Distributions if k not in map_of_results: 390*4d495c6eSApple OSS Distributions map_of_results[k] = [result] 391*4d495c6eSApple OSS Distributions else: 392*4d495c6eSApple OSS Distributions map_of_results[k].append(result) 393*4d495c6eSApple OSS Distributions 394*4d495c6eSApple OSS Distributions # prepare to iterate 395*4d495c6eSApple OSS Distributions prev_matrix = [] 396*4d495c6eSApple OSS Distributions iterable = [] 397*4d495c6eSApple OSS Distributions for k, result_list in map_of_results.items(): 398*4d495c6eSApple OSS Distributions one_2d_result = [] 399*4d495c6eSApple OSS Distributions matrix = [] 400*4d495c6eSApple OSS Distributions for result in result_list: 401*4d495c6eSApple OSS Distributions x = result.parameters[0] 402*4d495c6eSApple OSS Distributions y = result.parameters[1] 403*4d495c6eSApple OSS Distributions repl_result = Result(result.ret, (x, y)) 404*4d495c6eSApple OSS Distributions one_2d_result.append(repl_result) 405*4d495c6eSApple OSS Distributions matrix.append(result.ret) 406*4d495c6eSApple OSS Distributions if matrix == prev_matrix: 407*4d495c6eSApple OSS Distributions # if the return codes are the same everywhere, we will print successive tables only once 408*4d495c6eSApple OSS Distributions # note that this assumes that the sets of 2D labels are the same everywhere, and doesn't check that assumption 409*4d495c6eSApple OSS Distributions iterable[-1][0].append(k) 410*4d495c6eSApple OSS Distributions else: 411*4d495c6eSApple OSS Distributions iterable.append(([k], one_2d_result)) 412*4d495c6eSApple OSS Distributions prev_matrix = matrix 413*4d495c6eSApple OSS Distributions 414*4d495c6eSApple OSS Distributions # print 415*4d495c6eSApple OSS Distributions for iter in iterable: 416*4d495c6eSApple OSS Distributions print(iter[0]) 417*4d495c6eSApple OSS Distributions print_results_2D_try_condensed(iter[1], formatter) 418*4d495c6eSApple OSS Distributions 419*4d495c6eSApple OSS Distributions 420*4d495c6eSApple OSS Distributions# Print the results of a test that has two parameters 421*4d495c6eSApple OSS Distributions# (for example a test of addr only, or size only) 422*4d495c6eSApple OSS Distributions# If overrides!=None, use any non-SUCCESS return values from override in place of the other results. 423*4d495c6eSApple OSS Distributionsdef print_results_1D(results, formatter, overrides=None): 424*4d495c6eSApple OSS Distributions # complain if results and overrides have different dimensions 425*4d495c6eSApple OSS Distributions if overrides: 426*4d495c6eSApple OSS Distributions if len(overrides) != len(results): 427*4d495c6eSApple OSS Distributions print("WARNING: override results have a different height; overrides ignored") 428*4d495c6eSApple OSS Distributions for i, result in enumerate(results): 429*4d495c6eSApple OSS Distributions if len(overrides[i].parameters) != len(result.parameters): 430*4d495c6eSApple OSS Distributions print("WARNING: override results have a different width; overrides ignored") 431*4d495c6eSApple OSS Distributions 432*4d495c6eSApple OSS Distributions for i, result in enumerate(results): 433*4d495c6eSApple OSS Distributions if overrides: override = overrides[i].ret 434*4d495c6eSApple OSS Distributions 435*4d495c6eSApple OSS Distributions # indent, value, indent, label 436*4d495c6eSApple OSS Distributions print(format_indent, end='') 437*4d495c6eSApple OSS Distributions if overrides and override != RESULT_SUCCESS: 438*4d495c6eSApple OSS Distributions print_one_result(override, formatter) 439*4d495c6eSApple OSS Distributions else: 440*4d495c6eSApple OSS Distributions print_one_result(result.ret, formatter) 441*4d495c6eSApple OSS Distributions print(format_indent + result.parameters[0]) 442*4d495c6eSApple OSS Distributions 443*4d495c6eSApple OSS Distributionsdef print_results_nD(results, testname, overrides=None): 444*4d495c6eSApple OSS Distributions formatter = formatter_for_testname(testname) 445*4d495c6eSApple OSS Distributions 446*4d495c6eSApple OSS Distributions if (dimensions(results) == 1): 447*4d495c6eSApple OSS Distributions print_results_1D(results, formatter, overrides) 448*4d495c6eSApple OSS Distributions elif (dimensions(results) == 2): 449*4d495c6eSApple OSS Distributions print_results_2D(results, formatter, overrides) 450*4d495c6eSApple OSS Distributions elif dimensions(results) == 3: 451*4d495c6eSApple OSS Distributions print_results_3D(results, formatter, testname) 452*4d495c6eSApple OSS Distributions elif dimensions(results) == 4: 453*4d495c6eSApple OSS Distributions print_results_4D(results, formatter) 454*4d495c6eSApple OSS Distributions else: 455*4d495c6eSApple OSS Distributions print(format_indent + 'too many dimensions') 456*4d495c6eSApple OSS Distributions 457*4d495c6eSApple OSS Distributions 458*4d495c6eSApple OSS Distributionsdef main(): 459*4d495c6eSApple OSS Distributions data = sys.stdin.readlines() 460*4d495c6eSApple OSS Distributions 461*4d495c6eSApple OSS Distributions 462*4d495c6eSApple OSS Distributions # remove any lines that don't start with "TESTNAME" or "TESTCONFIG" or "RESULT" 463*4d495c6eSApple OSS Distributions # (including darwintest output like "PASS" or "FAIL") 464*4d495c6eSApple OSS Distributions # and print them now 465*4d495c6eSApple OSS Distributions # Also verify that the counts of "TEST BEGIN" == "TEST END" 466*4d495c6eSApple OSS Distributions # (they will mismatch if a test suite crashed) 467*4d495c6eSApple OSS Distributions testbegincount = 0 468*4d495c6eSApple OSS Distributions testendcount = 0 469*4d495c6eSApple OSS Distributions testlines = [] 470*4d495c6eSApple OSS Distributions for line in data: 471*4d495c6eSApple OSS Distributions unmodified_line = line 472*4d495c6eSApple OSS Distributions # count TEST BEGIN and TEST END 473*4d495c6eSApple OSS Distributions if ('TEST BEGIN' in line): 474*4d495c6eSApple OSS Distributions testbegincount += 1 475*4d495c6eSApple OSS Distributions if ('TEST END' in line): 476*4d495c6eSApple OSS Distributions testendcount += 1 477*4d495c6eSApple OSS Distributions # remove any T_LOG() timestamp prefixes and KTEST prefixes 478*4d495c6eSApple OSS Distributions line = re.sub('^\s*\d+:\d+:\d+ ', '', line) 479*4d495c6eSApple OSS Distributions line = re.sub('^\[KTEST\]\s+[A-Z]+\s+\d+\s+(\d+\s+)?\S+\s+\d+\s+', '', line) 480*4d495c6eSApple OSS Distributions line = line.lstrip() 481*4d495c6eSApple OSS Distributions 482*4d495c6eSApple OSS Distributions if (line.startswith('TESTNAME') or line.startswith('RESULT') 483*4d495c6eSApple OSS Distributions or line.startswith('TESTCONFIG') or line.startswith('TESTCOMPAT')): 484*4d495c6eSApple OSS Distributions testlines.append(line) # line is test output 485*4d495c6eSApple OSS Distributions elif line == '': 486*4d495c6eSApple OSS Distributions pass # ignore empty lines 487*4d495c6eSApple OSS Distributions else: 488*4d495c6eSApple OSS Distributions print(unmodified_line, end='') # line is other output 489*4d495c6eSApple OSS Distributions 490*4d495c6eSApple OSS Distributions # parse test output into Test and Result objects 491*4d495c6eSApple OSS Distributions 492*4d495c6eSApple OSS Distributions testnum = 0 493*4d495c6eSApple OSS Distributions def group_by_test(line): 494*4d495c6eSApple OSS Distributions nonlocal testnum 495*4d495c6eSApple OSS Distributions if line.startswith('TESTNAME '): 496*4d495c6eSApple OSS Distributions testnum = testnum+1 497*4d495c6eSApple OSS Distributions return testnum 498*4d495c6eSApple OSS Distributions 499*4d495c6eSApple OSS Distributions tests = [] 500*4d495c6eSApple OSS Distributions for _, group in itertools.groupby(testlines, group_by_test): 501*4d495c6eSApple OSS Distributions lines = list(group) 502*4d495c6eSApple OSS Distributions 503*4d495c6eSApple OSS Distributions name = lines.pop(0).removeprefix('TESTNAME ').rstrip() 504*4d495c6eSApple OSS Distributions config = lines.pop(0).removeprefix('TESTCONFIG ').rstrip() 505*4d495c6eSApple OSS Distributions compat = [] 506*4d495c6eSApple OSS Distributions results = [] 507*4d495c6eSApple OSS Distributions for line in lines: 508*4d495c6eSApple OSS Distributions if line.startswith('RESULT'): 509*4d495c6eSApple OSS Distributions components = line.removeprefix('RESULT ').rstrip().split(', ') 510*4d495c6eSApple OSS Distributions ret = int(components.pop(0)) 511*4d495c6eSApple OSS Distributions results.append(Result(ret, components)) 512*4d495c6eSApple OSS Distributions 513*4d495c6eSApple OSS Distributions tests.append(Test(name, config, compat, results)) 514*4d495c6eSApple OSS Distributions 515*4d495c6eSApple OSS Distributions print('found %d tests' % (len(tests))) 516*4d495c6eSApple OSS Distributions 517*4d495c6eSApple OSS Distributions # stats to print at the end 518*4d495c6eSApple OSS Distributions test_count = len(tests) 519*4d495c6eSApple OSS Distributions all_configurations = set() 520*4d495c6eSApple OSS Distributions 521*4d495c6eSApple OSS Distributions # print test output 522*4d495c6eSApple OSS Distributions for test in tests: 523*4d495c6eSApple OSS Distributions # print test name and test config on separate lines 524*4d495c6eSApple OSS Distributions # `diff` handles this better than putting both on the same line 525*4d495c6eSApple OSS Distributions print('test ' + test.testname) 526*4d495c6eSApple OSS Distributions 527*4d495c6eSApple OSS Distributions print(format_indent + 'config ' + test.config) 528*4d495c6eSApple OSS Distributions all_configurations.add(test.config) 529*4d495c6eSApple OSS Distributions 530*4d495c6eSApple OSS Distributions if len(test.results) == 0: 531*4d495c6eSApple OSS Distributions print(format_indent + 'no results') 532*4d495c6eSApple OSS Distributions else: 533*4d495c6eSApple OSS Distributions print_legend(test) 534*4d495c6eSApple OSS Distributions print_results_nD(test.results, test.testname) 535*4d495c6eSApple OSS Distributions 536*4d495c6eSApple OSS Distributions 537*4d495c6eSApple OSS Distributions print('end ' + test.testname) 538*4d495c6eSApple OSS Distributions 539*4d495c6eSApple OSS Distributions print() 540*4d495c6eSApple OSS Distributions print(str(test_count) + ' test(s) performed') 541*4d495c6eSApple OSS Distributions 542*4d495c6eSApple OSS Distributions if (testbegincount != testendcount): 543*4d495c6eSApple OSS Distributions print('### error: %d TEST BEGINs, %d TEST ENDs - some tests may have crashed' 544*4d495c6eSApple OSS Distributions % (testbegincount, testendcount)) 545*4d495c6eSApple OSS Distributions 546*4d495c6eSApple OSS Distributions print(str(len(all_configurations)) + ' configuration(s) tested:') 547*4d495c6eSApple OSS Distributions for config in sorted(all_configurations): 548*4d495c6eSApple OSS Distributions print(format_indent + '[' + config + ']') 549*4d495c6eSApple OSS Distributions 550*4d495c6eSApple OSS Distributions 551*4d495c6eSApple OSS Distributionsmain() 552