|
@@ -46,6 +46,7 @@
|
|
|
#endif
|
|
|
#include "gsi.h"
|
|
|
#include "ipa_stats.h"
|
|
|
+#include <linux/suspend.h>
|
|
|
|
|
|
#ifdef CONFIG_ARM64
|
|
|
|
|
@@ -139,6 +140,11 @@ static int ipa_alloc_pkt_init_ex(void);
|
|
|
static void ipa3_free_pkt_init(void);
|
|
|
static void ipa3_free_pkt_init_ex(void);
|
|
|
|
|
|
+#ifdef CONFIG_DEEPSLEEP
|
|
|
+static void ipa3_deepsleep_resume(void);
|
|
|
+static void ipa3_deepsleep_suspend(void);
|
|
|
+#endif
|
|
|
+
|
|
|
static void ipa3_load_ipa_fw(struct work_struct *work);
|
|
|
static DECLARE_WORK(ipa3_fw_loading_work, ipa3_load_ipa_fw);
|
|
|
static DECLARE_DELAYED_WORK(ipa3_fw_load_failure_handle, ipa3_load_ipa_fw);
|
|
@@ -483,6 +489,38 @@ int ipa_smmu_free_sgt(struct sg_table **out_sgt_ptr)
|
|
|
}
|
|
|
EXPORT_SYMBOL(ipa_smmu_free_sgt);
|
|
|
|
|
|
+/**
|
|
|
+ * ipa_pm_notify() - PM notify to listen suspend events
|
|
|
+ *
|
|
|
+ * This callback will be invoked by the pm framework to suspend
|
|
|
+ * operation is invoked.
|
|
|
+ *
|
|
|
+ * Returns NOTIFY_DONE to pm framework completed operation.
|
|
|
+ */
|
|
|
+
|
|
|
+static int ipa_pm_notify(struct notifier_block *b, unsigned long event, void *p)
|
|
|
+{
|
|
|
+ IPAERR("Entry\n");
|
|
|
+ switch (event) {
|
|
|
+ case PM_POST_SUSPEND:
|
|
|
+#ifdef CONFIG_DEEPSLEEP
|
|
|
+ if (mem_sleep_current == PM_SUSPEND_MEM && ipa3_ctx->deepsleep) {
|
|
|
+ IPADBG("Enter deepsleep resume\n");
|
|
|
+ ipa3_deepsleep_resume();
|
|
|
+ IPADBG("Exit deepsleep resume\n");
|
|
|
+ }
|
|
|
+#endif
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ IPAERR("Exit\n");
|
|
|
+ return NOTIFY_DONE;
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+static struct notifier_block ipa_pm_notifier = {
|
|
|
+ .notifier_call = ipa_pm_notify,
|
|
|
+};
|
|
|
+
|
|
|
static const struct dev_pm_ops ipa_pm_ops = {
|
|
|
.suspend_late = ipa3_ap_suspend,
|
|
|
.resume_early = ipa3_ap_resume,
|
|
@@ -5995,6 +6033,7 @@ static void ipa3_teardown_apps_pipes(void)
|
|
|
__ipa3_del_rt_rule(ipa3_ctx->dflt_v4_rt_rule_hdl);
|
|
|
__ipa3_del_hdr(ipa3_ctx->excp_hdr_hdl, false);
|
|
|
ipa3_teardown_sys_pipe(ipa3_ctx->clnt_hdl_cmd);
|
|
|
+ ipa3_dealloc_common_event_ring();
|
|
|
}
|
|
|
|
|
|
#ifdef CONFIG_COMPAT
|
|
@@ -7292,6 +7331,12 @@ static inline void ipa3_enable_napi_lan_rx(void)
|
|
|
napi_enable(&ipa3_ctx->napi_lan_rx);
|
|
|
}
|
|
|
|
|
|
+static inline void ipa3_disable_napi_lan_rx(void)
|
|
|
+{
|
|
|
+ if (ipa3_ctx->lan_rx_napi_enable)
|
|
|
+ napi_disable(&ipa3_ctx->napi_lan_rx);
|
|
|
+}
|
|
|
+
|
|
|
static inline void ipa3_register_to_fmwk(void)
|
|
|
{
|
|
|
struct ipa_core_data data;
|
|
@@ -7789,6 +7834,10 @@ static int ipa3_post_init(const struct ipa3_plat_drv_res *resource_p,
|
|
|
/* init uc-activation tbl*/
|
|
|
ipa3_setup_uc_act_tbl();
|
|
|
|
|
|
+#ifdef CONFIG_DEEPSLEEP
|
|
|
+ if (!ipa3_is_ready())
|
|
|
+ ipa_fmwk_deepsleep_exit_ipa();
|
|
|
+#endif
|
|
|
complete_all(&ipa3_ctx->init_completion_obj);
|
|
|
|
|
|
ipa_ut_module_init();
|
|
@@ -7986,13 +8035,12 @@ static int ipa3_mdt_load_ipa_fws(const char *sub_sys)
|
|
|
|
|
|
static int ipa3_pil_load_ipa_fws(const char *sub_sys)
|
|
|
{
|
|
|
- void *subsystem_get_retval = NULL;
|
|
|
|
|
|
IPADBG("PIL FW loading process initiated sub_sys=%s\n",
|
|
|
sub_sys);
|
|
|
|
|
|
- subsystem_get_retval = subsystem_get(sub_sys);
|
|
|
- if (IS_ERR_OR_NULL(subsystem_get_retval)) {
|
|
|
+ ipa3_ctx->subsystem_get_retval = subsystem_get(sub_sys);
|
|
|
+ if (IS_ERR_OR_NULL(ipa3_ctx->subsystem_get_retval)) {
|
|
|
IPAERR("Unable to PIL load FW for sub_sys=%s\n", sub_sys);
|
|
|
return -EINVAL;
|
|
|
}
|
|
@@ -8002,6 +8050,20 @@ static int ipa3_pil_load_ipa_fws(const char *sub_sys)
|
|
|
}
|
|
|
#endif /* IS_ENABLED(CONFIG_QCOM_MDT_LOADER) */
|
|
|
|
|
|
+#ifdef CONFIG_DEEPSLEEP
|
|
|
+static int ipa3_pil_unload_ipa_fws(void)
|
|
|
+{
|
|
|
+
|
|
|
+ IPADBG("PIL FW unloading process initiated sub_sys\n");
|
|
|
+
|
|
|
+ if (ipa3_ctx->subsystem_get_retval)
|
|
|
+ subsystem_put(ipa3_ctx->subsystem_get_retval);
|
|
|
+
|
|
|
+ IPADBG("PIL FW unloading process is complete sub_sys\n");
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
static void ipa3_load_ipa_fw(struct work_struct *work)
|
|
|
{
|
|
|
int result;
|
|
@@ -11154,7 +11216,8 @@ int ipa3_ap_suspend(struct device *dev)
|
|
|
int i;
|
|
|
|
|
|
IPADBG("Enter...\n");
|
|
|
-
|
|
|
+ if (!of_device_is_compatible(dev->of_node,"qcom,ipa"))
|
|
|
+ return 0;
|
|
|
/* In case there is a tx/rx handler in polling mode fail to suspend */
|
|
|
for (i = 0; i < ipa3_ctx->ipa_num_pipes; i++) {
|
|
|
if (ipa3_ctx->ep[i].sys &&
|
|
@@ -11165,6 +11228,13 @@ int ipa3_ap_suspend(struct device *dev)
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+#ifdef CONFIG_DEEPSLEEP
|
|
|
+ if (mem_sleep_current == PM_SUSPEND_MEM) {
|
|
|
+ IPADBG("Enter deepsleep suspend\n");
|
|
|
+ ipa3_deepsleep_suspend();
|
|
|
+ IPADBG("Exit deepsleep suspend\n");
|
|
|
+ }
|
|
|
+#endif
|
|
|
ipa_pm_deactivate_all_deferred();
|
|
|
|
|
|
IPADBG("Exit\n");
|
|
@@ -11197,6 +11267,59 @@ bool ipa3_get_lan_rx_napi(void)
|
|
|
return ipa3_ctx->lan_rx_napi_enable;
|
|
|
}
|
|
|
|
|
|
+
|
|
|
+#ifdef CONFIG_DEEPSLEEP
|
|
|
+static void ipa3_deepsleep_suspend(void)
|
|
|
+{
|
|
|
+ IPADBG("Entry\n");
|
|
|
+ IPA_ACTIVE_CLIENTS_INC_SIMPLE();
|
|
|
+
|
|
|
+ /* To allow default routing table delection using this flag */
|
|
|
+ ipa3_ctx->deepsleep = true;
|
|
|
+ /*Disabling the LAN NAPI*/
|
|
|
+ ipa3_disable_napi_lan_rx();
|
|
|
+ /*NOt allow uC related operations until uC load again*/
|
|
|
+ ipa3_ctx->uc_ctx.uc_loaded = false;
|
|
|
+ /*Disconnecting LAN PROD/LAN CONS/CMD PROD apps pipes*/
|
|
|
+ ipa3_teardown_apps_pipes();
|
|
|
+ /*Deregistering the GSI driver*/
|
|
|
+ gsi_deregister_device(ipa3_ctx->gsi_dev_hdl, false);
|
|
|
+ /*Destroying filter table ids*/
|
|
|
+ ipa3_destroy_flt_tbl_idrs();
|
|
|
+ /*Disabling IPA interrupt*/
|
|
|
+ ipa3_remove_interrupt_handler(IPA_TX_SUSPEND_IRQ);
|
|
|
+ ipa3_interrupts_destroy(ipa3_res.ipa_irq, &ipa3_ctx->master_pdev->dev);
|
|
|
+ ipa3_uc_interface_destroy();
|
|
|
+ /*Destroy the NAT device*/
|
|
|
+ ipa3_nat_ipv6ct_destroy_devices();
|
|
|
+ /*Freeing memory allocated for coalesing and dma task*/
|
|
|
+ ipa3_free_coal_close_frame();
|
|
|
+ ipa3_free_dma_task_for_gsi();
|
|
|
+ /*Destroying ipa hal module*/
|
|
|
+ ipahal_destroy();
|
|
|
+ ipa3_ctx->ipa_initialization_complete = false;
|
|
|
+ ipa3_debugfs_remove();
|
|
|
+ /*Unloading IPA FW to allow FW load in resume*/
|
|
|
+ ipa3_pil_unload_ipa_fws();
|
|
|
+ /*Calling framework API to reset IPA ready flag to false*/
|
|
|
+ ipa_fmwk_deepsleep_entry_ipa();
|
|
|
+ IPA_ACTIVE_CLIENTS_DEC_SIMPLE();
|
|
|
+ IPADBG("Exit\n");
|
|
|
+}
|
|
|
+
|
|
|
+static void ipa3_deepsleep_resume(void)
|
|
|
+{
|
|
|
+
|
|
|
+ IPADBG("Entry\n");
|
|
|
+ /*After deeplseep exit we shouldn't allow delete the default routing table*/
|
|
|
+ ipa3_ctx->deepsleep = false;
|
|
|
+ /*Scheduling WQ to load IPA FW*/
|
|
|
+ queue_work(ipa3_ctx->transport_power_mgmt_wq,
|
|
|
+ &ipa3_fw_loading_work);
|
|
|
+ IPADBG("Exit\n");
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
static void ipa_gsi_notify_cb(struct gsi_per_notify *notify)
|
|
|
{
|
|
|
/*
|
|
@@ -11648,6 +11771,8 @@ static int __init ipa_module_init(void)
|
|
|
/* Register as a PCI device driver */
|
|
|
return pci_register_driver(&ipa_pci_driver);
|
|
|
}
|
|
|
+
|
|
|
+ register_pm_notifier(&ipa_pm_notifier);
|
|
|
/* Register as a platform device driver */
|
|
|
return platform_driver_register(&ipa_plat_drv);
|
|
|
}
|
|
@@ -11662,6 +11787,7 @@ static void __exit ipa_module_exit(void)
|
|
|
kfree(ipa3_ctx->hw_stats);
|
|
|
ipa3_ctx->hw_stats = NULL;
|
|
|
}
|
|
|
+ unregister_pm_notifier(&ipa_pm_notifier);
|
|
|
kfree(ipa3_ctx);
|
|
|
ipa3_ctx = NULL;
|
|
|
}
|