Caller could retry them differently, return into userspace
as -ENOMEM or silently ignore in cases like disk readahead.
+ memory.max.effective
+ A read-only single value file which exists on non-root cgroups.
+
+ The effective limit of the cgroup i.e. the minimum memory.max
+ of all ancestors including itself. This is useful for environments
+ where cgroup namespace is being used and the application does not
+ have full view of the hierarchy.
+
memory.reclaim
A write-only nested-keyed file which exists for all cgroups.
Swap usage hard limit. If a cgroup's swap usage reaches this
limit, anonymous memory of the cgroup will not be swapped out.
+ memory.swap.max.effective
+ A read-only single value file which exists on non-root cgroups.
+
+ The effective limit of the cgroup i.e. the minimum memory.swap.max
+ of all ancestors including itself. This is useful for environments
+ where cgroup namespace is being used and the application does not
+ have full view of the hierarchy.
+
memory.swap.events
A read-only flat-keyed file which exists on non-root cgroups.
The following entries are defined. Unless specified
limit, it will refuse to take any more stores before existing
entries fault back in or are written out to disk.
+ memory.zswap.max.effective
+ A read-only single value file which exists on non-root cgroups.
+
+ The effective limit of the cgroup i.e. the minimum memory.zswap.max
+ of all ancestors including itself. This is useful for environments
+ where cgroup namespace is being used and the application does not
+ have full view of the hierarchy.
+
memory.zswap.writeback
A read-write single value file. The default value is "1".
Note that this setting is hierarchical, i.e. the writeback would be
READ_ONCE(mem_cgroup_from_seq(m)->memory.max));
}
+static int memory_max_effective_show(struct seq_file *m, void *v)
+{
+ unsigned long max = PAGE_COUNTER_MAX;
+ struct mem_cgroup *memcg = mem_cgroup_from_seq(m);
+
+ for (; memcg; memcg = parent_mem_cgroup(memcg))
+ max = min(max, READ_ONCE(memcg->memory.max));
+
+ return seq_puts_memcg_tunable(m, max);
+}
+
static ssize_t memory_max_write(struct kernfs_open_file *of,
char *buf, size_t nbytes, loff_t off)
{
.seq_show = memory_max_show,
.write = memory_max_write,
},
+ {
+ .name = "max.effective",
+ .flags = CFTYPE_NOT_ON_ROOT,
+ .seq_show = memory_max_effective_show,
+ },
{
.name = "events",
.flags = CFTYPE_NOT_ON_ROOT,
READ_ONCE(mem_cgroup_from_seq(m)->swap.max));
}
+static int swap_max_effective_show(struct seq_file *m, void *v)
+{
+ unsigned long max = PAGE_COUNTER_MAX;
+ struct mem_cgroup *memcg = mem_cgroup_from_seq(m);
+
+ for (; memcg; memcg = parent_mem_cgroup(memcg))
+ max = min(max, READ_ONCE(memcg->swap.max));
+
+ return seq_puts_memcg_tunable(m, max);
+}
+
static ssize_t swap_max_write(struct kernfs_open_file *of,
char *buf, size_t nbytes, loff_t off)
{
.seq_show = swap_max_show,
.write = swap_max_write,
},
+ {
+ .name = "swap.max.effective",
+ .flags = CFTYPE_NOT_ON_ROOT,
+ .seq_show = swap_max_effective_show,
+ },
{
.name = "swap.peak",
.flags = CFTYPE_NOT_ON_ROOT,
READ_ONCE(mem_cgroup_from_seq(m)->zswap_max));
}
+static int zswap_max_effective_show(struct seq_file *m, void *v)
+{
+ unsigned long max = PAGE_COUNTER_MAX;
+ struct mem_cgroup *memcg = mem_cgroup_from_seq(m);
+
+ for (; memcg; memcg = parent_mem_cgroup(memcg))
+ max = min(max, READ_ONCE(memcg->zswap_max));
+
+ return seq_puts_memcg_tunable(m, max);
+}
+
static ssize_t zswap_max_write(struct kernfs_open_file *of,
char *buf, size_t nbytes, loff_t off)
{
.seq_show = zswap_max_show,
.write = zswap_max_write,
},
+ {
+ .name = "zswap.max.effective",
+ .flags = CFTYPE_NOT_ON_ROOT,
+ .seq_show = zswap_max_effective_show,
+ },
{
.name = "zswap.writeback",
.seq_show = zswap_writeback_show,