Message ID | e60287ebdb0ab54c4075071b72568a40a75d0205.1736372610.git.mail@maciej.szmigiero.name (mailing list archive) |
---|---|
State | Handled Elsewhere, archived |
Headers | show |
Series | [v3] net: wwan: iosm: Fix hibernation by re-binding the driver around it | expand |
On 1/9/25 12:33 AM, Maciej S. Szmigiero wrote: @@ -530,3 +531,56 @@ void ipc_pcie_kfree_skb(struct iosm_pcie *ipc_pcie, struct sk_buff *skb) > IPC_CB(skb)->mapping = 0; > dev_kfree_skb(skb); > } > + > +static int pm_notify(struct notifier_block *nb, unsigned long mode, void *_unused) > +{ > + if (mode == PM_HIBERNATION_PREPARE || mode == PM_RESTORE_PREPARE) { > + if (pci_registered) { Out of sheer ignorance on my side, why 'mode == PM_RESTORE_PREPARE' is required above? Isn't the driver already unregistered by the previous PM_HIBERNATION_PREPARE call? Thanks, Paolo
On 14.01.2025 09:49, Paolo Abeni wrote: > On 1/9/25 12:33 AM, Maciej S. Szmigiero wrote: > @@ -530,3 +531,56 @@ void ipc_pcie_kfree_skb(struct iosm_pcie > *ipc_pcie, struct sk_buff *skb) >> IPC_CB(skb)->mapping = 0; >> dev_kfree_skb(skb); >> } >> + >> +static int pm_notify(struct notifier_block *nb, unsigned long mode, void *_unused) >> +{ >> + if (mode == PM_HIBERNATION_PREPARE || mode == PM_RESTORE_PREPARE) { >> + if (pci_registered) { > > Out of sheer ignorance on my side, why 'mode == PM_RESTORE_PREPARE' is > required above? Isn't the driver already unregistered by the previous > PM_HIBERNATION_PREPARE call? If the restore kernel had this driver loaded then it needs to be unregistered before restoring the hibernation image so it has chance to shut the modem firmware down rather than keep it running during the restore process. This way when the driver from the restored image re-binds the device it finds it in the proper non-running state. > Thanks, > > Paolo > Thanks, Maciej
Hello: This patch was applied to netdev/net-next.git (main) by Jakub Kicinski <kuba@kernel.org>: On Thu, 9 Jan 2025 00:33:50 +0100 you wrote: > Currently, the driver is seriously broken with respect to the > hibernation (S4): after image restore the device is back into > IPC_MEM_EXEC_STAGE_BOOT (which AFAIK means bootloader stage) and needs > full re-launch of the rest of its firmware, but the driver restore > handler treats the device as merely sleeping and just sends it a > wake-up command. > > [...] Here is the summary with links: - [v3] net: wwan: iosm: Fix hibernation by re-binding the driver around it https://git.kernel.org/netdev/net-next/c/0b6f6593aa8c You are awesome, thank you!
diff --git a/drivers/net/wwan/iosm/iosm_ipc_pcie.c b/drivers/net/wwan/iosm/iosm_ipc_pcie.c index 04517bd3325a..a066977af0be 100644 --- a/drivers/net/wwan/iosm/iosm_ipc_pcie.c +++ b/drivers/net/wwan/iosm/iosm_ipc_pcie.c @@ -6,6 +6,7 @@ #include <linux/acpi.h> #include <linux/bitfield.h> #include <linux/module.h> +#include <linux/suspend.h> #include <net/rtnetlink.h> #include "iosm_ipc_imem.h" @@ -18,6 +19,7 @@ MODULE_LICENSE("GPL v2"); /* WWAN GUID */ static guid_t wwan_acpi_guid = GUID_INIT(0xbad01b75, 0x22a8, 0x4f48, 0x87, 0x92, 0xbd, 0xde, 0x94, 0x67, 0x74, 0x7d); +static bool pci_registered; static void ipc_pcie_resources_release(struct iosm_pcie *ipc_pcie) { @@ -448,7 +450,6 @@ static struct pci_driver iosm_ipc_driver = { }, .id_table = iosm_ipc_ids, }; -module_pci_driver(iosm_ipc_driver); int ipc_pcie_addr_map(struct iosm_pcie *ipc_pcie, unsigned char *data, size_t size, dma_addr_t *mapping, int direction) @@ -530,3 +531,56 @@ void ipc_pcie_kfree_skb(struct iosm_pcie *ipc_pcie, struct sk_buff *skb) IPC_CB(skb)->mapping = 0; dev_kfree_skb(skb); } + +static int pm_notify(struct notifier_block *nb, unsigned long mode, void *_unused) +{ + if (mode == PM_HIBERNATION_PREPARE || mode == PM_RESTORE_PREPARE) { + if (pci_registered) { + pci_unregister_driver(&iosm_ipc_driver); + pci_registered = false; + } + } else if (mode == PM_POST_HIBERNATION || mode == PM_POST_RESTORE) { + if (!pci_registered) { + int ret; + + ret = pci_register_driver(&iosm_ipc_driver); + if (ret) { + pr_err(KBUILD_MODNAME ": unable to re-register PCI driver: %d\n", + ret); + } else { + pci_registered = true; + } + } + } + + return 0; +} + +static struct notifier_block pm_notifier = { + .notifier_call = pm_notify, +}; + +static int __init iosm_ipc_driver_init(void) +{ + int ret; + + ret = pci_register_driver(&iosm_ipc_driver); + if (ret) + return ret; + + pci_registered = true; + + register_pm_notifier(&pm_notifier); + + return 0; +} +module_init(iosm_ipc_driver_init); + +static void __exit iosm_ipc_driver_exit(void) +{ + unregister_pm_notifier(&pm_notifier); + + if (pci_registered) + pci_unregister_driver(&iosm_ipc_driver); +} +module_exit(iosm_ipc_driver_exit);