Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id 560DEC64EC4 for ; Thu, 9 Mar 2023 17:14:36 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S230384AbjCIROf (ORCPT ); Thu, 9 Mar 2023 12:14:35 -0500 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:51004 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S231182AbjCIRN4 (ORCPT ); Thu, 9 Mar 2023 12:13:56 -0500 Received: from smtp-out1.suse.de (smtp-out1.suse.de [IPv6:2001:67c:2178:6::1c]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 4864DF2894 for ; Thu, 9 Mar 2023 09:11:33 -0800 (PST) Received: from imap2.suse-dmz.suse.de (imap2.suse-dmz.suse.de [192.168.254.74]) (using TLSv1.3 with cipher TLS_AES_256_GCM_SHA384 (256/256 bits) key-exchange X25519 server-signature ECDSA (P-521) server-digest SHA512) (No client certificate requested) by smtp-out1.suse.de (Postfix) with ESMTPS id 8ED7622201; Thu, 9 Mar 2023 17:11:30 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=suse.cz; s=susede2_rsa; t=1678381890; h=from:from:reply-to:date:date:message-id:message-id:to:to:cc:cc: mime-version:mime-version:content-type:content-type: content-transfer-encoding:content-transfer-encoding: in-reply-to:in-reply-to:references:references; bh=cLLOvBxBvwKab8OGnBJMO2dLGmXmHB/7/d82oXim7LE=; b=AHUIm1cPkSBtprWnNAUzuuCpUq7ESTd4sIbN8ON/LfmkkrAUhQRWz0/iKqAxCLue2QX+7f YROTSBqvLUHceoM8L8e5w/IU9vT68xdDyf0Qa+tGAgd4o3oDVVa9LQUejL3EhsQzTIxNCx uXTpMZiMVDz1LHQcB2xnsDPt3+BkL74= DKIM-Signature: v=1; a=ed25519-sha256; c=relaxed/relaxed; d=suse.cz; s=susede2_ed25519; t=1678381890; h=from:from:reply-to:date:date:message-id:message-id:to:to:cc:cc: mime-version:mime-version:content-type:content-type: content-transfer-encoding:content-transfer-encoding: in-reply-to:in-reply-to:references:references; bh=cLLOvBxBvwKab8OGnBJMO2dLGmXmHB/7/d82oXim7LE=; b=DdcQfPeqLKmom03PAiXedlUk8GQD/+v5Wr2OecU/4M4XaflJUTxlTf+JSkDjNroS/hKrf3 OYExQXrXSa/rStAQ== Received: from imap2.suse-dmz.suse.de (imap2.suse-dmz.suse.de [192.168.254.74]) (using TLSv1.3 with cipher TLS_AES_256_GCM_SHA384 (256/256 bits) key-exchange X25519 server-signature ECDSA (P-521) server-digest SHA512) (No client certificate requested) by imap2.suse-dmz.suse.de (Postfix) with ESMTPS id 5B48513A10; Thu, 9 Mar 2023 17:11:30 +0000 (UTC) Received: from dovecot-director2.suse.de ([192.168.254.65]) by imap2.suse-dmz.suse.de with ESMTPSA id 7RGLFUITCmRpEwAAMHmgww (envelope-from ); Thu, 09 Mar 2023 17:11:30 +0000 Message-ID: <015212c4-8e3b-253a-f307-56a3e331060e@suse.cz> Date: Thu, 9 Mar 2023 18:11:30 +0100 MIME-Version: 1.0 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:102.0) Gecko/20100101 Thunderbird/102.8.0 Subject: Re: [PATCH v4 01/12] mm/vmstat: remove remote node draining Content-Language: en-US To: Marcelo Tosatti , Christoph Lameter , Mel Gorman Cc: Aaron Tomlin , Frederic Weisbecker , Andrew Morton , linux-kernel@vger.kernel.org, linux-mm@kvack.org, Russell King , Huacai Chen , Heiko Carstens , x86@kernel.org, David Hildenbrand References: <20230305133657.255737580@redhat.com> <20230305134053.436759976@redhat.com> From: Vlastimil Babka In-Reply-To: <20230305134053.436759976@redhat.com> Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 3/5/23 14:36, Marcelo Tosatti wrote: > Draining of pages from the local pcp for a remote zone should not be > necessary, since once the system is low on memory (or compaction on a > zone is in effect), drain_all_pages should be called freeing any unused > pcps. Hmm I can't easily say this is a good idea without a proper evaluation. It's true that drain_all_pages() will be called from __alloc_pages_direct_reclaim(), but if you look closely, it's only after __perform_reclaim() and subsequent get_page_from_freelist() failure, which means very low on memory. There's also kcompactd_do_work() caller, but that shouldn't respond to order-0 pressure. Maybe part of the problem is that pages sitting in pcplist are not counted as NR_FREE_PAGES, so watermark checks will not "see" them as available. That would be true for both local and remote nodes as we process the zonelist, but if pages are stuck on remote pcplists, it could make the problem worse than if they stayed only in local ones. But the good news should be that for this series you shouldn't actually need this patch, AFAIU. Last year Mel did the "Drain remote per-cpu directly" series and followups, so remote draining of pcplists is today done without sending an IPI to the remote CPU. BTW I kinda like the implementation that relies on "per-cpu spin lock" that's used mostly locally thus uncontended, and only rarely remotely. Much simpler than complicated cmpxchg schemes, wonder if those are really still such a win these days, e.g. in SLUB or here in vmstat... > For reference, the original commit which introduces remote node > draining is 4037d452202e34214e8a939fa5621b2b3bbb45b7. > > Acked-by: David Hildenbrand > Signed-off-by: Marcelo Tosatti > > Index: linux-vmstat-remote/include/linux/mmzone.h > =================================================================== > --- linux-vmstat-remote.orig/include/linux/mmzone.h > +++ linux-vmstat-remote/include/linux/mmzone.h > @@ -679,9 +679,6 @@ struct per_cpu_pages { > int high; /* high watermark, emptying needed */ > int batch; /* chunk size for buddy add/remove */ > short free_factor; /* batch scaling factor during free */ > -#ifdef CONFIG_NUMA > - short expire; /* When 0, remote pagesets are drained */ > -#endif > > /* Lists of pages, one per migrate type stored on the pcp-lists */ > struct list_head lists[NR_PCP_LISTS]; > Index: linux-vmstat-remote/mm/vmstat.c > =================================================================== > --- linux-vmstat-remote.orig/mm/vmstat.c > +++ linux-vmstat-remote/mm/vmstat.c > @@ -803,20 +803,16 @@ static int fold_diff(int *zone_diff, int > * > * The function returns the number of global counters updated. > */ > -static int refresh_cpu_vm_stats(bool do_pagesets) > +static int refresh_cpu_vm_stats(void) > { > struct pglist_data *pgdat; > struct zone *zone; > int i; > int global_zone_diff[NR_VM_ZONE_STAT_ITEMS] = { 0, }; > int global_node_diff[NR_VM_NODE_STAT_ITEMS] = { 0, }; > - int changes = 0; > > for_each_populated_zone(zone) { > struct per_cpu_zonestat __percpu *pzstats = zone->per_cpu_zonestats; > -#ifdef CONFIG_NUMA > - struct per_cpu_pages __percpu *pcp = zone->per_cpu_pageset; > -#endif > > for (i = 0; i < NR_VM_ZONE_STAT_ITEMS; i++) { > int v; > @@ -826,44 +822,8 @@ static int refresh_cpu_vm_stats(bool do_ > > atomic_long_add(v, &zone->vm_stat[i]); > global_zone_diff[i] += v; > -#ifdef CONFIG_NUMA > - /* 3 seconds idle till flush */ > - __this_cpu_write(pcp->expire, 3); > -#endif > } > } > -#ifdef CONFIG_NUMA > - > - if (do_pagesets) { > - cond_resched(); > - /* > - * Deal with draining the remote pageset of this > - * processor > - * > - * Check if there are pages remaining in this pageset > - * if not then there is nothing to expire. > - */ > - if (!__this_cpu_read(pcp->expire) || > - !__this_cpu_read(pcp->count)) > - continue; > - > - /* > - * We never drain zones local to this processor. > - */ > - if (zone_to_nid(zone) == numa_node_id()) { > - __this_cpu_write(pcp->expire, 0); > - continue; > - } > - > - if (__this_cpu_dec_return(pcp->expire)) > - continue; > - > - if (__this_cpu_read(pcp->count)) { > - drain_zone_pages(zone, this_cpu_ptr(pcp)); > - changes++; > - } > - } > -#endif > } > > for_each_online_pgdat(pgdat) { > @@ -880,8 +840,7 @@ static int refresh_cpu_vm_stats(bool do_ > } > } > > - changes += fold_diff(global_zone_diff, global_node_diff); > - return changes; > + return fold_diff(global_zone_diff, global_node_diff); > } > > /* > @@ -1867,7 +1826,7 @@ int sysctl_stat_interval __read_mostly = > #ifdef CONFIG_PROC_FS > static void refresh_vm_stats(struct work_struct *work) > { > - refresh_cpu_vm_stats(true); > + refresh_cpu_vm_stats(); > } > > int vmstat_refresh(struct ctl_table *table, int write, > @@ -1877,6 +1836,8 @@ int vmstat_refresh(struct ctl_table *tab > int err; > int i; > > + drain_all_pages(NULL); > + > /* > * The regular update, every sysctl_stat_interval, may come later > * than expected: leaving a significant amount in per_cpu buckets. > @@ -1931,7 +1892,7 @@ int vmstat_refresh(struct ctl_table *tab > > static void vmstat_update(struct work_struct *w) > { > - if (refresh_cpu_vm_stats(true)) { > + if (refresh_cpu_vm_stats()) { > /* > * Counters were updated so we expect more updates > * to occur in the future. Keep on running the > @@ -1994,7 +1955,7 @@ void quiet_vmstat(void) > * it would be too expensive from this path. > * vmstat_shepherd will take care about that for us. > */ > - refresh_cpu_vm_stats(false); > + refresh_cpu_vm_stats(); > } > > /* > Index: linux-vmstat-remote/mm/page_alloc.c > =================================================================== > --- linux-vmstat-remote.orig/mm/page_alloc.c > +++ linux-vmstat-remote/mm/page_alloc.c > @@ -3176,26 +3176,6 @@ static int rmqueue_bulk(struct zone *zon > return allocated; > } > > -#ifdef CONFIG_NUMA > -/* > - * Called from the vmstat counter updater to drain pagesets of this > - * currently executing processor on remote nodes after they have > - * expired. > - */ > -void drain_zone_pages(struct zone *zone, struct per_cpu_pages *pcp) > -{ > - int to_drain, batch; > - > - batch = READ_ONCE(pcp->batch); > - to_drain = min(pcp->count, batch); > - if (to_drain > 0) { > - spin_lock(&pcp->lock); > - free_pcppages_bulk(zone, to_drain, pcp, 0); > - spin_unlock(&pcp->lock); > - } > -} > -#endif > - > /* > * Drain pcplists of the indicated processor and zone. > */ > > >