14ece0189SSeongJae Park#!/usr/bin/env python3 24ece0189SSeongJae Park# SPDX-License-Identifier: GPL-2.0 34ece0189SSeongJae Park 44ece0189SSeongJae Parkimport json 54ece0189SSeongJae Parkimport os 64ece0189SSeongJae Parkimport subprocess 74ece0189SSeongJae Park 84ece0189SSeongJae Parkimport _damon_sysfs 94ece0189SSeongJae Park 104ece0189SSeongJae Parkdef dump_damon_status_dict(pid): 11cf20cb9aSSeongJae Park try: 12cf20cb9aSSeongJae Park subprocess.check_output(['which', 'drgn'], stderr=subprocess.DEVNULL) 13cf20cb9aSSeongJae Park except: 14cf20cb9aSSeongJae Park return None, 'drgn not found' 154ece0189SSeongJae Park file_dir = os.path.dirname(os.path.abspath(__file__)) 164ece0189SSeongJae Park dump_script = os.path.join(file_dir, 'drgn_dump_damon_status.py') 174ece0189SSeongJae Park rc = subprocess.call(['drgn', dump_script, pid, 'damon_dump_output'], 184ece0189SSeongJae Park stderr=subprocess.DEVNULL) 194ece0189SSeongJae Park if rc != 0: 204ece0189SSeongJae Park return None, 'drgn fail' 214ece0189SSeongJae Park try: 224ece0189SSeongJae Park with open('damon_dump_output', 'r') as f: 234ece0189SSeongJae Park return json.load(f), None 244ece0189SSeongJae Park except Exception as e: 254ece0189SSeongJae Park return None, 'json.load fail (%s)' % e 264ece0189SSeongJae Park 27ae3ab07eSSeongJae Parkdef fail(expectation, status): 28ae3ab07eSSeongJae Park print('unexpected %s' % expectation) 29ae3ab07eSSeongJae Park print(json.dumps(status, indent=4)) 30ae3ab07eSSeongJae Park exit(1) 31ae3ab07eSSeongJae Park 32b50c48deSSeongJae Parkdef assert_true(condition, expectation, status): 33b50c48deSSeongJae Park if condition is not True: 34b50c48deSSeongJae Park fail(expectation, status) 35b50c48deSSeongJae Park 36b50c48deSSeongJae Parkdef assert_watermarks_committed(watermarks, dump): 37b50c48deSSeongJae Park wmark_metric_val = { 38b50c48deSSeongJae Park 'none': 0, 39b50c48deSSeongJae Park 'free_mem_rate': 1, 40b50c48deSSeongJae Park } 41b50c48deSSeongJae Park assert_true(dump['metric'] == wmark_metric_val[watermarks.metric], 42b50c48deSSeongJae Park 'metric', dump) 43b50c48deSSeongJae Park assert_true(dump['interval'] == watermarks.interval, 'interval', dump) 44b50c48deSSeongJae Park assert_true(dump['high'] == watermarks.high, 'high', dump) 45b50c48deSSeongJae Park assert_true(dump['mid'] == watermarks.mid, 'mid', dump) 46b50c48deSSeongJae Park assert_true(dump['low'] == watermarks.low, 'low', dump) 47b50c48deSSeongJae Park 48*f797e709SSeongJae Parkdef assert_quota_committed(quota, dump): 49*f797e709SSeongJae Park assert_true(dump['reset_interval'] == quota.reset_interval_ms, 50*f797e709SSeongJae Park 'reset_interval', dump) 51*f797e709SSeongJae Park assert_true(dump['ms'] == quota.ms, 'ms', dump) 52*f797e709SSeongJae Park assert_true(dump['sz'] == quota.sz, 'sz', dump) 53*f797e709SSeongJae Park # TODO: assert goals are committed 54*f797e709SSeongJae Park assert_true(dump['weight_sz'] == quota.weight_sz_permil, 'weight_sz', dump) 55*f797e709SSeongJae Park assert_true(dump['weight_nr_accesses'] == quota.weight_nr_accesses_permil, 56*f797e709SSeongJae Park 'weight_nr_accesses', dump) 57*f797e709SSeongJae Park assert_true( 58*f797e709SSeongJae Park dump['weight_age'] == quota.weight_age_permil, 'weight_age', dump) 59*f797e709SSeongJae Park 604ece0189SSeongJae Parkdef main(): 614ece0189SSeongJae Park kdamonds = _damon_sysfs.Kdamonds( 627e6bcf35SSeongJae Park [_damon_sysfs.Kdamond( 637e6bcf35SSeongJae Park contexts=[_damon_sysfs.DamonCtx( 64603cb4aaSSeongJae Park targets=[_damon_sysfs.DamonTarget(pid=-1)], 65603cb4aaSSeongJae Park schemes=[_damon_sysfs.Damos()], 66603cb4aaSSeongJae Park )])]) 674ece0189SSeongJae Park err = kdamonds.start() 684ece0189SSeongJae Park if err is not None: 694ece0189SSeongJae Park print('kdamond start failed: %s' % err) 704ece0189SSeongJae Park exit(1) 714ece0189SSeongJae Park 724ece0189SSeongJae Park status, err = dump_damon_status_dict(kdamonds.kdamonds[0].pid) 734ece0189SSeongJae Park if err is not None: 744ece0189SSeongJae Park print(err) 75cf20cb9aSSeongJae Park kdamonds.stop() 764ece0189SSeongJae Park exit(1) 774ece0189SSeongJae Park 784ece0189SSeongJae Park if len(status['contexts']) != 1: 79ae3ab07eSSeongJae Park fail('number of contexts', status) 80ae3ab07eSSeongJae Park 81ae3ab07eSSeongJae Park ctx = status['contexts'][0] 82ae3ab07eSSeongJae Park attrs = ctx['attrs'] 83ae3ab07eSSeongJae Park if attrs['sample_interval'] != 5000: 84ae3ab07eSSeongJae Park fail('sample interval', status) 85ae3ab07eSSeongJae Park if attrs['aggr_interval'] != 100000: 86ae3ab07eSSeongJae Park fail('aggr interval', status) 87ae3ab07eSSeongJae Park if attrs['ops_update_interval'] != 1000000: 88ae3ab07eSSeongJae Park fail('ops updte interval', status) 89ae3ab07eSSeongJae Park 90ae3ab07eSSeongJae Park if attrs['intervals_goal'] != { 91ae3ab07eSSeongJae Park 'access_bp': 0, 'aggrs': 0, 92ae3ab07eSSeongJae Park 'min_sample_us': 0, 'max_sample_us': 0}: 93ae3ab07eSSeongJae Park fail('intervals goal') 94ae3ab07eSSeongJae Park 95ae3ab07eSSeongJae Park if attrs['min_nr_regions'] != 10: 96ae3ab07eSSeongJae Park fail('min_nr_regions') 97ae3ab07eSSeongJae Park if attrs['max_nr_regions'] != 1000: 98ae3ab07eSSeongJae Park fail('max_nr_regions') 99ae3ab07eSSeongJae Park 1007e6bcf35SSeongJae Park if ctx['adaptive_targets'] != [ 1017e6bcf35SSeongJae Park { 'pid': 0, 'nr_regions': 0, 'regions_list': []}]: 1027e6bcf35SSeongJae Park fail('adaptive targets', status) 103ae3ab07eSSeongJae Park 104603cb4aaSSeongJae Park if len(ctx['schemes']) != 1: 105603cb4aaSSeongJae Park fail('number of schemes', status) 106603cb4aaSSeongJae Park 107603cb4aaSSeongJae Park scheme = ctx['schemes'][0] 108603cb4aaSSeongJae Park if scheme['pattern'] != { 109603cb4aaSSeongJae Park 'min_sz_region': 0, 110603cb4aaSSeongJae Park 'max_sz_region': 2**64 - 1, 111603cb4aaSSeongJae Park 'min_nr_accesses': 0, 112603cb4aaSSeongJae Park 'max_nr_accesses': 2**32 - 1, 113603cb4aaSSeongJae Park 'min_age_region': 0, 114603cb4aaSSeongJae Park 'max_age_region': 2**32 - 1, 115603cb4aaSSeongJae Park }: 116603cb4aaSSeongJae Park fail('damos pattern', status) 117603cb4aaSSeongJae Park if scheme['action'] != 9: # stat 118603cb4aaSSeongJae Park fail('damos action', status) 119603cb4aaSSeongJae Park if scheme['apply_interval_us'] != 0: 120603cb4aaSSeongJae Park fail('damos apply interval', status) 121603cb4aaSSeongJae Park if scheme['target_nid'] != -1: 122603cb4aaSSeongJae Park fail('damos target nid', status) 123603cb4aaSSeongJae Park 124*f797e709SSeongJae Park migrate_dests = scheme['migrate_dests'] 125*f797e709SSeongJae Park if migrate_dests['nr_dests'] != 0: 126*f797e709SSeongJae Park fail('nr_dests', status) 127*f797e709SSeongJae Park if migrate_dests['node_id_arr'] != []: 128*f797e709SSeongJae Park fail('node_id_arr', status) 129*f797e709SSeongJae Park if migrate_dests['weight_arr'] != []: 130*f797e709SSeongJae Park fail('weight_arr', status) 131603cb4aaSSeongJae Park 132*f797e709SSeongJae Park assert_quota_committed(_damon_sysfs.DamosQuota(), scheme['quota']) 133b50c48deSSeongJae Park assert_watermarks_committed(_damon_sysfs.DamosWatermarks(), 134b50c48deSSeongJae Park scheme['wmarks']) 135ae3ab07eSSeongJae Park 1364ece0189SSeongJae Park kdamonds.stop() 1374ece0189SSeongJae Park 1384ece0189SSeongJae Parkif __name__ == '__main__': 1394ece0189SSeongJae Park main() 140