1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889 |
- % SPDX-License-Identifier: GPL-2.0
- %
- % run as: octave-cli memcg_protection.m
- %
- % This script simulates reclaim protection behavior on a single level of memcg
- % hierarchy to illustrate how overcommitted protection spreads among siblings
- % (as it depends also on their current consumption).
- %
- % Simulation assumes siblings consumed the initial amount of memory (w/out
- % reclaim) and then the reclaim starts, all memory is reclaimable, i.e. treated
- % same. It simulates only non-low reclaim and assumes all memory.min = 0.
- %
- % Input configurations
- % --------------------
- % E number parent effective protection
- % n vector nominal protection of siblings set at the given level (memory.low)
- % c vector current consumption -,,- (memory.current)
- % example from testcase (values in GB)
- E = 50 / 1024;
- n = [75 25 0 500 ] / 1024;
- c = [50 50 50 0] / 1024;
- % Reclaim parameters
- % ------------------
- % Minimal reclaim amount (GB)
- cluster = 32*4 / 2**20;
- % Reclaim coefficient (think as 0.5^sc->priority)
- alpha = .1
- % Simulation parameters
- % ---------------------
- epsilon = 1e-7;
- timeout = 1000;
- % Simulation loop
- % ---------------
- ch = [];
- eh = [];
- rh = [];
- for t = 1:timeout
- % low_usage
- u = min(c, n);
- siblings = sum(u);
- % effective_protection()
- protected = min(n, c); % start with nominal
- e = protected * min(1, E / siblings); % normalize overcommit
- % recursive protection
- unclaimed = max(0, E - siblings);
- parent_overuse = sum(c) - siblings;
- if (unclaimed > 0 && parent_overuse > 0)
- overuse = max(0, c - protected);
- e += unclaimed * (overuse / parent_overuse);
- endif
- % get_scan_count()
- r = alpha * c; % assume all memory is in a single LRU list
- % commit 1bc63fb1272b ("mm, memcg: make scan aggression always exclude protection")
- sz = max(e, c);
- r .*= (1 - (e+epsilon) ./ (sz+epsilon));
- % uncomment to debug prints
- % e, c, r
- % nothing to reclaim, reached equilibrium
- if max(r) < epsilon
- break;
- endif
- % SWAP_CLUSTER_MAX roundup
- r = max(r, (r > epsilon) .* cluster);
- % XXX here I do parallel reclaim of all siblings
- % in reality reclaim is serialized and each sibling recalculates own residual
- c = max(c - r, 0);
- ch = [ch ; c];
- eh = [eh ; e];
- rh = [rh ; r];
- endfor
- t
- c, e
|