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 4884dc442bSSeongJae Parkdef assert_quota_goal_committed(qgoal, dump): 4984dc442bSSeongJae Park metric_val = { 5084dc442bSSeongJae Park 'user_input': 0, 5184dc442bSSeongJae Park 'some_mem_psi_us': 1, 5284dc442bSSeongJae Park 'node_mem_used_bp': 2, 5384dc442bSSeongJae Park 'node_mem_free_bp': 3, 5484dc442bSSeongJae Park } 5584dc442bSSeongJae Park assert_true(dump['metric'] == metric_val[qgoal.metric], 'metric', dump) 5684dc442bSSeongJae Park assert_true(dump['target_value'] == qgoal.target_value, 'target_value', 5784dc442bSSeongJae Park dump) 5884dc442bSSeongJae Park if qgoal.metric == 'user_input': 5984dc442bSSeongJae Park assert_true(dump['current_value'] == qgoal.current_value, 6084dc442bSSeongJae Park 'current_value', dump) 6184dc442bSSeongJae Park assert_true(dump['nid'] == qgoal.nid, 'nid', dump) 6284dc442bSSeongJae Park 63f797e709SSeongJae Parkdef assert_quota_committed(quota, dump): 64f797e709SSeongJae Park assert_true(dump['reset_interval'] == quota.reset_interval_ms, 65f797e709SSeongJae Park 'reset_interval', dump) 66f797e709SSeongJae Park assert_true(dump['ms'] == quota.ms, 'ms', dump) 67f797e709SSeongJae Park assert_true(dump['sz'] == quota.sz, 'sz', dump) 6884dc442bSSeongJae Park for idx, qgoal in enumerate(quota.goals): 6984dc442bSSeongJae Park assert_quota_goal_committed(qgoal, dump['goals'][idx]) 70f797e709SSeongJae Park assert_true(dump['weight_sz'] == quota.weight_sz_permil, 'weight_sz', dump) 71f797e709SSeongJae Park assert_true(dump['weight_nr_accesses'] == quota.weight_nr_accesses_permil, 72f797e709SSeongJae Park 'weight_nr_accesses', dump) 73f797e709SSeongJae Park assert_true( 74f797e709SSeongJae Park dump['weight_age'] == quota.weight_age_permil, 'weight_age', dump) 75f797e709SSeongJae Park 76bd0487a7SSeongJae Park 77bd0487a7SSeongJae Parkdef assert_migrate_dests_committed(dests, dump): 78bd0487a7SSeongJae Park assert_true(dump['nr_dests'] == len(dests.dests), 'nr_dests', dump) 79bd0487a7SSeongJae Park for idx, dest in enumerate(dests.dests): 80bd0487a7SSeongJae Park assert_true(dump['node_id_arr'][idx] == dest.id, 'node_id', dump) 81bd0487a7SSeongJae Park assert_true(dump['weight_arr'][idx] == dest.weight, 'weight', dump) 82bd0487a7SSeongJae Park 8353f80058SSeongJae Parkdef assert_filter_committed(filter_, dump): 8453f80058SSeongJae Park assert_true(filter_.type_ == dump['type'], 'type', dump) 8553f80058SSeongJae Park assert_true(filter_.matching == dump['matching'], 'matching', dump) 8653f80058SSeongJae Park assert_true(filter_.allow == dump['allow'], 'allow', dump) 8753f80058SSeongJae Park # TODO: check memcg_path and memcg_id if type is memcg 8853f80058SSeongJae Park if filter_.type_ == 'addr': 8953f80058SSeongJae Park assert_true([filter_.addr_start, filter_.addr_end] == 9053f80058SSeongJae Park dump['addr_range'], 'addr_range', dump) 9153f80058SSeongJae Park elif filter_.type_ == 'target': 9253f80058SSeongJae Park assert_true(filter_.target_idx == dump['target_idx'], 'target_idx', 9353f80058SSeongJae Park dump) 9453f80058SSeongJae Park elif filter_.type_ == 'hugepage_size': 9553f80058SSeongJae Park assert_true([filter_.min_, filter_.max_] == dump['sz_range'], 9653f80058SSeongJae Park 'sz_range', dump) 9753f80058SSeongJae Park 98f22ff7b5SSeongJae Parkdef assert_access_pattern_committed(pattern, dump): 99f22ff7b5SSeongJae Park assert_true(dump['min_sz_region'] == pattern.size[0], 'min_sz_region', 100f22ff7b5SSeongJae Park dump) 101f22ff7b5SSeongJae Park assert_true(dump['max_sz_region'] == pattern.size[1], 'max_sz_region', 102f22ff7b5SSeongJae Park dump) 103f22ff7b5SSeongJae Park assert_true(dump['min_nr_accesses'] == pattern.nr_accesses[0], 104f22ff7b5SSeongJae Park 'min_nr_accesses', dump) 105f22ff7b5SSeongJae Park assert_true(dump['max_nr_accesses'] == pattern.nr_accesses[1], 106f22ff7b5SSeongJae Park 'max_nr_accesses', dump) 107f22ff7b5SSeongJae Park assert_true(dump['min_age_region'] == pattern.age[0], 'min_age_region', 108f22ff7b5SSeongJae Park dump) 109f22ff7b5SSeongJae Park assert_true(dump['max_age_region'] == pattern.age[1], 'miaxage_region', 110f22ff7b5SSeongJae Park dump) 111f22ff7b5SSeongJae Park 112f22ff7b5SSeongJae Parkdef assert_scheme_committed(scheme, dump): 113f22ff7b5SSeongJae Park assert_access_pattern_committed(scheme.access_pattern, dump['pattern']) 114f22ff7b5SSeongJae Park action_val = { 115f22ff7b5SSeongJae Park 'willneed': 0, 116f22ff7b5SSeongJae Park 'cold': 1, 117f22ff7b5SSeongJae Park 'pageout': 2, 118f22ff7b5SSeongJae Park 'hugepage': 3, 119f22ff7b5SSeongJae Park 'nohugeapge': 4, 120f22ff7b5SSeongJae Park 'lru_prio': 5, 121f22ff7b5SSeongJae Park 'lru_deprio': 6, 122f22ff7b5SSeongJae Park 'migrate_hot': 7, 123f22ff7b5SSeongJae Park 'migrate_cold': 8, 124f22ff7b5SSeongJae Park 'stat': 9, 125f22ff7b5SSeongJae Park } 126f22ff7b5SSeongJae Park assert_true(dump['action'] == action_val[scheme.action], 'action', dump) 127f22ff7b5SSeongJae Park assert_true(dump['apply_interval_us'] == scheme. apply_interval_us, 128f22ff7b5SSeongJae Park 'apply_interval_us', dump) 129f22ff7b5SSeongJae Park assert_true(dump['target_nid'] == scheme.target_nid, 'target_nid', dump) 130f22ff7b5SSeongJae Park assert_migrate_dests_committed(scheme.dests, dump['migrate_dests']) 131f22ff7b5SSeongJae Park assert_quota_committed(scheme.quota, dump['quota']) 132f22ff7b5SSeongJae Park assert_watermarks_committed(scheme.watermarks, dump['wmarks']) 13353f80058SSeongJae Park # TODO: test filters directory 13453f80058SSeongJae Park for idx, f in enumerate(scheme.core_filters.filters): 13553f80058SSeongJae Park assert_filter_committed(f, dump['filters'][idx]) 13653f80058SSeongJae Park for idx, f in enumerate(scheme.ops_filters.filters): 13753f80058SSeongJae Park assert_filter_committed(f, dump['ops_filters'][idx]) 138f22ff7b5SSeongJae Park 139771d7754SSeongJae Parkdef assert_schemes_committed(schemes, dump): 140771d7754SSeongJae Park assert_true(len(schemes) == len(dump), 'len_schemes', dump) 141771d7754SSeongJae Park for idx, scheme in enumerate(schemes): 142771d7754SSeongJae Park assert_scheme_committed(scheme, dump[idx]) 143771d7754SSeongJae Park 144a4027b5fSSeongJae Parkdef assert_monitoring_attrs_committed(attrs, dump): 145a4027b5fSSeongJae Park assert_true(dump['sample_interval'] == attrs.sample_us, 'sample_interval', 146a4027b5fSSeongJae Park dump) 147a4027b5fSSeongJae Park assert_true(dump['aggr_interval'] == attrs.aggr_us, 'aggr_interval', dump) 148a4027b5fSSeongJae Park assert_true(dump['intervals_goal']['access_bp'] == 149a4027b5fSSeongJae Park attrs.intervals_goal.access_bp, 'access_bp', 150a4027b5fSSeongJae Park dump['intervals_goal']) 151a4027b5fSSeongJae Park assert_true(dump['intervals_goal']['aggrs'] == attrs.intervals_goal.aggrs, 152a4027b5fSSeongJae Park 'aggrs', dump['intervals_goal']) 153a4027b5fSSeongJae Park assert_true(dump['intervals_goal']['min_sample_us'] == 154a4027b5fSSeongJae Park attrs.intervals_goal.min_sample_us, 'min_sample_us', 155a4027b5fSSeongJae Park dump['intervals_goal']) 156a4027b5fSSeongJae Park assert_true(dump['intervals_goal']['max_sample_us'] == 157a4027b5fSSeongJae Park attrs.intervals_goal.max_sample_us, 'max_sample_us', 158a4027b5fSSeongJae Park dump['intervals_goal']) 159a4027b5fSSeongJae Park 160a4027b5fSSeongJae Park assert_true(dump['ops_update_interval'] == attrs.update_us, 161a4027b5fSSeongJae Park 'ops_update_interval', dump) 162a4027b5fSSeongJae Park assert_true(dump['min_nr_regions'] == attrs.min_nr_regions, 163a4027b5fSSeongJae Park 'min_nr_regions', dump) 164a4027b5fSSeongJae Park assert_true(dump['max_nr_regions'] == attrs.max_nr_regions, 165a4027b5fSSeongJae Park 'max_nr_regions', dump) 166a4027b5fSSeongJae Park 16716797a55SSeongJae Parkdef assert_ctx_committed(ctx, dump): 16816797a55SSeongJae Park ops_val = { 16916797a55SSeongJae Park 'vaddr': 0, 17016797a55SSeongJae Park 'fvaddr': 1, 17116797a55SSeongJae Park 'paddr': 2, 17216797a55SSeongJae Park } 17316797a55SSeongJae Park assert_true(dump['ops']['id'] == ops_val[ctx.ops], 'ops_id', dump) 17416797a55SSeongJae Park assert_monitoring_attrs_committed(ctx.monitoring_attrs, dump['attrs']) 17516797a55SSeongJae Park assert_schemes_committed(ctx.schemes, dump['schemes']) 17616797a55SSeongJae Park 17716797a55SSeongJae Parkdef assert_ctxs_committed(ctxs, dump): 17816797a55SSeongJae Park assert_true(len(ctxs) == len(dump), 'ctxs length', dump) 17916797a55SSeongJae Park for idx, ctx in enumerate(ctxs): 18016797a55SSeongJae Park assert_ctx_committed(ctx, dump[idx]) 18116797a55SSeongJae Park 1824ece0189SSeongJae Parkdef main(): 1834ece0189SSeongJae Park kdamonds = _damon_sysfs.Kdamonds( 1847e6bcf35SSeongJae Park [_damon_sysfs.Kdamond( 1857e6bcf35SSeongJae Park contexts=[_damon_sysfs.DamonCtx( 186603cb4aaSSeongJae Park targets=[_damon_sysfs.DamonTarget(pid=-1)], 187603cb4aaSSeongJae Park schemes=[_damon_sysfs.Damos()], 188603cb4aaSSeongJae Park )])]) 1894ece0189SSeongJae Park err = kdamonds.start() 1904ece0189SSeongJae Park if err is not None: 1914ece0189SSeongJae Park print('kdamond start failed: %s' % err) 1924ece0189SSeongJae Park exit(1) 1934ece0189SSeongJae Park 1944ece0189SSeongJae Park status, err = dump_damon_status_dict(kdamonds.kdamonds[0].pid) 1954ece0189SSeongJae Park if err is not None: 1964ece0189SSeongJae Park print(err) 197cf20cb9aSSeongJae Park kdamonds.stop() 1984ece0189SSeongJae Park exit(1) 1994ece0189SSeongJae Park 20016797a55SSeongJae Park assert_ctxs_committed(kdamonds.kdamonds[0].contexts, status['contexts']) 201ae3ab07eSSeongJae Park 20262b7b1ffSSeongJae Park context = _damon_sysfs.DamonCtx( 20362b7b1ffSSeongJae Park monitoring_attrs=_damon_sysfs.DamonAttrs( 20462b7b1ffSSeongJae Park sample_us=100000, aggr_us=2000000, 20562b7b1ffSSeongJae Park intervals_goal=_damon_sysfs.IntervalsGoal( 20662b7b1ffSSeongJae Park access_bp=400, aggrs=3, min_sample_us=5000, 20762b7b1ffSSeongJae Park max_sample_us=10000000), 20862b7b1ffSSeongJae Park update_us=2000000), 20962b7b1ffSSeongJae Park schemes=[_damon_sysfs.Damos( 21062b7b1ffSSeongJae Park action='pageout', 21162b7b1ffSSeongJae Park access_pattern=_damon_sysfs.DamosAccessPattern( 21262b7b1ffSSeongJae Park size=[4096, 2**10], 21362b7b1ffSSeongJae Park nr_accesses=[3, 317], 21462b7b1ffSSeongJae Park age=[5,71]), 21562b7b1ffSSeongJae Park quota=_damon_sysfs.DamosQuota( 21662b7b1ffSSeongJae Park sz=100*1024*1024, ms=100, 21762b7b1ffSSeongJae Park goals=[_damon_sysfs.DamosQuotaGoal( 21862b7b1ffSSeongJae Park metric='node_mem_used_bp', 21962b7b1ffSSeongJae Park target_value=9950, 22062b7b1ffSSeongJae Park nid=1)], 22162b7b1ffSSeongJae Park reset_interval_ms=1500, 22262b7b1ffSSeongJae Park weight_sz_permil=20, 22362b7b1ffSSeongJae Park weight_nr_accesses_permil=200, 22462b7b1ffSSeongJae Park weight_age_permil=1000), 22562b7b1ffSSeongJae Park watermarks=_damon_sysfs.DamosWatermarks( 22662b7b1ffSSeongJae Park metric = 'free_mem_rate', interval = 500000, # 500 ms 22762b7b1ffSSeongJae Park high = 500, mid = 400, low = 50), 22862b7b1ffSSeongJae Park target_nid=1, 22962b7b1ffSSeongJae Park apply_interval_us=1000000, 23062b7b1ffSSeongJae Park dests=_damon_sysfs.DamosDests( 23162b7b1ffSSeongJae Park dests=[_damon_sysfs.DamosDest(id=1, weight=30), 23262b7b1ffSSeongJae Park _damon_sysfs.DamosDest(id=0, weight=70)]), 23362b7b1ffSSeongJae Park core_filters=[ 23462b7b1ffSSeongJae Park _damon_sysfs.DamosFilter(type_='addr', matching=True, 23562b7b1ffSSeongJae Park allow=False, addr_start=42, 23662b7b1ffSSeongJae Park addr_end=4242), 23762b7b1ffSSeongJae Park ], 23862b7b1ffSSeongJae Park ops_filters=[ 23962b7b1ffSSeongJae Park _damon_sysfs.DamosFilter(type_='anon', matching=True, 24062b7b1ffSSeongJae Park allow=True), 24162b7b1ffSSeongJae Park ], 24262b7b1ffSSeongJae Park )]) 24362b7b1ffSSeongJae Park context.idx = 0 24462b7b1ffSSeongJae Park context.kdamond = kdamonds.kdamonds[0] 24562b7b1ffSSeongJae Park kdamonds.kdamonds[0].contexts = [context] 24662b7b1ffSSeongJae Park kdamonds.kdamonds[0].commit() 24762b7b1ffSSeongJae Park 24862b7b1ffSSeongJae Park status, err = dump_damon_status_dict(kdamonds.kdamonds[0].pid) 24962b7b1ffSSeongJae Park if err is not None: 25062b7b1ffSSeongJae Park print(err) 25162b7b1ffSSeongJae Park exit(1) 25262b7b1ffSSeongJae Park 25362b7b1ffSSeongJae Park assert_ctxs_committed(kdamonds.kdamonds[0].contexts, status['contexts']) 25462b7b1ffSSeongJae Park 255*da5973a0SSeongJae Park # test online commitment of minimum context. 256*da5973a0SSeongJae Park context = _damon_sysfs.DamonCtx() 257*da5973a0SSeongJae Park context.idx = 0 258*da5973a0SSeongJae Park context.kdamond = kdamonds.kdamonds[0] 259*da5973a0SSeongJae Park kdamonds.kdamonds[0].contexts = [context] 260*da5973a0SSeongJae Park kdamonds.kdamonds[0].commit() 261*da5973a0SSeongJae Park 262*da5973a0SSeongJae Park status, err = dump_damon_status_dict(kdamonds.kdamonds[0].pid) 263*da5973a0SSeongJae Park if err is not None: 264*da5973a0SSeongJae Park print(err) 265*da5973a0SSeongJae Park exit(1) 266*da5973a0SSeongJae Park 267*da5973a0SSeongJae Park assert_ctxs_committed(kdamonds.kdamonds[0].contexts, status['contexts']) 268*da5973a0SSeongJae Park 2694ece0189SSeongJae Park kdamonds.stop() 2704ece0189SSeongJae Park 2714ece0189SSeongJae Parkif __name__ == '__main__': 2724ece0189SSeongJae Park main() 273