xref: /linux/tools/testing/selftests/damon/sysfs.py (revision f797e709f741d5a28e4f3ca2592e5cc1d7e21e3b)
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
48*f797e709SSeongJae Parkdef assert_quota_committed(quota, dump):
49*f797e709SSeongJae Park    assert_true(dump['reset_interval'] == quota.reset_interval_ms,
50*f797e709SSeongJae Park                'reset_interval', dump)
51*f797e709SSeongJae Park    assert_true(dump['ms'] == quota.ms, 'ms', dump)
52*f797e709SSeongJae Park    assert_true(dump['sz'] == quota.sz, 'sz', dump)
53*f797e709SSeongJae Park    # TODO: assert goals are committed
54*f797e709SSeongJae Park    assert_true(dump['weight_sz'] == quota.weight_sz_permil, 'weight_sz', dump)
55*f797e709SSeongJae Park    assert_true(dump['weight_nr_accesses'] == quota.weight_nr_accesses_permil,
56*f797e709SSeongJae Park                'weight_nr_accesses', dump)
57*f797e709SSeongJae Park    assert_true(
58*f797e709SSeongJae Park            dump['weight_age'] == quota.weight_age_permil, 'weight_age', dump)
59*f797e709SSeongJae Park
604ece0189SSeongJae Parkdef main():
614ece0189SSeongJae Park    kdamonds = _damon_sysfs.Kdamonds(
627e6bcf35SSeongJae Park            [_damon_sysfs.Kdamond(
637e6bcf35SSeongJae Park                contexts=[_damon_sysfs.DamonCtx(
64603cb4aaSSeongJae Park                    targets=[_damon_sysfs.DamonTarget(pid=-1)],
65603cb4aaSSeongJae Park                    schemes=[_damon_sysfs.Damos()],
66603cb4aaSSeongJae Park                    )])])
674ece0189SSeongJae Park    err = kdamonds.start()
684ece0189SSeongJae Park    if err is not None:
694ece0189SSeongJae Park        print('kdamond start failed: %s' % err)
704ece0189SSeongJae Park        exit(1)
714ece0189SSeongJae Park
724ece0189SSeongJae Park    status, err = dump_damon_status_dict(kdamonds.kdamonds[0].pid)
734ece0189SSeongJae Park    if err is not None:
744ece0189SSeongJae Park        print(err)
75cf20cb9aSSeongJae Park        kdamonds.stop()
764ece0189SSeongJae Park        exit(1)
774ece0189SSeongJae Park
784ece0189SSeongJae Park    if len(status['contexts']) != 1:
79ae3ab07eSSeongJae Park        fail('number of contexts', status)
80ae3ab07eSSeongJae Park
81ae3ab07eSSeongJae Park    ctx = status['contexts'][0]
82ae3ab07eSSeongJae Park    attrs = ctx['attrs']
83ae3ab07eSSeongJae Park    if attrs['sample_interval'] != 5000:
84ae3ab07eSSeongJae Park        fail('sample interval', status)
85ae3ab07eSSeongJae Park    if attrs['aggr_interval'] != 100000:
86ae3ab07eSSeongJae Park        fail('aggr interval', status)
87ae3ab07eSSeongJae Park    if attrs['ops_update_interval'] != 1000000:
88ae3ab07eSSeongJae Park        fail('ops updte interval', status)
89ae3ab07eSSeongJae Park
90ae3ab07eSSeongJae Park    if attrs['intervals_goal'] != {
91ae3ab07eSSeongJae Park            'access_bp': 0, 'aggrs': 0,
92ae3ab07eSSeongJae Park            'min_sample_us': 0, 'max_sample_us': 0}:
93ae3ab07eSSeongJae Park        fail('intervals goal')
94ae3ab07eSSeongJae Park
95ae3ab07eSSeongJae Park    if attrs['min_nr_regions'] != 10:
96ae3ab07eSSeongJae Park        fail('min_nr_regions')
97ae3ab07eSSeongJae Park    if attrs['max_nr_regions'] != 1000:
98ae3ab07eSSeongJae Park        fail('max_nr_regions')
99ae3ab07eSSeongJae Park
1007e6bcf35SSeongJae Park    if ctx['adaptive_targets'] != [
1017e6bcf35SSeongJae Park            { 'pid': 0, 'nr_regions': 0, 'regions_list': []}]:
1027e6bcf35SSeongJae Park        fail('adaptive targets', status)
103ae3ab07eSSeongJae Park
104603cb4aaSSeongJae Park    if len(ctx['schemes']) != 1:
105603cb4aaSSeongJae Park        fail('number of schemes', status)
106603cb4aaSSeongJae Park
107603cb4aaSSeongJae Park    scheme = ctx['schemes'][0]
108603cb4aaSSeongJae Park    if scheme['pattern'] != {
109603cb4aaSSeongJae Park            'min_sz_region': 0,
110603cb4aaSSeongJae Park            'max_sz_region': 2**64 - 1,
111603cb4aaSSeongJae Park            'min_nr_accesses': 0,
112603cb4aaSSeongJae Park            'max_nr_accesses': 2**32 - 1,
113603cb4aaSSeongJae Park            'min_age_region': 0,
114603cb4aaSSeongJae Park            'max_age_region': 2**32 - 1,
115603cb4aaSSeongJae Park            }:
116603cb4aaSSeongJae Park        fail('damos pattern', status)
117603cb4aaSSeongJae Park    if scheme['action'] != 9:   # stat
118603cb4aaSSeongJae Park        fail('damos action', status)
119603cb4aaSSeongJae Park    if scheme['apply_interval_us'] != 0:
120603cb4aaSSeongJae Park        fail('damos apply interval', status)
121603cb4aaSSeongJae Park    if scheme['target_nid'] != -1:
122603cb4aaSSeongJae Park        fail('damos target nid', status)
123603cb4aaSSeongJae Park
124*f797e709SSeongJae Park    migrate_dests = scheme['migrate_dests']
125*f797e709SSeongJae Park    if migrate_dests['nr_dests'] != 0:
126*f797e709SSeongJae Park        fail('nr_dests', status)
127*f797e709SSeongJae Park    if migrate_dests['node_id_arr'] != []:
128*f797e709SSeongJae Park        fail('node_id_arr', status)
129*f797e709SSeongJae Park    if migrate_dests['weight_arr'] != []:
130*f797e709SSeongJae Park        fail('weight_arr', status)
131603cb4aaSSeongJae Park
132*f797e709SSeongJae Park    assert_quota_committed(_damon_sysfs.DamosQuota(), scheme['quota'])
133b50c48deSSeongJae Park    assert_watermarks_committed(_damon_sysfs.DamosWatermarks(),
134b50c48deSSeongJae Park                                scheme['wmarks'])
135ae3ab07eSSeongJae Park
1364ece0189SSeongJae Park    kdamonds.stop()
1374ece0189SSeongJae Park
1384ece0189SSeongJae Parkif __name__ == '__main__':
1394ece0189SSeongJae Park    main()
140