xref: /linux/tools/testing/selftests/damon/sysfs.py (revision 8d2b0853add1d7534dc0794e3c8e0b9e8c4ec640)
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