2025-06-28 09:04:25 -07:00
|
|
|
#!/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):
|
2025-07-21 23:03:30 -07:00
|
|
|
try:
|
|
|
|
subprocess.check_output(['which', 'drgn'], stderr=subprocess.DEVNULL)
|
|
|
|
except:
|
|
|
|
return None, 'drgn not found'
|
2025-06-28 09:04:25 -07:00
|
|
|
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
|
|
|
|
|
2025-06-28 09:04:26 -07:00
|
|
|
def fail(expectation, status):
|
|
|
|
print('unexpected %s' % expectation)
|
|
|
|
print(json.dumps(status, indent=4))
|
|
|
|
exit(1)
|
|
|
|
|
2025-07-20 10:16:42 -07:00
|
|
|
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)
|
|
|
|
|
2025-07-20 10:16:44 -07:00
|
|
|
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)
|
|
|
|
|
2025-07-20 10:16:43 -07:00
|
|
|
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)
|
2025-07-20 10:16:44 -07:00
|
|
|
for idx, qgoal in enumerate(quota.goals):
|
|
|
|
assert_quota_goal_committed(qgoal, dump['goals'][idx])
|
2025-07-20 10:16:43 -07:00
|
|
|
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)
|
|
|
|
|
2025-07-20 10:16:45 -07:00
|
|
|
|
|
|
|
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)
|
|
|
|
|
2025-07-20 10:16:47 -07:00
|
|
|
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)
|
|
|
|
|
2025-07-20 10:16:46 -07:00
|
|
|
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'])
|
2025-07-20 10:16:47 -07:00
|
|
|
# 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])
|
2025-07-20 10:16:46 -07:00
|
|
|
|
2025-07-20 10:16:48 -07:00
|
|
|
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])
|
|
|
|
|
2025-07-20 10:16:49 -07:00
|
|
|
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)
|
|
|
|
|
2025-07-20 10:16:50 -07:00
|
|
|
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])
|
|
|
|
|
2025-06-28 09:04:25 -07:00
|
|
|
def main():
|
|
|
|
kdamonds = _damon_sysfs.Kdamonds(
|
2025-06-28 09:04:27 -07:00
|
|
|
[_damon_sysfs.Kdamond(
|
|
|
|
contexts=[_damon_sysfs.DamonCtx(
|
2025-06-28 09:04:28 -07:00
|
|
|
targets=[_damon_sysfs.DamonTarget(pid=-1)],
|
|
|
|
schemes=[_damon_sysfs.Damos()],
|
|
|
|
)])])
|
2025-06-28 09:04:25 -07:00
|
|
|
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)
|
2025-07-21 23:03:30 -07:00
|
|
|
kdamonds.stop()
|
2025-06-28 09:04:25 -07:00
|
|
|
exit(1)
|
|
|
|
|
2025-07-20 10:16:50 -07:00
|
|
|
assert_ctxs_committed(kdamonds.kdamonds[0].contexts, status['contexts'])
|
2025-06-28 09:04:26 -07:00
|
|
|
|
2025-07-20 10:16:51 -07:00
|
|
|
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'])
|
|
|
|
|
2025-07-20 10:16:52 -07:00
|
|
|
# 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'])
|
|
|
|
|
2025-06-28 09:04:25 -07:00
|
|
|
kdamonds.stop()
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|