Re: [PATCH 13/14] mm: memcontrol: account socket memory in unified hierarchy memory controller
On Mon, Nov 23, 2015 at 01:00:59PM +0300, Vladimir Davydov wrote: > I've another question regarding this socket_work: its reclaim target > always equals CHARGE_BATCH. Can't it result in a workload exceeding > memory.high in case there are a lot of allocations coming from different > cpus? In this case the work might not manage to complete before another > allocation happens. May be, we should accumulate the number of pages to > be reclaimed by the work, as we do in try_charge? Actually, try_to_free_mem_cgroup_pages() rounds it up to 2MB anyway. I would hate to add locking or more atomics to accumulate a reclaim goal for the worker on spec, so let's wait to see if this is a real issue. > > > BTW why do we need this work at all? Why is reclaim_high called from > > > task_work not enough? > > > > The problem lies in the memcg association: the random task that gets > > interrupted by an arriving packet might not be in the same memcg as > > the one owning receiving socket. And multiple interrupts could happen > > while we're in the kernel already charging pages. We'd basically have > > to maintain a list of memcgs that need to run reclaim_high associated > > with current. > > > > Right, I think this is worth placing in a comment to memcg->socket_work. Okay, will do. > I wonder if we could use it *instead* of task_work for handling every > allocation, not only socket-related. Would it make any sense? May be, it > could reduce the latency experienced by tasks in memory cgroups. No, we *want* charging tasks to do reclaim work once memory.high is breached, in order to match their speed to memory availability. That needs to remain synchroneous. What we could try is make memcg->socket_work purely about the receive side when we're inside the softirq, and arm the per-task work when in process context on the sending side. I'll look into that. -- To unsubscribe from this list: send the line "unsubscribe netdev" in the body of a message to majord...@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
Re: [PATCH 13/14] mm: memcontrol: account socket memory in unified hierarchy memory controller
On Fri, Nov 20, 2015 at 02:25:06PM -0500, Johannes Weiner wrote: > On Fri, Nov 20, 2015 at 04:10:33PM +0300, Vladimir Davydov wrote: > > On Thu, Nov 12, 2015 at 06:41:32PM -0500, Johannes Weiner wrote: > > ... > > > @@ -5514,16 +5550,43 @@ void sock_release_memcg(struct sock *sk) > > > */ > > > bool mem_cgroup_charge_skmem(struct mem_cgroup *memcg, unsigned int > > > nr_pages) > > > { > > > + unsigned int batch = max(CHARGE_BATCH, nr_pages); > > > struct page_counter *counter; > > > + bool force = false; > > > > > > - if (page_counter_try_charge(&memcg->tcp_mem.memory_allocated, > > > - nr_pages, &counter)) { > > > - memcg->tcp_mem.memory_pressure = 0; > > > +#ifdef CONFIG_MEMCG_KMEM > > > + if (!cgroup_subsys_on_dfl(memory_cgrp_subsys)) { > > > + if (page_counter_try_charge(&memcg->tcp_mem.memory_allocated, > > > + nr_pages, &counter)) { > > > + memcg->tcp_mem.memory_pressure = 0; > > > + return true; > > > + } > > > + page_counter_charge(&memcg->tcp_mem.memory_allocated, nr_pages); > > > + memcg->tcp_mem.memory_pressure = 1; > > > + return false; > > > + } > > > +#endif > > > + if (consume_stock(memcg, nr_pages)) > > > return true; > > > +retry: > > > + if (page_counter_try_charge(&memcg->memory, batch, &counter)) > > > + goto done; > > > + > > > + if (batch > nr_pages) { > > > + batch = nr_pages; > > > + goto retry; > > > } > > > - page_counter_charge(&memcg->tcp_mem.memory_allocated, nr_pages); > > > - memcg->tcp_mem.memory_pressure = 1; > > > - return false; > > > + > > > + page_counter_charge(&memcg->memory, batch); > > > + force = true; > > > +done: > > > > > + css_get_many(&memcg->css, batch); > > > > Is there any point to get css reference per each charged page? For kmem > > it is absolutely necessary, because dangling slabs must block > > destruction of memcg's kmem caches, which are destroyed on css_free. But > > for sockets there's no such problem: memcg will be destroyed only after > > all sockets are destroyed and therefore uncharged (since > > sock_update_memcg pins css). > > I'm afraid we have to when we want to share 'stock' with cache and > anon pages, which hold individual references. drain_stock() always > assumes one reference per cached page. Missed that, you're right. > > > > + if (batch > nr_pages) > > > + refill_stock(memcg, batch - nr_pages); > > > + > > > + schedule_work(&memcg->socket_work); > > > > I think it's suboptimal to schedule the work even if we are below the > > high threshold. > > Hm, it seemed unnecessary to duplicate the hierarchy check since this > is in the batch-exhausted slowpath anyway. Dunno, may be you're right. I've another question regarding this socket_work: its reclaim target always equals CHARGE_BATCH. Can't it result in a workload exceeding memory.high in case there are a lot of allocations coming from different cpus? In this case the work might not manage to complete before another allocation happens. May be, we should accumulate the number of pages to be reclaimed by the work, as we do in try_charge? > > > BTW why do we need this work at all? Why is reclaim_high called from > > task_work not enough? > > The problem lies in the memcg association: the random task that gets > interrupted by an arriving packet might not be in the same memcg as > the one owning receiving socket. And multiple interrupts could happen > while we're in the kernel already charging pages. We'd basically have > to maintain a list of memcgs that need to run reclaim_high associated > with current. > Right, I think this is worth placing in a comment to memcg->socket_work. I wonder if we could use it *instead* of task_work for handling every allocation, not only socket-related. Would it make any sense? May be, it could reduce the latency experienced by tasks in memory cgroups. Thanks, Vladimir -- To unsubscribe from this list: send the line "unsubscribe netdev" in the body of a message to majord...@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
Re: [PATCH 13/14] mm: memcontrol: account socket memory in unified hierarchy memory controller
On Fri, Nov 20, 2015 at 04:10:33PM +0300, Vladimir Davydov wrote: > On Thu, Nov 12, 2015 at 06:41:32PM -0500, Johannes Weiner wrote: > ... > > @@ -5514,16 +5550,43 @@ void sock_release_memcg(struct sock *sk) > > */ > > bool mem_cgroup_charge_skmem(struct mem_cgroup *memcg, unsigned int > > nr_pages) > > { > > + unsigned int batch = max(CHARGE_BATCH, nr_pages); > > struct page_counter *counter; > > + bool force = false; > > > > - if (page_counter_try_charge(&memcg->tcp_mem.memory_allocated, > > - nr_pages, &counter)) { > > - memcg->tcp_mem.memory_pressure = 0; > > +#ifdef CONFIG_MEMCG_KMEM > > + if (!cgroup_subsys_on_dfl(memory_cgrp_subsys)) { > > + if (page_counter_try_charge(&memcg->tcp_mem.memory_allocated, > > + nr_pages, &counter)) { > > + memcg->tcp_mem.memory_pressure = 0; > > + return true; > > + } > > + page_counter_charge(&memcg->tcp_mem.memory_allocated, nr_pages); > > + memcg->tcp_mem.memory_pressure = 1; > > + return false; > > + } > > +#endif > > + if (consume_stock(memcg, nr_pages)) > > return true; > > +retry: > > + if (page_counter_try_charge(&memcg->memory, batch, &counter)) > > + goto done; > > + > > + if (batch > nr_pages) { > > + batch = nr_pages; > > + goto retry; > > } > > - page_counter_charge(&memcg->tcp_mem.memory_allocated, nr_pages); > > - memcg->tcp_mem.memory_pressure = 1; > > - return false; > > + > > + page_counter_charge(&memcg->memory, batch); > > + force = true; > > +done: > > > + css_get_many(&memcg->css, batch); > > Is there any point to get css reference per each charged page? For kmem > it is absolutely necessary, because dangling slabs must block > destruction of memcg's kmem caches, which are destroyed on css_free. But > for sockets there's no such problem: memcg will be destroyed only after > all sockets are destroyed and therefore uncharged (since > sock_update_memcg pins css). I'm afraid we have to when we want to share 'stock' with cache and anon pages, which hold individual references. drain_stock() always assumes one reference per cached page. > > + if (batch > nr_pages) > > + refill_stock(memcg, batch - nr_pages); > > + > > + schedule_work(&memcg->socket_work); > > I think it's suboptimal to schedule the work even if we are below the > high threshold. Hm, it seemed unnecessary to duplicate the hierarchy check since this is in the batch-exhausted slowpath anyway. > BTW why do we need this work at all? Why is reclaim_high called from > task_work not enough? The problem lies in the memcg association: the random task that gets interrupted by an arriving packet might not be in the same memcg as the one owning receiving socket. And multiple interrupts could happen while we're in the kernel already charging pages. We'd basically have to maintain a list of memcgs that need to run reclaim_high associated with current. -- To unsubscribe from this list: send the line "unsubscribe netdev" in the body of a message to majord...@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
Re: [PATCH 13/14] mm: memcontrol: account socket memory in unified hierarchy memory controller
On Thu, Nov 12, 2015 at 06:41:32PM -0500, Johannes Weiner wrote: ... > @@ -5514,16 +5550,43 @@ void sock_release_memcg(struct sock *sk) > */ > bool mem_cgroup_charge_skmem(struct mem_cgroup *memcg, unsigned int nr_pages) > { > + unsigned int batch = max(CHARGE_BATCH, nr_pages); > struct page_counter *counter; > + bool force = false; > > - if (page_counter_try_charge(&memcg->tcp_mem.memory_allocated, > - nr_pages, &counter)) { > - memcg->tcp_mem.memory_pressure = 0; > +#ifdef CONFIG_MEMCG_KMEM > + if (!cgroup_subsys_on_dfl(memory_cgrp_subsys)) { > + if (page_counter_try_charge(&memcg->tcp_mem.memory_allocated, > + nr_pages, &counter)) { > + memcg->tcp_mem.memory_pressure = 0; > + return true; > + } > + page_counter_charge(&memcg->tcp_mem.memory_allocated, nr_pages); > + memcg->tcp_mem.memory_pressure = 1; > + return false; > + } > +#endif > + if (consume_stock(memcg, nr_pages)) > return true; > +retry: > + if (page_counter_try_charge(&memcg->memory, batch, &counter)) > + goto done; > + > + if (batch > nr_pages) { > + batch = nr_pages; > + goto retry; > } > - page_counter_charge(&memcg->tcp_mem.memory_allocated, nr_pages); > - memcg->tcp_mem.memory_pressure = 1; > - return false; > + > + page_counter_charge(&memcg->memory, batch); > + force = true; > +done: > + css_get_many(&memcg->css, batch); Is there any point to get css reference per each charged page? For kmem it is absolutely necessary, because dangling slabs must block destruction of memcg's kmem caches, which are destroyed on css_free. But for sockets there's no such problem: memcg will be destroyed only after all sockets are destroyed and therefore uncharged (since sock_update_memcg pins css). > + if (batch > nr_pages) > + refill_stock(memcg, batch - nr_pages); > + > + schedule_work(&memcg->socket_work); I think it's suboptimal to schedule the work even if we are below the high threshold. BTW why do we need this work at all? Why is reclaim_high called from task_work not enough? Thanks, Vladimir > + > + return !force; > } > > /** -- To unsubscribe from this list: send the line "unsubscribe netdev" in the body of a message to majord...@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
Re: [PATCH 13/14] mm: memcontrol: account socket memory in unified hierarchy memory controller
On Thu, Nov 19, 2015 at 02:50:24PM +0100, Michal Hocko wrote: > On Wed 18-11-15 16:48:22, Johannes Weiner wrote: > [...] > > So I ran perf record -g -a netperf -t TCP_STREAM multiple times inside > > a memory-controlled cgroup, but mostly mem_cgroup_charge_skmem() does > > not show up in the profile at all. Once it was there with 0.00%. > > OK, this sounds very good! This means that most workloads which are not > focusing solely on the network traffic shouldn't even notice. I can > imagine that workloads with high throughput demands would notice but I > would also expect them to disable the feature. Even for high throughput, the cost of this is a function of number of packets sent. E.g. the 13MB/s over wifi showed the socket charging at 0.02%. But I just did an http transfer over 1Gbit ethernet at around 110MB/s, ten times the bandwidth, and the charge function is at 0.00%. > Could you add this information to the changelog, please? Sure, but which information exactly? If we had found a realistic networking workload that is expected to be containerized and had shown that load to be negatively affected by the charging calls, that would have been worth bringing up in conjunction with the boot-time flag. But what do we have to say here? People care about cost. It seems unnecessary to point out the absence of it. -- To unsubscribe from this list: send the line "unsubscribe netdev" in the body of a message to majord...@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
Re: [PATCH 13/14] mm: memcontrol: account socket memory in unified hierarchy memory controller
On Wed 18-11-15 16:48:22, Johannes Weiner wrote: [...] > So I ran perf record -g -a netperf -t TCP_STREAM multiple times inside > a memory-controlled cgroup, but mostly mem_cgroup_charge_skmem() does > not show up in the profile at all. Once it was there with 0.00%. OK, this sounds very good! This means that most workloads which are not focusing solely on the network traffic shouldn't even notice. I can imagine that workloads with high throughput demands would notice but I would also expect them to disable the feature. Could you add this information to the changelog, please? > I ran another test that downloads the latest kernel image from > kernel.org at 13MB/s (on my i5 laptop) and it looks like this: > > 0.02% 0.01% irq/44-iwlwifi [kernel.kallsyms] [k] > mem_cgroup_charge_skmem > | > ---mem_cgroup_charge_skmem > __sk_mem_schedule > tcp_try_rmem_schedule > tcp_data_queue > tcp_rcv_established > tcp_v4_do_rcv > tcp_v4_rcv > ip_local_deliver > ip_rcv > __netif_receive_skb_core > __netif_receive_skb > netif_receive_skb_internal > napi_gro_complete > > The runs vary too much for this to be measurable in elapsed time. -- Michal Hocko SUSE Labs -- To unsubscribe from this list: send the line "unsubscribe netdev" in the body of a message to majord...@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
Re: [PATCH 13/14] mm: memcontrol: account socket memory in unified hierarchy memory controller
On Wed, Nov 18, 2015 at 05:22:56PM +0100, Michal Hocko wrote: > On Mon 16-11-15 13:18:10, Johannes Weiner wrote: > > What load would you test and what would be the baseline to compare it > > to? > > It seems like netperf with a stream load running in a memcg with no > limits vs. in root memcg (and no other cgroups) should give at least a > hint about the runtime overhead, no? Comparing root vs. dedicated group generally doesn't make sense since you either need containment or you don't. It makes more sense to test both times inside a memory-controlled cgroup, one with a regular boot, one with cgroup.memory=nosocket. So I ran perf record -g -a netperf -t TCP_STREAM multiple times inside a memory-controlled cgroup, but mostly mem_cgroup_charge_skmem() does not show up in the profile at all. Once it was there with 0.00%. I ran another test that downloads the latest kernel image from kernel.org at 13MB/s (on my i5 laptop) and it looks like this: 0.02% 0.01% irq/44-iwlwifi [kernel.kallsyms] [k] mem_cgroup_charge_skmem | ---mem_cgroup_charge_skmem __sk_mem_schedule tcp_try_rmem_schedule tcp_data_queue tcp_rcv_established tcp_v4_do_rcv tcp_v4_rcv ip_local_deliver ip_rcv __netif_receive_skb_core __netif_receive_skb netif_receive_skb_internal napi_gro_complete The runs vary too much for this to be measurable in elapsed time. -- To unsubscribe from this list: send the line "unsubscribe netdev" in the body of a message to majord...@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
Re: [PATCH 13/14] mm: memcontrol: account socket memory in unified hierarchy memory controller
On Mon 16-11-15 13:18:10, Johannes Weiner wrote: > On Mon, Nov 16, 2015 at 04:59:25PM +0100, Michal Hocko wrote: > > On Thu 12-11-15 18:41:32, Johannes Weiner wrote: > > > Socket memory can be a significant share of overall memory consumed by > > > common workloads. In order to provide reasonable resource isolation in > > > the unified hierarchy, this type of memory needs to be included in the > > > tracking/accounting of a cgroup under active memory resource control. > > > > > > Overhead is only incurred when a non-root control group is created AND > > > the memory controller is instructed to track and account the memory > > > footprint of that group. cgroup.memory=nosocket can be specified on > > > the boot commandline to override any runtime configuration and > > > forcibly exclude socket memory from active memory resource control. > > > > Do you have any numbers about the overhead? > > Hm? Performance numbers make sense when you have a specific scenario > and a theory on how to optimize the implementation for it. The fact that there was a strong push to use static branches to put the code out of line to reduce an overhead before the feature was merged shows that people are sensitive to network performance and that significant effort has been spent to eliminate it. My point was that you are enabling the feature for all memcg users in unified hierarchy now without having a performance impact overview which users can use to judge whether to keep it enabled or disable before they start seeing regressions or to make regression easier to track once it happens. > What load would you test and what would be the baseline to compare it > to? It seems like netperf with a stream load running in a memcg with no limits vs. in root memcg (and no other cgroups) should give at least a hint about the runtime overhead, no? -- Michal Hocko SUSE Labs -- To unsubscribe from this list: send the line "unsubscribe netdev" in the body of a message to majord...@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
Re: [PATCH 13/14] mm: memcontrol: account socket memory in unified hierarchy memory controller
On Mon, Nov 16, 2015 at 04:59:25PM +0100, Michal Hocko wrote: > On Thu 12-11-15 18:41:32, Johannes Weiner wrote: > > Socket memory can be a significant share of overall memory consumed by > > common workloads. In order to provide reasonable resource isolation in > > the unified hierarchy, this type of memory needs to be included in the > > tracking/accounting of a cgroup under active memory resource control. > > > > Overhead is only incurred when a non-root control group is created AND > > the memory controller is instructed to track and account the memory > > footprint of that group. cgroup.memory=nosocket can be specified on > > the boot commandline to override any runtime configuration and > > forcibly exclude socket memory from active memory resource control. > > Do you have any numbers about the overhead? Hm? Performance numbers make sense when you have a specific scenario and a theory on how to optimize the implementation for it. What load would you test and what would be the baseline to compare it to? > > Signed-off-by: Johannes Weiner > > With a way to disable this feature I am OK with it. > cgroup.memory=nosocket should be documented (at least in > Documentation/kernel-parameters.txt) diff --git a/Documentation/kernel-parameters.txt b/Documentation/kernel-parameters.txt index f8aae63..d518340 100644 --- a/Documentation/kernel-parameters.txt +++ b/Documentation/kernel-parameters.txt @@ -599,6 +599,10 @@ bytes respectively. Such letter suffixes can also be entirely omitted. cut the overhead, others just disable the usage. So only cgroup_disable=memory is actually worthy} + cgroup.memory= [KNL] Pass options to the cgroup memory controller. + Format: + nosocket -- Disable socket memory accounting. + checkreqprot[SELINUX] Set initial checkreqprot flag value. Format: { "0" | "1" } See security/selinux/Kconfig help text. > Other than that > Acked-by: Michal Hocko Thanks! --- >From abe0670cec1424a5b4f43dfabff8bb27a1007ced Mon Sep 17 00:00:00 2001 From: Johannes Weiner Date: Wed, 14 Oct 2015 22:25:20 -0700 Subject: [PATCH] mm: memcontrol: account socket memory in unified hierarchy memory controller Socket memory can be a significant share of overall memory consumed by common workloads. In order to provide reasonable resource isolation in the unified hierarchy, this type of memory needs to be included in the tracking/accounting of a cgroup under active memory resource control. Overhead is only incurred when a non-root control group is created AND the memory controller is instructed to track and account the memory footprint of that group. cgroup.memory=nosocket can be specified on the boot commandline to override any runtime configuration and forcibly exclude socket memory from active memory resource control. Signed-off-by: Johannes Weiner Acked-by: Michal Hocko --- Documentation/kernel-parameters.txt | 4 ++ include/linux/memcontrol.h | 12 +++- mm/memcontrol.c | 131 +--- 3 files changed, 122 insertions(+), 25 deletions(-) diff --git a/Documentation/kernel-parameters.txt b/Documentation/kernel-parameters.txt index f8aae63..d518340 100644 --- a/Documentation/kernel-parameters.txt +++ b/Documentation/kernel-parameters.txt @@ -599,6 +599,10 @@ bytes respectively. Such letter suffixes can also be entirely omitted. cut the overhead, others just disable the usage. So only cgroup_disable=memory is actually worthy} + cgroup.memory= [KNL] Pass options to the cgroup memory controller. + Format: + nosocket -- Disable socket memory accounting. + checkreqprot[SELINUX] Set initial checkreqprot flag value. Format: { "0" | "1" } See security/selinux/Kconfig help text. diff --git a/include/linux/memcontrol.h b/include/linux/memcontrol.h index 4cf5afa..809d6de 100644 --- a/include/linux/memcontrol.h +++ b/include/linux/memcontrol.h @@ -256,6 +256,10 @@ struct mem_cgroup { struct wb_domain cgwb_domain; #endif +#ifdef CONFIG_INET + struct work_struct socket_work; +#endif + /* List of events which userspace want to receive */ struct list_head event_list; spinlock_t event_list_lock; @@ -691,7 +695,7 @@ static inline void mem_cgroup_wb_stats(struct bdi_writeback *wb, #endif /* CONFIG_CGROUP_WRITEBACK */ -#if defined(CONFIG_INET) && defined(CONFIG_MEMCG_KMEM) +#ifdef CONFIG_INET struct sock; extern struct static_key memcg_sockets_enabled_key; #define mem_cgroup_sockets_enabled static_key_false(&memcg_sockets_enabled_key) @@ -701,11 +705,15 @@ bool mem_cgroup_charge_skmem(struct mem_cgroup *memcg, unsigned int nr_pages); void mem_cgroup_uncharge_skmem(struct mem_cgroup *
Re: [PATCH 13/14] mm: memcontrol: account socket memory in unified hierarchy memory controller
On Thu 12-11-15 18:41:32, Johannes Weiner wrote: > Socket memory can be a significant share of overall memory consumed by > common workloads. In order to provide reasonable resource isolation in > the unified hierarchy, this type of memory needs to be included in the > tracking/accounting of a cgroup under active memory resource control. > > Overhead is only incurred when a non-root control group is created AND > the memory controller is instructed to track and account the memory > footprint of that group. cgroup.memory=nosocket can be specified on > the boot commandline to override any runtime configuration and > forcibly exclude socket memory from active memory resource control. Do you have any numbers about the overhead? > Signed-off-by: Johannes Weiner With a way to disable this feature I am OK with it. cgroup.memory=nosocket should be documented (at least in Documentation/kernel-parameters.txt) Other than that Acked-by: Michal Hocko > --- > include/linux/memcontrol.h | 12 - > mm/memcontrol.c| 131 > + > 2 files changed, 118 insertions(+), 25 deletions(-) > > diff --git a/include/linux/memcontrol.h b/include/linux/memcontrol.h > index 4cf5afa..809d6de 100644 > --- a/include/linux/memcontrol.h > +++ b/include/linux/memcontrol.h > @@ -256,6 +256,10 @@ struct mem_cgroup { > struct wb_domain cgwb_domain; > #endif > > +#ifdef CONFIG_INET > + struct work_struct socket_work; > +#endif > + > /* List of events which userspace want to receive */ > struct list_head event_list; > spinlock_t event_list_lock; > @@ -691,7 +695,7 @@ static inline void mem_cgroup_wb_stats(struct > bdi_writeback *wb, > > #endif /* CONFIG_CGROUP_WRITEBACK */ > > -#if defined(CONFIG_INET) && defined(CONFIG_MEMCG_KMEM) > +#ifdef CONFIG_INET > struct sock; > extern struct static_key memcg_sockets_enabled_key; > #define mem_cgroup_sockets_enabled > static_key_false(&memcg_sockets_enabled_key) > @@ -701,11 +705,15 @@ bool mem_cgroup_charge_skmem(struct mem_cgroup *memcg, > unsigned int nr_pages); > void mem_cgroup_uncharge_skmem(struct mem_cgroup *memcg, unsigned int > nr_pages); > static inline bool mem_cgroup_under_socket_pressure(struct mem_cgroup *memcg) > { > +#ifdef CONFIG_MEMCG_KMEM > return memcg->tcp_mem.memory_pressure; > +#else > + return false; > +#endif > } > #else > #define mem_cgroup_sockets_enabled 0 > -#endif /* CONFIG_INET && CONFIG_MEMCG_KMEM */ > +#endif /* CONFIG_INET */ > > #ifdef CONFIG_MEMCG_KMEM > extern struct static_key memcg_kmem_enabled_key; > diff --git a/mm/memcontrol.c b/mm/memcontrol.c > index 408fb04..cad9525 100644 > --- a/mm/memcontrol.c > +++ b/mm/memcontrol.c > @@ -80,6 +80,9 @@ struct mem_cgroup *root_mem_cgroup __read_mostly; > > #define MEM_CGROUP_RECLAIM_RETRIES 5 > > +/* Socket memory accounting disabled? */ > +static bool cgroup_memory_nosocket; > + > /* Whether the swap controller is active */ > #ifdef CONFIG_MEMCG_SWAP > int do_swap_account __read_mostly; > @@ -1923,6 +1926,18 @@ static int memcg_cpu_hotplug_callback(struct > notifier_block *nb, > return NOTIFY_OK; > } > > +static void reclaim_high(struct mem_cgroup *memcg, > + unsigned int nr_pages, > + gfp_t gfp_mask) > +{ > + do { > + if (page_counter_read(&memcg->memory) <= memcg->high) > + continue; > + mem_cgroup_events(memcg, MEMCG_HIGH, 1); > + try_to_free_mem_cgroup_pages(memcg, nr_pages, gfp_mask, true); > + } while ((memcg = parent_mem_cgroup(memcg))); > +} > + > /* > * Scheduled by try_charge() to be executed from the userland return path > * and reclaims memory over the high limit. > @@ -1930,20 +1945,13 @@ static int memcg_cpu_hotplug_callback(struct > notifier_block *nb, > void mem_cgroup_handle_over_high(void) > { > unsigned int nr_pages = current->memcg_nr_pages_over_high; > - struct mem_cgroup *memcg, *pos; > + struct mem_cgroup *memcg; > > if (likely(!nr_pages)) > return; > > - pos = memcg = get_mem_cgroup_from_mm(current->mm); > - > - do { > - if (page_counter_read(&pos->memory) <= pos->high) > - continue; > - mem_cgroup_events(pos, MEMCG_HIGH, 1); > - try_to_free_mem_cgroup_pages(pos, nr_pages, GFP_KERNEL, true); > - } while ((pos = parent_mem_cgroup(pos))); > - > + memcg = get_mem_cgroup_from_mm(current->mm); > + reclaim_high(memcg, nr_pages, GFP_KERNEL); > css_put(&memcg->css); > current->memcg_nr_pages_over_high = 0; > } > @@ -4141,6 +4149,8 @@ struct mem_cgroup *parent_mem_cgroup(struct mem_cgroup > *memcg) > } > EXPORT_SYMBOL(parent_mem_cgroup); > > +static void socket_work_func(struct work_struct *work); > + > static struct cgroup_subsys_state * __ref > mem_cgroup_css_alloc(struct cgroup_subsys_state *parent_css)
[PATCH 13/14] mm: memcontrol: account socket memory in unified hierarchy memory controller
Socket memory can be a significant share of overall memory consumed by common workloads. In order to provide reasonable resource isolation in the unified hierarchy, this type of memory needs to be included in the tracking/accounting of a cgroup under active memory resource control. Overhead is only incurred when a non-root control group is created AND the memory controller is instructed to track and account the memory footprint of that group. cgroup.memory=nosocket can be specified on the boot commandline to override any runtime configuration and forcibly exclude socket memory from active memory resource control. Signed-off-by: Johannes Weiner --- include/linux/memcontrol.h | 12 - mm/memcontrol.c| 131 + 2 files changed, 118 insertions(+), 25 deletions(-) diff --git a/include/linux/memcontrol.h b/include/linux/memcontrol.h index 4cf5afa..809d6de 100644 --- a/include/linux/memcontrol.h +++ b/include/linux/memcontrol.h @@ -256,6 +256,10 @@ struct mem_cgroup { struct wb_domain cgwb_domain; #endif +#ifdef CONFIG_INET + struct work_struct socket_work; +#endif + /* List of events which userspace want to receive */ struct list_head event_list; spinlock_t event_list_lock; @@ -691,7 +695,7 @@ static inline void mem_cgroup_wb_stats(struct bdi_writeback *wb, #endif /* CONFIG_CGROUP_WRITEBACK */ -#if defined(CONFIG_INET) && defined(CONFIG_MEMCG_KMEM) +#ifdef CONFIG_INET struct sock; extern struct static_key memcg_sockets_enabled_key; #define mem_cgroup_sockets_enabled static_key_false(&memcg_sockets_enabled_key) @@ -701,11 +705,15 @@ bool mem_cgroup_charge_skmem(struct mem_cgroup *memcg, unsigned int nr_pages); void mem_cgroup_uncharge_skmem(struct mem_cgroup *memcg, unsigned int nr_pages); static inline bool mem_cgroup_under_socket_pressure(struct mem_cgroup *memcg) { +#ifdef CONFIG_MEMCG_KMEM return memcg->tcp_mem.memory_pressure; +#else + return false; +#endif } #else #define mem_cgroup_sockets_enabled 0 -#endif /* CONFIG_INET && CONFIG_MEMCG_KMEM */ +#endif /* CONFIG_INET */ #ifdef CONFIG_MEMCG_KMEM extern struct static_key memcg_kmem_enabled_key; diff --git a/mm/memcontrol.c b/mm/memcontrol.c index 408fb04..cad9525 100644 --- a/mm/memcontrol.c +++ b/mm/memcontrol.c @@ -80,6 +80,9 @@ struct mem_cgroup *root_mem_cgroup __read_mostly; #define MEM_CGROUP_RECLAIM_RETRIES 5 +/* Socket memory accounting disabled? */ +static bool cgroup_memory_nosocket; + /* Whether the swap controller is active */ #ifdef CONFIG_MEMCG_SWAP int do_swap_account __read_mostly; @@ -1923,6 +1926,18 @@ static int memcg_cpu_hotplug_callback(struct notifier_block *nb, return NOTIFY_OK; } +static void reclaim_high(struct mem_cgroup *memcg, +unsigned int nr_pages, +gfp_t gfp_mask) +{ + do { + if (page_counter_read(&memcg->memory) <= memcg->high) + continue; + mem_cgroup_events(memcg, MEMCG_HIGH, 1); + try_to_free_mem_cgroup_pages(memcg, nr_pages, gfp_mask, true); + } while ((memcg = parent_mem_cgroup(memcg))); +} + /* * Scheduled by try_charge() to be executed from the userland return path * and reclaims memory over the high limit. @@ -1930,20 +1945,13 @@ static int memcg_cpu_hotplug_callback(struct notifier_block *nb, void mem_cgroup_handle_over_high(void) { unsigned int nr_pages = current->memcg_nr_pages_over_high; - struct mem_cgroup *memcg, *pos; + struct mem_cgroup *memcg; if (likely(!nr_pages)) return; - pos = memcg = get_mem_cgroup_from_mm(current->mm); - - do { - if (page_counter_read(&pos->memory) <= pos->high) - continue; - mem_cgroup_events(pos, MEMCG_HIGH, 1); - try_to_free_mem_cgroup_pages(pos, nr_pages, GFP_KERNEL, true); - } while ((pos = parent_mem_cgroup(pos))); - + memcg = get_mem_cgroup_from_mm(current->mm); + reclaim_high(memcg, nr_pages, GFP_KERNEL); css_put(&memcg->css); current->memcg_nr_pages_over_high = 0; } @@ -4141,6 +4149,8 @@ struct mem_cgroup *parent_mem_cgroup(struct mem_cgroup *memcg) } EXPORT_SYMBOL(parent_mem_cgroup); +static void socket_work_func(struct work_struct *work); + static struct cgroup_subsys_state * __ref mem_cgroup_css_alloc(struct cgroup_subsys_state *parent_css) { @@ -4180,6 +4190,9 @@ mem_cgroup_css_alloc(struct cgroup_subsys_state *parent_css) #ifdef CONFIG_CGROUP_WRITEBACK INIT_LIST_HEAD(&memcg->cgwb_list); #endif +#ifdef CONFIG_INET + INIT_WORK(&memcg->socket_work, socket_work_func); +#endif return &memcg->css; free_out: @@ -4237,6 +4250,11 @@ mem_cgroup_css_online(struct cgroup_subsys_state *css) if (ret) return ret; +#ifdef CONFIG_INET + if (cgroup_subsys_on_dfl(memory_cgr