blob: 7902a727d3b6ea3c5c02adc5ad5e8a11e784b83b [file] [log] [blame]
Greg Kroah-Hartmanb2441312017-11-01 15:07:57 +01001/* SPDX-License-Identifier: GPL-2.0 */
Johannes Weiner3e32cb22014-12-10 15:42:31 -08002#ifndef _LINUX_PAGE_COUNTER_H
3#define _LINUX_PAGE_COUNTER_H
4
5#include <linux/atomic.h>
6#include <linux/kernel.h>
7#include <asm/page.h>
8
9struct page_counter {
Roman Gushchinbbec2e12018-06-07 17:06:18 -070010 atomic_long_t usage;
11 unsigned long max;
Roman Gushchin23067152018-06-07 17:06:22 -070012 unsigned long low;
Johannes Weiner3e32cb22014-12-10 15:42:31 -080013 struct page_counter *parent;
14
Roman Gushchin23067152018-06-07 17:06:22 -070015 /* effective memory.low and memory.low usage tracking */
16 unsigned long elow;
17 atomic_long_t low_usage;
18 atomic_long_t children_low_usage;
19
Johannes Weiner3e32cb22014-12-10 15:42:31 -080020 /* legacy */
21 unsigned long watermark;
22 unsigned long failcnt;
23};
24
25#if BITS_PER_LONG == 32
26#define PAGE_COUNTER_MAX LONG_MAX
27#else
28#define PAGE_COUNTER_MAX (LONG_MAX / PAGE_SIZE)
29#endif
30
31static inline void page_counter_init(struct page_counter *counter,
32 struct page_counter *parent)
33{
Roman Gushchinbbec2e12018-06-07 17:06:18 -070034 atomic_long_set(&counter->usage, 0);
35 counter->max = PAGE_COUNTER_MAX;
Johannes Weiner3e32cb22014-12-10 15:42:31 -080036 counter->parent = parent;
37}
38
39static inline unsigned long page_counter_read(struct page_counter *counter)
40{
Roman Gushchinbbec2e12018-06-07 17:06:18 -070041 return atomic_long_read(&counter->usage);
Johannes Weiner3e32cb22014-12-10 15:42:31 -080042}
43
Johannes Weiner64f21992014-12-10 15:42:45 -080044void page_counter_cancel(struct page_counter *counter, unsigned long nr_pages);
Johannes Weiner3e32cb22014-12-10 15:42:31 -080045void page_counter_charge(struct page_counter *counter, unsigned long nr_pages);
Johannes Weiner6071ca52015-11-05 18:50:26 -080046bool page_counter_try_charge(struct page_counter *counter,
47 unsigned long nr_pages,
48 struct page_counter **fail);
Johannes Weiner64f21992014-12-10 15:42:45 -080049void page_counter_uncharge(struct page_counter *counter, unsigned long nr_pages);
Roman Gushchinbbec2e12018-06-07 17:06:18 -070050int page_counter_set_max(struct page_counter *counter, unsigned long nr_pages);
Roman Gushchin23067152018-06-07 17:06:22 -070051void page_counter_set_low(struct page_counter *counter, unsigned long nr_pages);
Johannes Weiner650c5e52015-02-11 15:26:03 -080052int page_counter_memparse(const char *buf, const char *max,
53 unsigned long *nr_pages);
Johannes Weiner3e32cb22014-12-10 15:42:31 -080054
55static inline void page_counter_reset_watermark(struct page_counter *counter)
56{
57 counter->watermark = page_counter_read(counter);
58}
59
60#endif /* _LINUX_PAGE_COUNTER_H */