#!/usr/bin/env python3
# SPDX-License-Identifier: GPL-2.0
import json
import os
import subprocess
import _damon_sysfs
def dump_damon_status_dict(pid):
try :
subprocess.check_output(['which' , 'drgn' ], stderr=subprocess.DEVNULL)
except :
return None , 'drgn not found'
file_dir = os.path.dirname(os.path.abspath(__file__))
dump_script = os.path.join(file_dir, 'drgn_dump_damon_status.py' )
rc = subprocess.call(['drgn' , dump_script, pid, 'damon_dump_output' ],
stderr=subprocess.DEVNULL)
if rc != 0:
return None , 'drgn fail'
try :
with open('damon_dump_output' , 'r' ) as f:
return json.load(f), None
except Exception as e:
return None , 'json.load fail (%s)' % e
def fail(expectation, status):
print('unexpected %s' % expectation)
print(json.dumps(status, indent=4))
exit(1)
def assert_true(condition, expectation, status):
if condition is not True :
fail(expectation, status)
def assert_watermarks_committed(watermarks, dump):
wmark_metric_val = {
'none' : 0,
'free_mem_rate' : 1,
}
assert_true(dump['metric' ] == wmark_metric_val[watermarks.metric],
'metric' , dump)
assert_true(dump['interval' ] == watermarks.interval, 'interval' , dump)
assert_true(dump['high' ] == watermarks.high, 'high' , dump)
assert_true(dump['mid' ] == watermarks.mid, 'mid' , dump)
assert_true(dump['low' ] == watermarks.low, 'low' , dump)
def assert_quota_goal_committed(qgoal, dump):
metric_val = {
'user_input' : 0,
'some_mem_psi_us' : 1,
'node_mem_used_bp' : 2,
'node_mem_free_bp' : 3,
}
assert_true(dump['metric' ] == metric_val[qgoal.metric], 'metric' , dump)
assert_true(dump['target_value' ] == qgoal.target_value, 'target_value' ,
dump)
if qgoal.metric == 'user_input' :
assert_true(dump['current_value' ] == qgoal.current_value,
'current_value' , dump)
assert_true(dump['nid' ] == qgoal.nid, 'nid' , dump)
def assert_quota_committed(quota, dump):
assert_true(dump['reset_interval' ] == quota.reset_interval_ms,
'reset_interval' , dump)
assert_true(dump['ms' ] == quota.ms, 'ms' , dump)
assert_true(dump['sz' ] == quota.sz, 'sz' , dump)
for idx, qgoal in enumerate(quota.goals):
assert_quota_goal_committed(qgoal, dump['goals' ][idx])
assert_true(dump['weight_sz' ] == quota.weight_sz_permil, 'weight_sz' , dump)
assert_true(dump['weight_nr_accesses' ] == quota.weight_nr_accesses_permil,
'weight_nr_accesses' , dump)
assert_true(
dump['weight_age' ] == quota.weight_age_permil, 'weight_age' , dump)
def assert_migrate_dests_committed(dests, dump):
assert_true(dump['nr_dests' ] == len(dests.dests), 'nr_dests' , dump)
for idx, dest in enumerate(dests.dests):
assert_true(dump['node_id_arr' ][idx] == dest.id, 'node_id' , dump)
assert_true(dump['weight_arr' ][idx] == dest.weight, 'weight' , dump)
def assert_filter_committed(filter_, dump):
assert_true(filter_.type_ == dump['type' ], 'type' , dump)
assert_true(filter_.matching == dump['matching' ], 'matching' , dump)
assert_true(filter_.allow == dump['allow' ], 'allow' , dump)
# TODO: check memcg_path and memcg_id if type is memcg
if filter_.type_ == 'addr' :
assert_true([filter_.addr_start, filter_.addr_end] ==
dump['addr_range' ], 'addr_range' , dump)
elif filter_.type_ == 'target' :
assert_true(filter_.target_idx == dump['target_idx' ], 'target_idx' ,
dump)
elif filter_.type_ == 'hugepage_size' :
assert_true([filter_.min_, filter_.max_] == dump['sz_range' ],
'sz_range' , dump)
def assert_access_pattern_committed(pattern, dump):
assert_true(dump['min_sz_region' ] == pattern.size[0], 'min_sz_region' ,
dump)
assert_true(dump['max_sz_region' ] == pattern.size[1], 'max_sz_region' ,
dump)
assert_true(dump['min_nr_accesses' ] == pattern.nr_accesses[0],
'min_nr_accesses' , dump)
assert_true(dump['max_nr_accesses' ] == pattern.nr_accesses[1],
'max_nr_accesses' , dump)
assert_true(dump['min_age_region' ] == pattern.age[0], 'min_age_region' ,
dump)
assert_true(dump['max_age_region' ] == pattern.age[1], 'miaxage_region' ,
dump)
def assert_scheme_committed(scheme, dump):
assert_access_pattern_committed(scheme.access_pattern, dump['pattern' ])
action_val = {
'willneed' : 0,
'cold' : 1,
'pageout' : 2,
'hugepage' : 3,
'nohugeapge' : 4,
'lru_prio' : 5,
'lru_deprio' : 6,
'migrate_hot' : 7,
'migrate_cold' : 8,
'stat' : 9,
}
assert_true(dump['action' ] == action_val[scheme.action], 'action' , dump)
assert_true(dump['apply_interval_us' ] == scheme. apply_interval_us,
'apply_interval_us' , dump)
assert_true(dump['target_nid' ] == scheme.target_nid, 'target_nid' , dump)
assert_migrate_dests_committed(scheme.dests, dump['migrate_dests' ])
assert_quota_committed(scheme.quota, dump['quota' ])
assert_watermarks_committed(scheme.watermarks, dump['wmarks' ])
# TODO: test filters directory
for idx, f in enumerate(scheme.core_filters.filters):
assert_filter_committed(f, dump['filters' ][idx])
for idx, f in enumerate(scheme.ops_filters.filters):
assert_filter_committed(f, dump['ops_filters' ][idx])
def assert_schemes_committed(schemes, dump):
assert_true(len(schemes) == len(dump), 'len_schemes' , dump)
for idx, scheme in enumerate(schemes):
assert_scheme_committed(scheme, dump[idx])
def assert_monitoring_attrs_committed(attrs, dump):
assert_true(dump['sample_interval' ] == attrs.sample_us, 'sample_interval' ,
dump)
assert_true(dump['aggr_interval' ] == attrs.aggr_us, 'aggr_interval' , dump)
assert_true(dump['intervals_goal' ]['access_bp' ] ==
attrs.intervals_goal.access_bp, 'access_bp' ,
dump['intervals_goal' ])
assert_true(dump['intervals_goal' ]['aggrs' ] == attrs.intervals_goal.aggrs,
'aggrs' , dump['intervals_goal' ])
assert_true(dump['intervals_goal' ]['min_sample_us' ] ==
attrs.intervals_goal.min_sample_us, 'min_sample_us' ,
dump['intervals_goal' ])
assert_true(dump['intervals_goal' ]['max_sample_us' ] ==
attrs.intervals_goal.max_sample_us, 'max_sample_us' ,
dump['intervals_goal' ])
assert_true(dump['ops_update_interval' ] == attrs.update_us,
'ops_update_interval' , dump)
assert_true(dump['min_nr_regions' ] == attrs.min_nr_regions,
'min_nr_regions' , dump)
assert_true(dump['max_nr_regions' ] == attrs.max_nr_regions,
'max_nr_regions' , dump)
def assert_ctx_committed(ctx, dump):
ops_val = {
'vaddr' : 0,
'fvaddr' : 1,
'paddr' : 2,
}
assert_true(dump['ops' ]['id' ] == ops_val[ctx.ops], 'ops_id' , dump)
assert_monitoring_attrs_committed(ctx.monitoring_attrs, dump['attrs' ])
assert_schemes_committed(ctx.schemes, dump['schemes' ])
def assert_ctxs_committed(ctxs, dump):
assert_true(len(ctxs) == len(dump), 'ctxs length' , dump)
for idx, ctx in enumerate(ctxs):
assert_ctx_committed(ctx, dump[idx])
def main():
kdamonds = _damon_sysfs.Kdamonds(
[_damon_sysfs.Kdamond(
contexts=[_damon_sysfs.DamonCtx(
targets=[_damon_sysfs.DamonTarget(pid=-1)],
schemes=[_damon_sysfs.Damos()],
)])])
err = kdamonds.start()
if err is not None :
print('kdamond start failed: %s' % err)
exit(1)
status, err = dump_damon_status_dict(kdamonds.kdamonds[0].pid)
if err is not None :
print(err)
kdamonds.stop()
exit(1)
assert_ctxs_committed(kdamonds.kdamonds[0].contexts, status['contexts' ])
context = _damon_sysfs.DamonCtx(
monitoring_attrs=_damon_sysfs.DamonAttrs(
sample_us=100000, aggr_us=2000000,
intervals_goal=_damon_sysfs.IntervalsGoal(
access_bp=400, aggrs=3, min_sample_us=5000,
max_sample_us=10000000),
update_us=2000000),
schemes=[_damon_sysfs.Damos(
action='pageout' ,
access_pattern=_damon_sysfs.DamosAccessPattern(
size=[4096, 2**10],
nr_accesses=[3, 317],
age=[5,71]),
quota=_damon_sysfs.DamosQuota(
sz=100*1024*1024, ms=100,
goals=[_damon_sysfs.DamosQuotaGoal(
metric='node_mem_used_bp' ,
target_value=9950,
nid=1)],
reset_interval_ms=1500,
weight_sz_permil=20,
weight_nr_accesses_permil=200,
weight_age_permil=1000),
watermarks=_damon_sysfs.DamosWatermarks(
metric = 'free_mem_rate' , interval = 500000, # 500 ms
high = 500, mid = 400, low = 50),
target_nid=1,
apply_interval_us=1000000,
dests=_damon_sysfs.DamosDests(
dests=[_damon_sysfs.DamosDest(id=1, weight=30),
_damon_sysfs.DamosDest(id=0, weight=70)]),
core_filters=[
_damon_sysfs.DamosFilter(type_='addr' , matching=True ,
allow=False , addr_start=42,
addr_end=4242),
],
ops_filters=[
_damon_sysfs.DamosFilter(type_='anon' , matching=True ,
allow=True ),
],
)])
context.idx = 0
context.kdamond = kdamonds.kdamonds[0]
kdamonds.kdamonds[0].contexts = [context]
kdamonds.kdamonds[0].commit()
status, err = dump_damon_status_dict(kdamonds.kdamonds[0].pid)
if err is not None :
print(err)
exit(1)
assert_ctxs_committed(kdamonds.kdamonds[0].contexts, status['contexts' ])
# test online commitment of minimum context.
context = _damon_sysfs.DamonCtx()
context.idx = 0
context.kdamond = kdamonds.kdamonds[0]
kdamonds.kdamonds[0].contexts = [context]
kdamonds.kdamonds[0].commit()
status, err = dump_damon_status_dict(kdamonds.kdamonds[0].pid)
if err is not None :
print(err)
exit(1)
assert_ctxs_committed(kdamonds.kdamonds[0].contexts, status['contexts' ])
kdamonds.stop()
if __name__ == '__main__' :
main()
Messung V0.5 C=94 H=94 G=93
¤ Dauer der Verarbeitung: 0.4 Sekunden
¤
*© Formatika GbR, Deutschland