Received: by 2002:a25:8b91:0:0:0:0:0 with SMTP id j17csp826990ybl; Fri, 24 Jan 2020 10:13:22 -0800 (PST) X-Google-Smtp-Source: APXvYqx1tc8j0Yv7D5TOfWdUpH2eWc2USg5G4HDTXORtiKDbfnzyfSNIK07d74BAlnPDkiM54hGA X-Received: by 2002:a9d:68d1:: with SMTP id i17mr3495933oto.367.1579889601893; Fri, 24 Jan 2020 10:13:21 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1579889601; cv=none; d=google.com; s=arc-20160816; b=qBpy1yTF0Pl3MnY0ljNC831YZZLxSWXGuJvYG+VvgBj7iDxthWA1EI/9NHfakUm2wE VoNPjK3TjzYjzDce/KwRiltqNI7QJ5u5Au5SAvd/03AOnOYR8axtwayxIt5TYkdhsfKW t2tDsiXZUudWXyE7nJArEapKM6hGVkhdany/f9xAB2rfGbbAYpi//3cvcgiA9j25+f2/ I4lqUO2yutljJ3MkZLqfDSrp+04PYhSoHSlf5Q7Mx3NDV8GenFrkd36j+mM0xdO9Z8SR w+r4NXmpWDmzk8TwFp2eJpKZGSKBaUlDBA4Bbmsay8eMDxphisSJNEBwSt2anKOThtwd +8rA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :user-agent:references:in-reply-to:message-id:date:subject:cc:to :from:dkim-signature; bh=Xk6Fzjxeb8qNQ4I5NHSDzIp9cEozvCWU5J6CTIStlNE=; b=juxe3y/yvdDhiE7RNUuti4mEGXhGG4fryVOcSfIHvbkKUj+/xhGJcJ0zkL5AjRwk5H gkJcKL4zqyVIkHrQzH/Z+kWjimfiZAbE/YxL1ceOJovH6kFKd61zzK8uMWu0Al1VJPrX Q5Ks0V96yToHgKwzDIF/fpc2ULX37yfL2N5fMTPurDc96LdMTD78ITGQ9UVf2a03puKh gFtprhFmyQu5YR+OBGZVGP+nYFqiXeOgNXJh7eWhESKeEUNzM+Zi9kn6Ze247bzrEvcR KTUZslUjkWHaQbjW1Lin09zolBhTgutcSCQKtfqcdWfT9hnBJT0+SWDcHDyVgCwbVoJO nYQQ== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b=GHW3U4K1; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id h9si3101691otb.49.2020.01.24.10.13.09; Fri, 24 Jan 2020 10:13:21 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b=GHW3U4K1; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S2390881AbgAXLVi (ORCPT + 99 others); Fri, 24 Jan 2020 06:21:38 -0500 Received: from mail.kernel.org ([198.145.29.99]:60034 "EHLO mail.kernel.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S2390482AbgAXLVg (ORCPT ); Fri, 24 Jan 2020 06:21:36 -0500 Received: from localhost (ip-213-127-102-57.ip.prioritytelecom.net [213.127.102.57]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by mail.kernel.org (Postfix) with ESMTPSA id 22EC7206D4; Fri, 24 Jan 2020 11:21:34 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=kernel.org; s=default; t=1579864895; bh=3Ex/mXexqXxXMMGo1YyKzygo6nVNMvVs8fQmSKw30Sw=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=GHW3U4K1LFpIWsNDUFrckIom88W8l9FWdYvu8wYXCuhGInVO/sTJJ+TitpEYQuERF 4Xl3tDdeNTbCRLEvt3WDBVVbEkKZseKoqXN403cQ173oUsIPRube/UBbF6D3C1aqlz y41L4E56RVyFxlAge2YKxkzOZzECid/v0c8xwBTs= From: Greg Kroah-Hartman To: linux-kernel@vger.kernel.org Cc: Greg Kroah-Hartman , stable@vger.kernel.org, "Rafael J. Wysocki" , Keith Busch , Mika Westerberg , Sasha Levin Subject: [PATCH 4.19 395/639] PCI: PM: Avoid possible suspend-to-idle issue Date: Fri, 24 Jan 2020 10:29:25 +0100 Message-Id: <20200124093136.400482106@linuxfoundation.org> X-Mailer: git-send-email 2.25.0 In-Reply-To: <20200124093047.008739095@linuxfoundation.org> References: <20200124093047.008739095@linuxfoundation.org> User-Agent: quilt/0.66 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: Rafael J. Wysocki [ Upstream commit d491f2b75237ef37d8867830ab7fad8d9659e853 ] If a PCI driver leaves the device handled by it in D0 and calls pci_save_state() on the device in its ->suspend() or ->suspend_late() callback, it can expect the device to stay in D0 over the whole s2idle cycle. However, that may not be the case if there is a spurious wakeup while the system is suspended, because in that case pci_pm_suspend_noirq() will run again after pci_pm_resume_noirq() which calls pci_restore_state(), via pci_pm_default_resume_early(), so state_saved is cleared and the second iteration of pci_pm_suspend_noirq() will invoke pci_prepare_to_sleep() which may change the power state of the device. To avoid that, add a new internal flag, skip_bus_pm, that will be set by pci_pm_suspend_noirq() when it runs for the first time during the given system suspend-resume cycle if the state of the device has been saved already and the device is still in D0. Setting that flag will cause the next iterations of pci_pm_suspend_noirq() to set state_saved for pci_pm_resume_noirq(), so that it always restores the device state from the originally saved data, and avoid calling pci_prepare_to_sleep() for the device. Fixes: 33e4f80ee69b ("ACPI / PM: Ignore spurious SCI wakeups from suspend-to-idle") Signed-off-by: Rafael J. Wysocki Reviewed-by: Keith Busch Reviewed-by: Mika Westerberg Signed-off-by: Sasha Levin --- drivers/pci/pci-driver.c | 17 ++++++++++++++++- include/linux/pci.h | 1 + 2 files changed, 17 insertions(+), 1 deletion(-) diff --git a/drivers/pci/pci-driver.c b/drivers/pci/pci-driver.c index bc1ff41ce3d35..5c9873fcbd08b 100644 --- a/drivers/pci/pci-driver.c +++ b/drivers/pci/pci-driver.c @@ -736,6 +736,8 @@ static int pci_pm_suspend(struct device *dev) struct pci_dev *pci_dev = to_pci_dev(dev); const struct dev_pm_ops *pm = dev->driver ? dev->driver->pm : NULL; + pci_dev->skip_bus_pm = false; + if (pci_has_legacy_pm_support(pci_dev)) return pci_legacy_suspend(dev, PMSG_SUSPEND); @@ -829,7 +831,20 @@ static int pci_pm_suspend_noirq(struct device *dev) } } - if (!pci_dev->state_saved) { + if (pci_dev->skip_bus_pm) { + /* + * The function is running for the second time in a row without + * going through full resume, which is possible only during + * suspend-to-idle in a spurious wakeup case. Moreover, the + * device was originally left in D0, so its power state should + * not be changed here and the device register values saved + * originally should be restored on resume again. + */ + pci_dev->state_saved = true; + } else if (pci_dev->state_saved) { + if (pci_dev->current_state == PCI_D0) + pci_dev->skip_bus_pm = true; + } else { pci_save_state(pci_dev); if (pci_power_manageable(pci_dev)) pci_prepare_to_sleep(pci_dev); diff --git a/include/linux/pci.h b/include/linux/pci.h index b1f297f4b7b0b..94853094b6ef4 100644 --- a/include/linux/pci.h +++ b/include/linux/pci.h @@ -342,6 +342,7 @@ struct pci_dev { D3cold, not set for devices powered on/off by the corresponding bridge */ + unsigned int skip_bus_pm:1; /* Internal: Skip bus-level PM */ unsigned int ignore_hotplug:1; /* Ignore hotplug events */ unsigned int hotplug_user_indicators:1; /* SlotCtl indicators controlled exclusively by -- 2.20.1