|
@@ -1,5 +1,8 @@
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
|
|
-/* Copyright (c) 2016-2021, The Linux Foundation. All rights reserved. */
|
|
|
+/*
|
|
|
+ * Copyright (c) 2016-2021, The Linux Foundation. All rights reserved.
|
|
|
+ * Copyright (c) 2021 Qualcomm Innovation Center, Inc. All rights reserved.
|
|
|
+ */
|
|
|
|
|
|
#include <linux/cma.h>
|
|
|
#include <linux/io.h>
|
|
@@ -220,6 +223,7 @@ static const struct mhi_controller_config cnss_mhi_config = {
|
|
|
.ch_cfg = cnss_mhi_channels,
|
|
|
.num_events = ARRAY_SIZE(cnss_mhi_events),
|
|
|
.event_cfg = cnss_mhi_events,
|
|
|
+ .m2_no_db = true,
|
|
|
};
|
|
|
|
|
|
static struct cnss_pci_reg ce_src[] = {
|
|
@@ -1647,12 +1651,13 @@ EXPORT_SYMBOL(cnss_pci_unlock_reg_window);
|
|
|
*/
|
|
|
static void cnss_pci_dump_bl_sram_mem(struct cnss_pci_data *pci_priv)
|
|
|
{
|
|
|
- struct cnss_plat_data *plat_priv = pci_priv->plat_priv;
|
|
|
u32 mem_addr, val, pbl_log_max_size, sbl_log_max_size;
|
|
|
- u32 pbl_log_sram_start, sbl_log_def_start, sbl_log_def_end;
|
|
|
+ u32 pbl_log_sram_start;
|
|
|
u32 pbl_stage, sbl_log_start, sbl_log_size;
|
|
|
u32 pbl_wlan_boot_cfg, pbl_bootstrap_status;
|
|
|
u32 pbl_bootstrap_status_reg = PBL_BOOTSTRAP_STATUS;
|
|
|
+ u32 sbl_log_def_start = SRAM_START;
|
|
|
+ u32 sbl_log_def_end = SRAM_END;
|
|
|
int i;
|
|
|
|
|
|
switch (pci_priv->device_id) {
|
|
@@ -1660,28 +1665,17 @@ static void cnss_pci_dump_bl_sram_mem(struct cnss_pci_data *pci_priv)
|
|
|
pbl_log_sram_start = QCA6390_DEBUG_PBL_LOG_SRAM_START;
|
|
|
pbl_log_max_size = QCA6390_DEBUG_PBL_LOG_SRAM_MAX_SIZE;
|
|
|
sbl_log_max_size = QCA6390_DEBUG_SBL_LOG_SRAM_MAX_SIZE;
|
|
|
- sbl_log_def_start = QCA6390_V2_SBL_DATA_START;
|
|
|
- sbl_log_def_end = QCA6390_V2_SBL_DATA_END;
|
|
|
break;
|
|
|
case QCA6490_DEVICE_ID:
|
|
|
pbl_log_sram_start = QCA6490_DEBUG_PBL_LOG_SRAM_START;
|
|
|
pbl_log_max_size = QCA6490_DEBUG_PBL_LOG_SRAM_MAX_SIZE;
|
|
|
sbl_log_max_size = QCA6490_DEBUG_SBL_LOG_SRAM_MAX_SIZE;
|
|
|
- if (plat_priv->device_version.major_version == FW_V2_NUMBER) {
|
|
|
- sbl_log_def_start = QCA6490_V2_SBL_DATA_START;
|
|
|
- sbl_log_def_end = QCA6490_V2_SBL_DATA_END;
|
|
|
- } else {
|
|
|
- sbl_log_def_start = QCA6490_V1_SBL_DATA_START;
|
|
|
- sbl_log_def_end = QCA6490_V1_SBL_DATA_END;
|
|
|
- }
|
|
|
break;
|
|
|
case WCN7850_DEVICE_ID:
|
|
|
pbl_bootstrap_status_reg = WCN7850_PBL_BOOTSTRAP_STATUS;
|
|
|
pbl_log_sram_start = WCN7850_DEBUG_PBL_LOG_SRAM_START;
|
|
|
pbl_log_max_size = WCN7850_DEBUG_PBL_LOG_SRAM_MAX_SIZE;
|
|
|
sbl_log_max_size = WCN7850_DEBUG_SBL_LOG_SRAM_MAX_SIZE;
|
|
|
- sbl_log_def_start = WCN7850_SBL_DATA_START;
|
|
|
- sbl_log_def_end = WCN7850_SBL_DATA_END;
|
|
|
default:
|
|
|
return;
|
|
|
}
|
|
@@ -1945,6 +1939,12 @@ retry_mhi_suspend:
|
|
|
break;
|
|
|
case CNSS_MHI_TRIGGER_RDDM:
|
|
|
ret = mhi_force_rddm_mode(pci_priv->mhi_ctrl);
|
|
|
+ if (ret) {
|
|
|
+ cnss_pr_err("Failed to trigger RDDM, err = %d\n", ret);
|
|
|
+
|
|
|
+ cnss_pr_dbg("Sending host reset req\n");
|
|
|
+ ret = mhi_force_reset(pci_priv->mhi_ctrl);
|
|
|
+ }
|
|
|
break;
|
|
|
case CNSS_MHI_RDDM_DONE:
|
|
|
break;
|
|
@@ -2689,7 +2689,7 @@ static int cnss_qca6290_powerup(struct cnss_pci_data *pci_priv)
|
|
|
int ret = 0;
|
|
|
struct cnss_plat_data *plat_priv = pci_priv->plat_priv;
|
|
|
unsigned int timeout;
|
|
|
- int retry = 0;
|
|
|
+ int retry = 0, bt_en_gpio = plat_priv->pinctrl_info.bt_en_gpio;
|
|
|
|
|
|
if (plat_priv->ramdump_info_v2.dump_data_valid) {
|
|
|
cnss_pci_clear_dump_info(pci_priv);
|
|
@@ -2721,6 +2721,15 @@ retry:
|
|
|
}
|
|
|
if (ret == -EAGAIN && retry++ < POWER_ON_RETRY_MAX_TIMES) {
|
|
|
cnss_power_off_device(plat_priv);
|
|
|
+ /* Force toggle BT_EN GPIO low */
|
|
|
+ if (retry == POWER_ON_RETRY_MAX_TIMES) {
|
|
|
+ cnss_pr_dbg("Retry #%d. Set BT_EN GPIO(%u) low\n",
|
|
|
+ retry, bt_en_gpio);
|
|
|
+ if (bt_en_gpio >= 0)
|
|
|
+ gpio_direction_output(bt_en_gpio, 0);
|
|
|
+ cnss_pr_dbg("BT_EN GPIO val: %d\n",
|
|
|
+ gpio_get_value(bt_en_gpio));
|
|
|
+ }
|
|
|
cnss_pr_dbg("Retry to resume PCI link #%d\n", retry);
|
|
|
msleep(POWER_ON_RETRY_DELAY_MS * retry);
|
|
|
goto retry;
|
|
@@ -3014,7 +3023,10 @@ static void cnss_wlan_reg_driver_work(struct work_struct *work)
|
|
|
if (test_bit(CNSS_COLD_BOOT_CAL_DONE, &plat_priv->driver_state)) {
|
|
|
goto reg_driver;
|
|
|
} else {
|
|
|
- cnss_pr_err("Calibration still not done\n");
|
|
|
+ cnss_pr_err("Timeout waiting for calibration to complete\n");
|
|
|
+ del_timer(&plat_priv->fw_boot_timer);
|
|
|
+ if (!test_bit(CNSS_IN_REBOOT, &plat_priv->driver_state))
|
|
|
+ CNSS_ASSERT(0);
|
|
|
cal_info = kzalloc(sizeof(*cal_info), GFP_KERNEL);
|
|
|
if (!cal_info)
|
|
|
return;
|
|
@@ -3022,8 +3034,6 @@ static void cnss_wlan_reg_driver_work(struct work_struct *work)
|
|
|
cnss_driver_event_post(plat_priv,
|
|
|
CNSS_DRIVER_EVENT_COLD_BOOT_CAL_DONE,
|
|
|
0, cal_info);
|
|
|
- /* Temporarily return for bringup. CBC will not be triggered */
|
|
|
- return;
|
|
|
}
|
|
|
reg_driver:
|
|
|
if (test_bit(CNSS_IN_REBOOT, &plat_priv->driver_state)) {
|
|
@@ -5188,6 +5198,8 @@ void cnss_pci_collect_dump_info(struct cnss_pci_data *pci_priv, bool in_panic)
|
|
|
rddm_image = pci_priv->mhi_ctrl->rddm_image;
|
|
|
dump_data->nentries = 0;
|
|
|
|
|
|
+ cnss_mhi_dump_sfr(pci_priv);
|
|
|
+
|
|
|
if (!dump_seg) {
|
|
|
cnss_pr_warn("FW image dump collection not setup");
|
|
|
goto skip_dump;
|
|
@@ -5219,8 +5231,6 @@ void cnss_pci_collect_dump_info(struct cnss_pci_data *pci_priv, bool in_panic)
|
|
|
|
|
|
dump_data->nentries += rddm_image->entries;
|
|
|
|
|
|
- cnss_mhi_dump_sfr(pci_priv);
|
|
|
-
|
|
|
cnss_pr_dbg("Collect remote heap dump segment\n");
|
|
|
|
|
|
for (i = 0, j = 0; i < plat_priv->fw_mem_seg_len; i++) {
|
|
@@ -5761,6 +5771,8 @@ static irqreturn_t cnss_pci_wake_handler(int irq, void *data)
|
|
|
{
|
|
|
struct cnss_pci_data *pci_priv = data;
|
|
|
struct cnss_plat_data *plat_priv = pci_priv->plat_priv;
|
|
|
+ enum rpm_status status;
|
|
|
+ struct device *dev;
|
|
|
|
|
|
pci_priv->wake_counter++;
|
|
|
cnss_pr_dbg("WLAN PCI wake IRQ (%u) is asserted #%u\n",
|
|
@@ -5776,8 +5788,12 @@ static irqreturn_t cnss_pci_wake_handler(int irq, void *data)
|
|
|
*/
|
|
|
pm_system_wakeup();
|
|
|
|
|
|
- if (cnss_pci_get_monitor_wake_intr(pci_priv) &&
|
|
|
- cnss_pci_get_auto_suspended(pci_priv)) {
|
|
|
+ dev = &pci_priv->pci_dev->dev;
|
|
|
+ status = dev->power.runtime_status;
|
|
|
+
|
|
|
+ if ((cnss_pci_get_monitor_wake_intr(pci_priv) &&
|
|
|
+ cnss_pci_get_auto_suspended(pci_priv)) ||
|
|
|
+ (status == RPM_SUSPENDING || status == RPM_SUSPENDED)) {
|
|
|
cnss_pci_set_monitor_wake_intr(pci_priv, false);
|
|
|
cnss_pci_pm_request_resume(pci_priv);
|
|
|
}
|