Received: by 2002:a05:6a10:1287:0:0:0:0 with SMTP id d7csp632520pxv; Thu, 15 Jul 2021 12:02:07 -0700 (PDT) X-Google-Smtp-Source: ABdhPJy/lDdVSj9H5ZnD+UE2KEVo/ufPU9/xDicymSXItLWZABlNT/6KG5Lx+qlchmnpxHvzyuWx X-Received: by 2002:a02:8521:: with SMTP id g30mr5289752jai.113.1626375727326; Thu, 15 Jul 2021 12:02:07 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1626375727; cv=none; d=google.com; s=arc-20160816; b=SlwZWTu8Rtgnfx+0y825MD4zXS/VBw04wGvTUzN6Rg9HQLfggoEnMC+CMg1UDAyfnP sPQnmxGMD2AA4pcF4anC1QkisLope+1jeuJzIgwGx2DK6qaNCFb6KFr8upk4Vi5j3r6r r/gdksDqIyp/tf7gPl0g4n7h+Dv811QZzAiYGN9iLpzfF4NMpDRXq5Y27q0LKOhZhwDv x4V20moWvNUDHuyKpHzoAR/+WR2CI+646N0FrOBFzyiuO1t/aJZsFDpdMgQa7cNkW2ro MMTndjX/KDMXkPgopp9Px2/xwqym4rUvxkuIkW8+9+qh3+8ZxNRtxoB9ZEV063cBver8 zCXw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:mime-version :user-agent:references:in-reply-to:message-id:date:subject:cc:to :from:dkim-signature; bh=xhoykkyi3IIOIPIEamQyWOHkrljmxjjm4GFnBjb7cX8=; b=nFCIs1l8Gbna25uF1gfqo8wE1P9fZhBKZi0uIS2FQgdOCzjsEpcAf6BnuZ6738A1Ms Ua1jEGE6SBqFWyqwMwwYlfZimwvVCckuxR9n54B67SCXl7GX80nDIphBCxvdo1W3BKTu DTpcpiMojfJ41Jr1ji5FRqEKKqwSud6KNJTQnUK7TGwi+/BMeaKynmeRKKPKq95ipJwZ xqLh6YRXpiOJLETMc3Q3kLa5xYKWPpv3F7kHlAPjaCrbggwJLVQZKky1tyRTjmK9J+x3 x10Yw3/ATxdE6UiTUnMAtYE/oDRqIUyXoeE3BDR+yB6oOU4+IRpcaC2eS4MPgIzMTA8g lxoA== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@linuxfoundation.org header.s=korg header.b=aeQaizZZ; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=linuxfoundation.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id d1si5509523jak.77.2021.07.15.12.01.53; Thu, 15 Jul 2021 12:02:07 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; dkim=pass header.i=@linuxfoundation.org header.s=korg header.b=aeQaizZZ; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=linuxfoundation.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S242043AbhGOTDZ (ORCPT + 99 others); Thu, 15 Jul 2021 15:03:25 -0400 Received: from mail.kernel.org ([198.145.29.99]:59242 "EHLO mail.kernel.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S242345AbhGOSzD (ORCPT ); Thu, 15 Jul 2021 14:55:03 -0400 Received: by mail.kernel.org (Postfix) with ESMTPSA id BD46C613C4; Thu, 15 Jul 2021 18:52:09 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=linuxfoundation.org; s=korg; t=1626375130; bh=z5VAaCN//t6D1rGf7JEjGoZ5GirbsmHOsqPiy8wjwQU=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=aeQaizZZ5sY1GK4SUNaeZI3QRsBODDPGpScB4VDdLScexkQPBXr1Wt9ooDID9sUXj 9pYsEjSpU7kjaoNm3clRC/h8Bc/+d2R6J69Hy/RwSbZaa2/Be8iKz6ADq6ggFIEsKc Nug5LBJNd88axScykFfSQ9r6x+t2rg6EJZMfSsX4= From: Greg Kroah-Hartman To: linux-kernel@vger.kernel.org Cc: Greg Kroah-Hartman , stable@vger.kernel.org, Alexey Klimov , Joshua Baker , Thomas Gleixner Subject: [PATCH 5.10 172/215] cpu/hotplug: Cure the cpusets trainwreck Date: Thu, 15 Jul 2021 20:39:04 +0200 Message-Id: <20210715182629.937643721@linuxfoundation.org> X-Mailer: git-send-email 2.32.0 In-Reply-To: <20210715182558.381078833@linuxfoundation.org> References: <20210715182558.381078833@linuxfoundation.org> User-Agent: quilt/0.66 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: Thomas Gleixner commit b22afcdf04c96ca58327784e280e10288cfd3303 upstream. Alexey and Joshua tried to solve a cpusets related hotplug problem which is user space visible and results in unexpected behaviour for some time after a CPU has been plugged in and the corresponding uevent was delivered. cpusets delegate the hotplug work (rebuilding cpumasks etc.) to a workqueue. This is done because the cpusets code has already a lock nesting of cgroups_mutex -> cpu_hotplug_lock. A synchronous callback or waiting for the work to finish with cpu_hotplug_lock held can and will deadlock because that results in the reverse lock order. As a consequence the uevent can be delivered before cpusets have consistent state which means that a user space invocation of sched_setaffinity() to move a task to the plugged CPU fails up to the point where the scheduled work has been processed. The same is true for CPU unplug, but that does not create user observable failure (yet). It's still inconsistent to claim that an operation is finished before it actually is and that's the real issue at hand. uevents just make it reliably observable. Obviously the problem should be fixed in cpusets/cgroups, but untangling that is pretty much impossible because according to the changelog of the commit which introduced this 8 years ago: 3a5a6d0c2b03("cpuset: don't nest cgroup_mutex inside get_online_cpus()") the lock order cgroups_mutex -> cpu_hotplug_lock is a design decision and the whole code is built around that. So bite the bullet and invoke the relevant cpuset function, which waits for the work to finish, in _cpu_up/down() after dropping cpu_hotplug_lock and only when tasks are not frozen by suspend/hibernate because that would obviously wait forever. Waiting there with cpu_add_remove_lock, which is protecting the present and possible CPU maps, held is not a problem at all because neither work queues nor cpusets/cgroups have any lockchains related to that lock. Waiting in the hotplug machinery is not problematic either because there are already state callbacks which wait for hardware queues to drain. It makes the operations slightly slower, but hotplug is slow anyway. This ensures that state is consistent before returning from a hotplug up/down operation. It's still inconsistent during the operation, but that's a different story. Add a large comment which explains why this is done and why this is not a dump ground for the hack of the day to work around half thought out locking schemes. Document also the implications vs. hotplug operations and serialization or the lack of it. Thanks to Alexy and Joshua for analyzing why this temporary sched_setaffinity() failure happened. Fixes: 3a5a6d0c2b03("cpuset: don't nest cgroup_mutex inside get_online_cpus()") Reported-by: Alexey Klimov Reported-by: Joshua Baker Signed-off-by: Thomas Gleixner Tested-by: Alexey Klimov Cc: stable@vger.kernel.org Link: https://lore.kernel.org/r/87tuowcnv3.ffs@nanos.tec.linutronix.de Signed-off-by: Greg Kroah-Hartman --- kernel/cpu.c | 49 +++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 49 insertions(+) --- a/kernel/cpu.c +++ b/kernel/cpu.c @@ -32,6 +32,7 @@ #include #include #include +#include #include #define CREATE_TRACE_POINTS @@ -814,6 +815,52 @@ void __init cpuhp_threads_init(void) kthread_unpark(this_cpu_read(cpuhp_state.thread)); } +/* + * + * Serialize hotplug trainwrecks outside of the cpu_hotplug_lock + * protected region. + * + * The operation is still serialized against concurrent CPU hotplug via + * cpu_add_remove_lock, i.e. CPU map protection. But it is _not_ + * serialized against other hotplug related activity like adding or + * removing of state callbacks and state instances, which invoke either the + * startup or the teardown callback of the affected state. + * + * This is required for subsystems which are unfixable vs. CPU hotplug and + * evade lock inversion problems by scheduling work which has to be + * completed _before_ cpu_up()/_cpu_down() returns. + * + * Don't even think about adding anything to this for any new code or even + * drivers. It's only purpose is to keep existing lock order trainwrecks + * working. + * + * For cpu_down() there might be valid reasons to finish cleanups which are + * not required to be done under cpu_hotplug_lock, but that's a different + * story and would be not invoked via this. + */ +static void cpu_up_down_serialize_trainwrecks(bool tasks_frozen) +{ + /* + * cpusets delegate hotplug operations to a worker to "solve" the + * lock order problems. Wait for the worker, but only if tasks are + * _not_ frozen (suspend, hibernate) as that would wait forever. + * + * The wait is required because otherwise the hotplug operation + * returns with inconsistent state, which could even be observed in + * user space when a new CPU is brought up. The CPU plug uevent + * would be delivered and user space reacting on it would fail to + * move tasks to the newly plugged CPU up to the point where the + * work has finished because up to that point the newly plugged CPU + * is not assignable in cpusets/cgroups. On unplug that's not + * necessarily a visible issue, but it is still inconsistent state, + * which is the real problem which needs to be "fixed". This can't + * prevent the transient state between scheduling the work and + * returning from waiting for it. + */ + if (!tasks_frozen) + cpuset_wait_for_hotplug(); +} + #ifdef CONFIG_HOTPLUG_CPU #ifndef arch_clear_mm_cpumask_cpu #define arch_clear_mm_cpumask_cpu(cpu, mm) cpumask_clear_cpu(cpu, mm_cpumask(mm)) @@ -1051,6 +1098,7 @@ out: */ lockup_detector_cleanup(); arch_smt_update(); + cpu_up_down_serialize_trainwrecks(tasks_frozen); return ret; } @@ -1247,6 +1295,7 @@ static int _cpu_up(unsigned int cpu, int out: cpus_write_unlock(); arch_smt_update(); + cpu_up_down_serialize_trainwrecks(tasks_frozen); return ret; }