Merge tag 'trace-printf-v6.13' of git://git.kernel.org/pub/scm/linux/kernel/git/trace...
[drm/drm-misc.git] / tools / testing / selftests / cgroup / memcg_protection.m
blob051daa3477b67ca6e94d2de4cd30d14d389c211e
1 % SPDX-License-Identifier: GPL-2.0
3 % run as: octave-cli memcg_protection.m
5 % This script simulates reclaim protection behavior on a single level of memcg
6 % hierarchy to illustrate how overcommitted protection spreads among siblings
7 % (as it depends also on their current consumption).
9 % Simulation assumes siblings consumed the initial amount of memory (w/out
10 % reclaim) and then the reclaim starts, all memory is reclaimable, i.e. treated
11 % same. It simulates only non-low reclaim and assumes all memory.min = 0.
13 % Input configurations
14 % --------------------
15 % E number      parent effective protection
16 % n vector      nominal protection of siblings set at the given level (memory.low)
17 % c vector      current consumption -,,- (memory.current)
19 % example from testcase (values in GB)
20 E = 50 / 1024;
21 n = [75 25 0 500 ] / 1024;
22 c = [50 50 50 0] / 1024;
24 % Reclaim parameters
25 % ------------------
27 % Minimal reclaim amount (GB)
28 cluster = 32*4 / 2**20;
30 % Reclaim coefficient (think as 0.5^sc->priority)
31 alpha = .1
33 % Simulation parameters
34 % ---------------------
35 epsilon = 1e-7;
36 timeout = 1000;
38 % Simulation loop
39 % ---------------
41 ch = [];
42 eh = [];
43 rh = [];
45 for t = 1:timeout
46         % low_usage
47         u = min(c, n);
48         siblings = sum(u);
50         % effective_protection()
51         protected = min(n, c);                % start with nominal
52         e = protected * min(1, E / siblings); % normalize overcommit
54         % recursive protection
55         unclaimed = max(0, E - siblings);
56         parent_overuse = sum(c) - siblings;
57         if (unclaimed > 0 && parent_overuse > 0)
58                 overuse = max(0, c - protected);
59                 e += unclaimed * (overuse / parent_overuse);
60         endif
62         % get_scan_count()
63         r = alpha * c;             % assume all memory is in a single LRU list
65         % commit 1bc63fb1272b ("mm, memcg: make scan aggression always exclude protection")
66         sz = max(e, c);
67         r .*= (1 - (e+epsilon) ./ (sz+epsilon));
69         % uncomment to debug prints
70         % e, c, r
72         % nothing to reclaim, reached equilibrium
73         if max(r) < epsilon
74                 break;
75         endif
77         % SWAP_CLUSTER_MAX roundup
78         r = max(r, (r > epsilon) .* cluster);
79         % XXX here I do parallel reclaim of all siblings
80         % in reality reclaim is serialized and each sibling recalculates own residual
81         c = max(c - r, 0);
83         ch = [ch ; c];
84         eh = [eh ; e];
85         rh = [rh ; r];
86 endfor
89 c, e