Path: blob/master/tools/testing/selftests/damon/sysfs.py
49191 views
#!/usr/bin/env python31# SPDX-License-Identifier: GPL-2.023import json4import os5import subprocess67import _damon_sysfs89def dump_damon_status_dict(pid):10try:11subprocess.check_output(['which', 'drgn'], stderr=subprocess.DEVNULL)12except:13return None, 'drgn not found'14file_dir = os.path.dirname(os.path.abspath(__file__))15dump_script = os.path.join(file_dir, 'drgn_dump_damon_status.py')16rc = subprocess.call(['drgn', dump_script, pid, 'damon_dump_output'],17stderr=subprocess.DEVNULL)18if rc != 0:19return None, 'drgn fail'20try:21with open('damon_dump_output', 'r') as f:22return json.load(f), None23except Exception as e:24return None, 'json.load fail (%s)' % e2526def fail(expectation, status):27print('unexpected %s' % expectation)28print(json.dumps(status, indent=4))29exit(1)3031def assert_true(condition, expectation, status):32if condition is not True:33fail(expectation, status)3435def assert_watermarks_committed(watermarks, dump):36wmark_metric_val = {37'none': 0,38'free_mem_rate': 1,39}40assert_true(dump['metric'] == wmark_metric_val[watermarks.metric],41'metric', dump)42assert_true(dump['interval'] == watermarks.interval, 'interval', dump)43assert_true(dump['high'] == watermarks.high, 'high', dump)44assert_true(dump['mid'] == watermarks.mid, 'mid', dump)45assert_true(dump['low'] == watermarks.low, 'low', dump)4647def assert_quota_goal_committed(qgoal, dump):48metric_val = {49'user_input': 0,50'some_mem_psi_us': 1,51'node_mem_used_bp': 2,52'node_mem_free_bp': 3,53}54assert_true(dump['metric'] == metric_val[qgoal.metric], 'metric', dump)55assert_true(dump['target_value'] == qgoal.target_value, 'target_value',56dump)57if qgoal.metric == 'user_input':58assert_true(dump['current_value'] == qgoal.current_value,59'current_value', dump)60assert_true(dump['nid'] == qgoal.nid, 'nid', dump)6162def assert_quota_committed(quota, dump):63assert_true(dump['reset_interval'] == quota.reset_interval_ms,64'reset_interval', dump)65assert_true(dump['ms'] == quota.ms, 'ms', dump)66assert_true(dump['sz'] == quota.sz, 'sz', dump)67for idx, qgoal in enumerate(quota.goals):68assert_quota_goal_committed(qgoal, dump['goals'][idx])69assert_true(dump['weight_sz'] == quota.weight_sz_permil, 'weight_sz', dump)70assert_true(dump['weight_nr_accesses'] == quota.weight_nr_accesses_permil,71'weight_nr_accesses', dump)72assert_true(73dump['weight_age'] == quota.weight_age_permil, 'weight_age', dump)747576def assert_migrate_dests_committed(dests, dump):77assert_true(dump['nr_dests'] == len(dests.dests), 'nr_dests', dump)78for idx, dest in enumerate(dests.dests):79assert_true(dump['node_id_arr'][idx] == dest.id, 'node_id', dump)80assert_true(dump['weight_arr'][idx] == dest.weight, 'weight', dump)8182def assert_filter_committed(filter_, dump):83assert_true(filter_.type_ == dump['type'], 'type', dump)84assert_true(filter_.matching == dump['matching'], 'matching', dump)85assert_true(filter_.allow == dump['allow'], 'allow', dump)86# TODO: check memcg_path and memcg_id if type is memcg87if filter_.type_ == 'addr':88assert_true([filter_.addr_start, filter_.addr_end] ==89dump['addr_range'], 'addr_range', dump)90elif filter_.type_ == 'target':91assert_true(filter_.target_idx == dump['target_idx'], 'target_idx',92dump)93elif filter_.type_ == 'hugepage_size':94assert_true([filter_.min_, filter_.max_] == dump['sz_range'],95'sz_range', dump)9697def assert_access_pattern_committed(pattern, dump):98assert_true(dump['min_sz_region'] == pattern.size[0], 'min_sz_region',99dump)100assert_true(dump['max_sz_region'] == pattern.size[1], 'max_sz_region',101dump)102assert_true(dump['min_nr_accesses'] == pattern.nr_accesses[0],103'min_nr_accesses', dump)104assert_true(dump['max_nr_accesses'] == pattern.nr_accesses[1],105'max_nr_accesses', dump)106assert_true(dump['min_age_region'] == pattern.age[0], 'min_age_region',107dump)108assert_true(dump['max_age_region'] == pattern.age[1], 'miaxage_region',109dump)110111def assert_scheme_committed(scheme, dump):112assert_access_pattern_committed(scheme.access_pattern, dump['pattern'])113action_val = {114'willneed': 0,115'cold': 1,116'pageout': 2,117'hugepage': 3,118'nohugeapge': 4,119'lru_prio': 5,120'lru_deprio': 6,121'migrate_hot': 7,122'migrate_cold': 8,123'stat': 9,124}125assert_true(dump['action'] == action_val[scheme.action], 'action', dump)126assert_true(dump['apply_interval_us'] == scheme. apply_interval_us,127'apply_interval_us', dump)128assert_true(dump['target_nid'] == scheme.target_nid, 'target_nid', dump)129assert_migrate_dests_committed(scheme.dests, dump['migrate_dests'])130assert_quota_committed(scheme.quota, dump['quota'])131assert_watermarks_committed(scheme.watermarks, dump['wmarks'])132# TODO: test filters directory133for idx, f in enumerate(scheme.core_filters.filters):134assert_filter_committed(f, dump['core_filters'][idx])135for idx, f in enumerate(scheme.ops_filters.filters):136assert_filter_committed(f, dump['ops_filters'][idx])137138def assert_schemes_committed(schemes, dump):139assert_true(len(schemes) == len(dump), 'len_schemes', dump)140for idx, scheme in enumerate(schemes):141assert_scheme_committed(scheme, dump[idx])142143def assert_monitoring_attrs_committed(attrs, dump):144assert_true(dump['sample_interval'] == attrs.sample_us, 'sample_interval',145dump)146assert_true(dump['aggr_interval'] == attrs.aggr_us, 'aggr_interval', dump)147assert_true(dump['intervals_goal']['access_bp'] ==148attrs.intervals_goal.access_bp, 'access_bp',149dump['intervals_goal'])150assert_true(dump['intervals_goal']['aggrs'] == attrs.intervals_goal.aggrs,151'aggrs', dump['intervals_goal'])152assert_true(dump['intervals_goal']['min_sample_us'] ==153attrs.intervals_goal.min_sample_us, 'min_sample_us',154dump['intervals_goal'])155assert_true(dump['intervals_goal']['max_sample_us'] ==156attrs.intervals_goal.max_sample_us, 'max_sample_us',157dump['intervals_goal'])158159assert_true(dump['ops_update_interval'] == attrs.update_us,160'ops_update_interval', dump)161assert_true(dump['min_nr_regions'] == attrs.min_nr_regions,162'min_nr_regions', dump)163assert_true(dump['max_nr_regions'] == attrs.max_nr_regions,164'max_nr_regions', dump)165166def assert_monitoring_target_committed(target, dump):167# target.pid is the pid "number", while dump['pid'] is 'struct pid'168# pointer, and hence cannot be compared.169assert_true(dump['obsolete'] == target.obsolete, 'target obsolete', dump)170171def assert_monitoring_targets_committed(targets, dump):172assert_true(len(targets) == len(dump), 'len_targets', dump)173for idx, target in enumerate(targets):174assert_monitoring_target_committed(target, dump[idx])175176def assert_ctx_committed(ctx, dump):177ops_val = {178'vaddr': 0,179'fvaddr': 1,180'paddr': 2,181}182assert_true(dump['ops']['id'] == ops_val[ctx.ops], 'ops_id', dump)183assert_monitoring_attrs_committed(ctx.monitoring_attrs, dump['attrs'])184assert_monitoring_targets_committed(ctx.targets, dump['adaptive_targets'])185assert_schemes_committed(ctx.schemes, dump['schemes'])186187def assert_ctxs_committed(kdamonds):188status, err = dump_damon_status_dict(kdamonds.kdamonds[0].pid)189if err is not None:190print(err)191kdamonds.stop()192exit(1)193194ctxs = kdamonds.kdamonds[0].contexts195dump = status['contexts']196assert_true(len(ctxs) == len(dump), 'ctxs length', dump)197for idx, ctx in enumerate(ctxs):198assert_ctx_committed(ctx, dump[idx])199200def main():201kdamonds = _damon_sysfs.Kdamonds(202[_damon_sysfs.Kdamond(203contexts=[_damon_sysfs.DamonCtx(204targets=[_damon_sysfs.DamonTarget(pid=-1)],205schemes=[_damon_sysfs.Damos()],206)])])207err = kdamonds.start()208if err is not None:209print('kdamond start failed: %s' % err)210exit(1)211212assert_ctxs_committed(kdamonds)213214context = _damon_sysfs.DamonCtx(215monitoring_attrs=_damon_sysfs.DamonAttrs(216sample_us=100000, aggr_us=2000000,217intervals_goal=_damon_sysfs.IntervalsGoal(218access_bp=400, aggrs=3, min_sample_us=5000,219max_sample_us=10000000),220update_us=2000000),221schemes=[_damon_sysfs.Damos(222action='pageout',223access_pattern=_damon_sysfs.DamosAccessPattern(224size=[4096, 2**10],225nr_accesses=[3, 317],226age=[5,71]),227quota=_damon_sysfs.DamosQuota(228sz=100*1024*1024, ms=100,229goals=[_damon_sysfs.DamosQuotaGoal(230metric='node_mem_used_bp',231target_value=9950,232nid=1)],233reset_interval_ms=1500,234weight_sz_permil=20,235weight_nr_accesses_permil=200,236weight_age_permil=1000),237watermarks=_damon_sysfs.DamosWatermarks(238metric = 'free_mem_rate', interval = 500000, # 500 ms239high = 500, mid = 400, low = 50),240target_nid=1,241apply_interval_us=1000000,242dests=_damon_sysfs.DamosDests(243dests=[_damon_sysfs.DamosDest(id=1, weight=30),244_damon_sysfs.DamosDest(id=0, weight=70)]),245core_filters=[246_damon_sysfs.DamosFilter(type_='addr', matching=True,247allow=False, addr_start=42,248addr_end=4242),249],250ops_filters=[251_damon_sysfs.DamosFilter(type_='anon', matching=True,252allow=True),253],254)])255context.idx = 0256context.kdamond = kdamonds.kdamonds[0]257kdamonds.kdamonds[0].contexts = [context]258kdamonds.kdamonds[0].commit()259260assert_ctxs_committed(kdamonds)261262# test online commitment of minimum context.263context = _damon_sysfs.DamonCtx()264context.idx = 0265context.kdamond = kdamonds.kdamonds[0]266kdamonds.kdamonds[0].contexts = [context]267kdamonds.kdamonds[0].commit()268269assert_ctxs_committed(kdamonds)270271kdamonds.stop()272273# test obsolete_target.274proc1 = subprocess.Popen(['sh'], stdout=subprocess.PIPE,275stderr=subprocess.PIPE)276proc2 = subprocess.Popen(['sh'], stdout=subprocess.PIPE,277stderr=subprocess.PIPE)278proc3 = subprocess.Popen(['sh'], stdout=subprocess.PIPE,279stderr=subprocess.PIPE)280kdamonds = _damon_sysfs.Kdamonds(281[_damon_sysfs.Kdamond(282contexts=[_damon_sysfs.DamonCtx(283ops='vaddr',284targets=[285_damon_sysfs.DamonTarget(pid=proc1.pid),286_damon_sysfs.DamonTarget(pid=proc2.pid),287_damon_sysfs.DamonTarget(pid=proc3.pid),288],289schemes=[_damon_sysfs.Damos()],290)])])291err = kdamonds.start()292if err is not None:293print('kdamond start failed: %s' % err)294exit(1)295kdamonds.kdamonds[0].contexts[0].targets[1].obsolete = True296kdamonds.kdamonds[0].commit()297del kdamonds.kdamonds[0].contexts[0].targets[1]298assert_ctxs_committed(kdamonds)299kdamonds.stop()300301if __name__ == '__main__':302main()303304305