4 #include <linux/list.h>
6 #include <linux/mutex.h>
7 #include <linux/shrinker.h>
9 #include "tools-util.h"
11 static LIST_HEAD(shrinker_list);
12 static DEFINE_MUTEX(shrinker_lock);
14 int register_shrinker(struct shrinker *shrinker, const char *fmt, ...)
16 mutex_lock(&shrinker_lock);
17 list_add_tail(&shrinker->list, &shrinker_list);
18 mutex_unlock(&shrinker_lock);
22 void unregister_shrinker(struct shrinker *shrinker)
24 mutex_lock(&shrinker_lock);
25 list_del(&shrinker->list);
26 mutex_unlock(&shrinker_lock);
34 static u64 parse_meminfo_line(const char *line)
38 if (sscanf(line, " %llu kB", &v) < 1)
43 void si_meminfo(struct sysinfo *val)
50 memset(val, 0, sizeof(*val));
52 f = fopen("/proc/meminfo", "r");
56 while ((len = getline(&line, &n, f)) != -1) {
57 if ((v = strcmp_prefix(line, "MemTotal:")))
58 val->totalram = parse_meminfo_line(v);
60 if ((v = strcmp_prefix(line, "MemAvailable:")))
61 val->freeram = parse_meminfo_line(v);
68 static void run_shrinkers_allocation_failed(gfp_t gfp_mask)
70 struct shrinker *shrinker;
72 mutex_lock(&shrinker_lock);
73 list_for_each_entry(shrinker, &shrinker_list, list) {
74 struct shrink_control sc = { .gfp_mask = gfp_mask, };
76 unsigned long have = shrinker->count_objects(shrinker, &sc);
78 sc.nr_to_scan = have / 8;
80 shrinker->scan_objects(shrinker, &sc);
82 mutex_unlock(&shrinker_lock);
85 void run_shrinkers(gfp_t gfp_mask, bool allocation_failed)
87 struct shrinker *shrinker;
91 if (!(gfp_mask & GFP_KERNEL))
94 /* Fast out if there are no shrinkers to run. */
95 if (list_empty(&shrinker_list))
98 if (allocation_failed) {
99 run_shrinkers_allocation_failed(gfp_mask);
105 if (info.totalram && info.freeram) {
106 want_shrink = (info.totalram >> 2) - info.freeram;
108 if (want_shrink <= 0)
111 /* If we weren't able to read /proc/meminfo, we must be pretty
114 want_shrink = 8 << 20;
117 mutex_lock(&shrinker_lock);
118 list_for_each_entry(shrinker, &shrinker_list, list) {
119 struct shrink_control sc = {
120 .gfp_mask = gfp_mask,
121 .nr_to_scan = want_shrink >> PAGE_SHIFT
124 shrinker->scan_objects(shrinker, &sc);
126 mutex_unlock(&shrinker_lock);