xref: /linux/tools/testing/selftests/damon/sysfs.py (revision bd0487a7745ea84fb0ece7fd796ad51a92a12c70)
1#!/usr/bin/env python3
2# SPDX-License-Identifier: GPL-2.0
3
4import json
5import os
6import subprocess
7
8import _damon_sysfs
9
10def dump_damon_status_dict(pid):
11    try:
12        subprocess.check_output(['which', 'drgn'], stderr=subprocess.DEVNULL)
13    except:
14        return None, 'drgn not found'
15    file_dir = os.path.dirname(os.path.abspath(__file__))
16    dump_script = os.path.join(file_dir, 'drgn_dump_damon_status.py')
17    rc = subprocess.call(['drgn', dump_script, pid, 'damon_dump_output'],
18                         stderr=subprocess.DEVNULL)
19    if rc != 0:
20        return None, 'drgn fail'
21    try:
22        with open('damon_dump_output', 'r') as f:
23            return json.load(f), None
24    except Exception as e:
25        return None, 'json.load fail (%s)' % e
26
27def fail(expectation, status):
28    print('unexpected %s' % expectation)
29    print(json.dumps(status, indent=4))
30    exit(1)
31
32def assert_true(condition, expectation, status):
33    if condition is not True:
34        fail(expectation, status)
35
36def assert_watermarks_committed(watermarks, dump):
37    wmark_metric_val = {
38            'none': 0,
39            'free_mem_rate': 1,
40            }
41    assert_true(dump['metric'] == wmark_metric_val[watermarks.metric],
42                'metric', dump)
43    assert_true(dump['interval'] == watermarks.interval, 'interval', dump)
44    assert_true(dump['high'] == watermarks.high, 'high', dump)
45    assert_true(dump['mid'] == watermarks.mid, 'mid', dump)
46    assert_true(dump['low'] == watermarks.low, 'low', dump)
47
48def assert_quota_goal_committed(qgoal, dump):
49    metric_val = {
50            'user_input': 0,
51            'some_mem_psi_us': 1,
52            'node_mem_used_bp': 2,
53            'node_mem_free_bp': 3,
54            }
55    assert_true(dump['metric'] == metric_val[qgoal.metric], 'metric', dump)
56    assert_true(dump['target_value'] == qgoal.target_value, 'target_value',
57                dump)
58    if qgoal.metric == 'user_input':
59        assert_true(dump['current_value'] == qgoal.current_value,
60                    'current_value', dump)
61    assert_true(dump['nid'] == qgoal.nid, 'nid', dump)
62
63def assert_quota_committed(quota, dump):
64    assert_true(dump['reset_interval'] == quota.reset_interval_ms,
65                'reset_interval', dump)
66    assert_true(dump['ms'] == quota.ms, 'ms', dump)
67    assert_true(dump['sz'] == quota.sz, 'sz', dump)
68    for idx, qgoal in enumerate(quota.goals):
69        assert_quota_goal_committed(qgoal, dump['goals'][idx])
70    assert_true(dump['weight_sz'] == quota.weight_sz_permil, 'weight_sz', dump)
71    assert_true(dump['weight_nr_accesses'] == quota.weight_nr_accesses_permil,
72                'weight_nr_accesses', dump)
73    assert_true(
74            dump['weight_age'] == quota.weight_age_permil, 'weight_age', dump)
75
76
77def assert_migrate_dests_committed(dests, dump):
78    assert_true(dump['nr_dests'] == len(dests.dests), 'nr_dests', dump)
79    for idx, dest in enumerate(dests.dests):
80        assert_true(dump['node_id_arr'][idx] == dest.id, 'node_id', dump)
81        assert_true(dump['weight_arr'][idx] == dest.weight, 'weight', dump)
82
83def main():
84    kdamonds = _damon_sysfs.Kdamonds(
85            [_damon_sysfs.Kdamond(
86                contexts=[_damon_sysfs.DamonCtx(
87                    targets=[_damon_sysfs.DamonTarget(pid=-1)],
88                    schemes=[_damon_sysfs.Damos()],
89                    )])])
90    err = kdamonds.start()
91    if err is not None:
92        print('kdamond start failed: %s' % err)
93        exit(1)
94
95    status, err = dump_damon_status_dict(kdamonds.kdamonds[0].pid)
96    if err is not None:
97        print(err)
98        kdamonds.stop()
99        exit(1)
100
101    if len(status['contexts']) != 1:
102        fail('number of contexts', status)
103
104    ctx = status['contexts'][0]
105    attrs = ctx['attrs']
106    if attrs['sample_interval'] != 5000:
107        fail('sample interval', status)
108    if attrs['aggr_interval'] != 100000:
109        fail('aggr interval', status)
110    if attrs['ops_update_interval'] != 1000000:
111        fail('ops updte interval', status)
112
113    if attrs['intervals_goal'] != {
114            'access_bp': 0, 'aggrs': 0,
115            'min_sample_us': 0, 'max_sample_us': 0}:
116        fail('intervals goal')
117
118    if attrs['min_nr_regions'] != 10:
119        fail('min_nr_regions')
120    if attrs['max_nr_regions'] != 1000:
121        fail('max_nr_regions')
122
123    if ctx['adaptive_targets'] != [
124            { 'pid': 0, 'nr_regions': 0, 'regions_list': []}]:
125        fail('adaptive targets', status)
126
127    if len(ctx['schemes']) != 1:
128        fail('number of schemes', status)
129
130    scheme = ctx['schemes'][0]
131    if scheme['pattern'] != {
132            'min_sz_region': 0,
133            'max_sz_region': 2**64 - 1,
134            'min_nr_accesses': 0,
135            'max_nr_accesses': 2**32 - 1,
136            'min_age_region': 0,
137            'max_age_region': 2**32 - 1,
138            }:
139        fail('damos pattern', status)
140    if scheme['action'] != 9:   # stat
141        fail('damos action', status)
142    if scheme['apply_interval_us'] != 0:
143        fail('damos apply interval', status)
144    if scheme['target_nid'] != -1:
145        fail('damos target nid', status)
146
147    assert_migrate_dests_committed(_damon_sysfs.DamosDests(),
148                                   scheme['migrate_dests'])
149    assert_quota_committed(_damon_sysfs.DamosQuota(), scheme['quota'])
150    assert_watermarks_committed(_damon_sysfs.DamosWatermarks(),
151                                scheme['wmarks'])
152
153    kdamonds.stop()
154
155if __name__ == '__main__':
156    main()
157