/** @file moal_sdio_mmc.c * * @brief This file contains SDIO MMC IF (interface) module * related functions. * * * Copyright 2008-2022, 2024 NXP * * This software file (the File) is distributed by NXP * under the terms of the GNU General Public License Version 2, June 1991 * (the License). You may use, redistribute and/or modify the File in * accordance with the terms and conditions of the License, a copy of which * is available by writing to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA or on the * worldwide web at http://www.gnu.org/licenses/old-licenses/gpl-2.0.txt. * * THE FILE IS DISTRIBUTED AS-IS, WITHOUT WARRANTY OF ANY KIND, AND THE * IMPLIED WARRANTIES OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE * ARE EXPRESSLY DISCLAIMED. The License provides additional details about * this warranty disclaimer. * */ /**************************************************** Change log: 02/25/09: Initial creation - This file supports SDIO MMC only ****************************************************/ #include #if defined(STA_CFG80211) || defined(UAP_CFG80211) #include "moal_cfg80211.h" #endif #include "moal_sdio.h" #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 1, 0) #if IS_ENABLED(CONFIG_IPV6) #include #endif #endif /** define nxp vendor id */ #define NXP_VENDOR_ID 0x0471 #define MRVL_VENDOR_ID 0x02df /******************************************************** Local Variables ********************************************************/ /* moal interface ops */ static moal_if_ops sdiommc_ops; /******************************************************** Global Variables ********************************************************/ #ifdef SD8887 /** Device ID for SD8887 */ #define SD_DEVICE_ID_8887 (0x9135) #endif #ifdef SD8801 /** Device ID for SD8801 FN1 */ #define SD_DEVICE_ID_8801 (0x9139) #endif #ifdef SD8897 /** Device ID for SD8897 */ #define SD_DEVICE_ID_8897 (0x912d) #endif #ifdef SD8977 /** Device ID for SD8977 */ #define SD_DEVICE_ID_8977 (0x9145) #endif #ifdef SD8978 /** Device ID for SD8978 */ #define SD_DEVICE_ID_8978 (0x9159) #endif #ifdef SD8997 /** Device ID for SD8997 */ #define SD_DEVICE_ID_8997 (0x9141) #endif #ifdef SD8987 /** Device ID for SD8987 */ #define SD_DEVICE_ID_8987 (0x9149) #endif #ifdef SDAW693 /** Device ID for SDAW693 */ #define SD_DEVICE_ID_AW693_FN1 (0x0211) /** Device ID for SDAW693 */ #define SD_DEVICE_ID_AW693_FN2 (0x0212) #endif #ifdef SD9098 /** Device ID for SD9098 */ #define SD_DEVICE_ID_9098_FN1 (0x914D) /** Device ID for SD9098 */ #define SD_DEVICE_ID_9098_FN2 (0x914E) #endif #ifdef SD9097 /** Device ID for SD9097 */ #define SD_DEVICE_ID_9097 (0x9155) #endif #ifdef SD9177 /** Device ID for SD9177 */ #define SD_DEVICE_ID_9177 (0x0205) #endif #ifdef SDIW624 /** Device ID for SDIW624 */ #define SD_DEVICE_ID_IW624 (0x020D) #endif #ifdef SDIW615 /** Device ID for SDIW615 */ #define SD_DEVICE_ID_IW615 (0x020D) #endif /** WLAN IDs */ static const struct sdio_device_id wlan_ids[] = { #ifdef SD8887 {SDIO_DEVICE(MRVL_VENDOR_ID, SD_DEVICE_ID_8887)}, #endif #ifdef SD8801 {SDIO_DEVICE(MRVL_VENDOR_ID, SD_DEVICE_ID_8801)}, #endif #ifdef SD8897 {SDIO_DEVICE(MRVL_VENDOR_ID, SD_DEVICE_ID_8897)}, #endif #ifdef SD8977 {SDIO_DEVICE(MRVL_VENDOR_ID, SD_DEVICE_ID_8977)}, #endif #ifdef SD8978 {SDIO_DEVICE(MRVL_VENDOR_ID, SD_DEVICE_ID_8978)}, #endif #ifdef SD8997 {SDIO_DEVICE(MRVL_VENDOR_ID, SD_DEVICE_ID_8997)}, #endif #ifdef SD8987 {SDIO_DEVICE(MRVL_VENDOR_ID, SD_DEVICE_ID_8987)}, #endif #ifdef SDAW693 {SDIO_DEVICE(NXP_VENDOR_ID, SD_DEVICE_ID_AW693_FN1)}, {SDIO_DEVICE(NXP_VENDOR_ID, SD_DEVICE_ID_AW693_FN2)}, #endif #ifdef SD9098 {SDIO_DEVICE(MRVL_VENDOR_ID, SD_DEVICE_ID_9098_FN1)}, {SDIO_DEVICE(MRVL_VENDOR_ID, SD_DEVICE_ID_9098_FN2)}, #endif #ifdef SD9097 {SDIO_DEVICE(MRVL_VENDOR_ID, SD_DEVICE_ID_9097)}, #endif #ifdef SD9177 {SDIO_DEVICE(NXP_VENDOR_ID, SD_DEVICE_ID_9177)}, #endif #ifdef SDIW624 {SDIO_DEVICE(NXP_VENDOR_ID, SD_DEVICE_ID_IW624)}, #endif #ifdef SDIW615 {SDIO_DEVICE(NXP_VENDOR_ID, SD_DEVICE_ID_IW615)}, #endif {}, }; int woal_sdio_probe(struct sdio_func *func, const struct sdio_device_id *id); void woal_sdio_remove(struct sdio_func *func); static void woal_sdiommc_work(struct work_struct *work); #ifdef SDIO static void woal_sdiommc_reg_dbg(pmoal_handle handle); #endif #ifdef SDIO_SUSPEND_RESUME #ifdef MMC_PM_KEEP_POWER int woal_sdio_suspend(struct device *dev); int woal_sdio_resume(struct device *dev); static struct dev_pm_ops wlan_sdio_pm_ops = { .suspend = woal_sdio_suspend, .resume = woal_sdio_resume, }; void woal_sdio_shutdown(struct device *dev); #endif #endif // clang-format off static struct sdio_driver REFDATA wlan_sdio = { .name = "wlan_sdio", .id_table = wlan_ids, .probe = woal_sdio_probe, .remove = woal_sdio_remove, #if LINUX_VERSION_CODE > KERNEL_VERSION(2, 6, 29) .drv = { .owner = THIS_MODULE, #ifdef SDIO_SUSPEND_RESUME #ifdef MMC_PM_KEEP_POWER .pm = &wlan_sdio_pm_ops, .shutdown = woal_sdio_shutdown, #endif #endif } #else #ifdef SDIO_SUSPEND_RESUME #ifdef MMC_PM_KEEP_POWER .drv = { .pm = &wlan_sdio_pm_ops, .shutdown = woal_sdio_shutdown, } #endif #endif #endif }; // clang-format on /******************************************************** Local Functions ********************************************************/ static void woal_sdiommc_dump_fw_info(moal_handle *phandle); #if 0 /** @brief This function dump the sdio register * * @param handle A Pointer to the moal_handle structure * @return N/A */ static void woal_dump_sdio_reg(moal_handle *handle) { int ret = 0; t_u8 data, i; int fun0_reg[] = {0x05, 0x04}; t_u8 array_size = 0; #ifdef SD8897 int fun1_reg_8897[] = {0x03, 0x04, 0x05, 0x06, 0x07, 0xC0, 0xC1}; #endif int fun1_reg_other[] = {0x03, 0x04, 0x05, 0x60, 0x61}; int *fun1_reg = NULL; for (i = 0; i < ARRAY_SIZE(fun0_reg); i++) { data = sdio_f0_readb( ((sdio_mmc_card *)handle->card)->func, fun0_reg[i], &ret); PRINTM(MMSG, "fun0: reg 0x%02x=0x%02x ret=%d\n", fun0_reg[i], data, ret); } #ifdef SD8897 if (IS_SD8897(handle->card_type)) { fun1_reg = fun1_reg_8897; array_size = sizeof(fun1_reg_8897) / sizeof(int); } else { #endif fun1_reg = fun1_reg_other; array_size = sizeof(fun1_reg_other) / sizeof(int); #ifdef SD8897 } #endif for (i = 0; i < array_size; i++) { data = sdio_readb(((sdio_mmc_card *)handle->card)->func, fun1_reg[i], &ret); PRINTM(MMSG, "fun1: reg 0x%02x=0x%02x ret=%d\n", fun1_reg[i], data, ret); } return; } #endif /******************************************************** Global Functions ********************************************************/ /** * @brief This function handles the interrupt. * * @param func A pointer to the sdio_func structure * @return N/A */ static void woal_sdio_interrupt(struct sdio_func *func) { moal_handle *handle; sdio_mmc_card *card; mlan_status status; ENTER(); card = sdio_get_drvdata(func); if (!card || !card->handle) { PRINTM(MINFO, "sdio_mmc_interrupt(func = %p) card or handle is NULL, card=%p\n", func, card); LEAVE(); return; } handle = card->handle; if (handle->surprise_removed == MTRUE) { LEAVE(); return; } handle->main_state = MOAL_RECV_INT; PRINTM(MINFO, "*** IN SDIO IRQ ***\n"); PRINTM(MINTR, "*\n"); /* call mlan_interrupt to read int status */ status = mlan_interrupt(0, handle->pmlan_adapter); if (status == MLAN_STATUS_FAILURE) { PRINTM(MINTR, "mlan interrupt failed\n"); } #ifdef SDIO_SUSPEND_RESUME if (handle->is_suspended) { PRINTM(MINTR, "Receive interrupt in hs_suspended\n"); LEAVE(); return; } #endif handle->main_state = MOAL_START_MAIN_PROCESS; /* Call MLAN main process */ status = mlan_main_process(handle->pmlan_adapter); if (status == MLAN_STATUS_FAILURE) { PRINTM(MINTR, "mlan main process exited with failure\n"); } handle->main_state = MOAL_END_MAIN_PROCESS; LEAVE(); } /** @brief This function updates the card types * * @param handle A Pointer to the moal_handle structure * @param card A Pointer to card * * @return N/A */ static t_u16 woal_update_card_type(t_void *card) { sdio_mmc_card *cardp_sd = (sdio_mmc_card *)card; t_u16 card_type = 0; /* Update card type */ #ifdef SD8887 if (cardp_sd->func->device == SD_DEVICE_ID_8887) { card_type = CARD_TYPE_SD8887; moal_memcpy_ext(NULL, driver_version, CARD_SD8887, strlen(CARD_SD8887), strlen(driver_version)); moal_memcpy_ext( NULL, driver_version + strlen(INTF_CARDTYPE) + strlen(KERN_VERSION), V15, strlen(V15), strlen(driver_version) - (strlen(INTF_CARDTYPE) + strlen(KERN_VERSION))); } #endif #ifdef SD8801 if (cardp_sd->func->device == SD_DEVICE_ID_8801) { card_type = CARD_TYPE_SD8801; moal_memcpy_ext(NULL, driver_version, CARD_SD8801, strlen(CARD_SD8801), strlen(driver_version)); moal_memcpy_ext( NULL, driver_version + strlen(INTF_CARDTYPE) + strlen(KERN_VERSION), V14, strlen(V14), strlen(driver_version) - (strlen(INTF_CARDTYPE) + strlen(KERN_VERSION))); } #endif #ifdef SD8897 if (cardp_sd->func->device == SD_DEVICE_ID_8897) { card_type = CARD_TYPE_SD8897; moal_memcpy_ext(NULL, driver_version, CARD_SD8897, strlen(CARD_SD8897), strlen(driver_version)); moal_memcpy_ext( NULL, driver_version + strlen(INTF_CARDTYPE) + strlen(KERN_VERSION), V15, strlen(V15), strlen(driver_version) - (strlen(INTF_CARDTYPE) + strlen(KERN_VERSION))); } #endif #ifdef SD8977 if (cardp_sd->func->device == SD_DEVICE_ID_8977) { card_type = CARD_TYPE_SD8977; moal_memcpy_ext(NULL, driver_version, CARD_SD8977, strlen(CARD_SD8977), strlen(driver_version)); moal_memcpy_ext( NULL, driver_version + strlen(INTF_CARDTYPE) + strlen(KERN_VERSION), V16, strlen(V16), strlen(driver_version) - (strlen(INTF_CARDTYPE) + strlen(KERN_VERSION))); } #endif #ifdef SD8978 if (cardp_sd->func->device == SD_DEVICE_ID_8978) { card_type = CARD_TYPE_SD8978; moal_memcpy_ext(NULL, driver_version, "SDIW416", strlen("SDIW416"), strlen(driver_version)); moal_memcpy_ext( NULL, driver_version + strlen(INTF_CARDTYPE) + strlen(KERN_VERSION), V16, strlen(V16), strlen(driver_version) - (strlen(INTF_CARDTYPE) + strlen(KERN_VERSION))); } #endif #ifdef SD8997 if (cardp_sd->func->device == SD_DEVICE_ID_8997) { card_type = CARD_TYPE_SD8997; moal_memcpy_ext(NULL, driver_version, CARD_SD8997, strlen(CARD_SD8997), strlen(driver_version)); moal_memcpy_ext( NULL, driver_version + strlen(INTF_CARDTYPE) + strlen(KERN_VERSION), V16, strlen(V16), strlen(driver_version) - (strlen(INTF_CARDTYPE) + strlen(KERN_VERSION))); } #endif #ifdef SD8987 if (cardp_sd->func->device == SD_DEVICE_ID_8987) { card_type = CARD_TYPE_SD8987; moal_memcpy_ext(NULL, driver_version, CARD_SD8987, strlen(CARD_SD8987), strlen(driver_version)); moal_memcpy_ext( NULL, driver_version + strlen(INTF_CARDTYPE) + strlen(KERN_VERSION), V16, strlen(V16), strlen(driver_version) - (strlen(INTF_CARDTYPE) + strlen(KERN_VERSION))); } #endif #ifdef SDIW624 if (cardp_sd->func->device == SD_DEVICE_ID_IW624) { card_type = CARD_TYPE_SDIW624; moal_memcpy_ext(NULL, driver_version, CARD_SDIW624, strlen(CARD_SDIW624), strlen(driver_version)); moal_memcpy_ext( NULL, driver_version + strlen(INTF_CARDTYPE) + strlen(KERN_VERSION), V18, strlen(V18), strlen(driver_version) - (strlen(INTF_CARDTYPE) + strlen(KERN_VERSION))); } #endif #ifdef SDAW693 if (cardp_sd->func->device == SD_DEVICE_ID_AW693_FN1 || cardp_sd->func->device == SD_DEVICE_ID_AW693_FN2) { card_type = CARD_TYPE_SDAW693; moal_memcpy_ext(NULL, driver_version, CARD_SDAW693, strlen(CARD_SDAW693), strlen(driver_version)); moal_memcpy_ext( NULL, driver_version + strlen(INTF_CARDTYPE) + strlen(KERN_VERSION), V17, strlen(V17), strlen(driver_version) - (strlen(INTF_CARDTYPE) + strlen(KERN_VERSION))); } #endif #ifdef SD9097 if (cardp_sd->func->device == SD_DEVICE_ID_9097) { card_type = CARD_TYPE_SD9097; moal_memcpy_ext(NULL, driver_version, CARD_SD9097, strlen(CARD_SD9097), strlen(driver_version)); moal_memcpy_ext( NULL, driver_version + strlen(INTF_CARDTYPE) + strlen(KERN_VERSION), V17, strlen(V17), strlen(driver_version) - (strlen(INTF_CARDTYPE) + strlen(KERN_VERSION))); } #endif #ifdef SD9098 if (cardp_sd->func->device == SD_DEVICE_ID_9098_FN1 || cardp_sd->func->device == SD_DEVICE_ID_9098_FN2) { card_type = CARD_TYPE_SD9098; moal_memcpy_ext(NULL, driver_version, CARD_SD9098, strlen(CARD_SD9098), strlen(driver_version)); moal_memcpy_ext( NULL, driver_version + strlen(INTF_CARDTYPE) + strlen(KERN_VERSION), V17, strlen(V17), strlen(driver_version) - (strlen(INTF_CARDTYPE) + strlen(KERN_VERSION))); } #endif #ifdef SD9177 if (cardp_sd->func->device == SD_DEVICE_ID_9177) { card_type = CARD_TYPE_SD9177; moal_memcpy_ext(NULL, driver_version, CARD_SD9177, strlen(CARD_SD9177), strlen(driver_version)); moal_memcpy_ext( NULL, driver_version + strlen(INTF_CARDTYPE) + strlen(KERN_VERSION), V18, strlen(V18), strlen(driver_version) - (strlen(INTF_CARDTYPE) + strlen(KERN_VERSION))); } #endif #ifdef SDIW615 if (cardp_sd->func->device == SD_DEVICE_ID_IW615) { card_type = CARD_TYPE_SDIW615; moal_memcpy_ext(NULL, driver_version, CARD_SDIW615, strlen(CARD_SDIW615), strlen(driver_version)); moal_memcpy_ext( NULL, driver_version + strlen(INTF_CARDTYPE) + strlen(KERN_VERSION), V18, strlen(V18), strlen(driver_version) - (strlen(INTF_CARDTYPE) + strlen(KERN_VERSION))); } #endif return card_type; } /** @brief This function handles client driver probe. * * @param func A pointer to sdio_func structure. * @param id A pointer to sdio_device_id structure. * @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE/error code */ int woal_sdio_probe(struct sdio_func *func, const struct sdio_device_id *id) { int ret = MLAN_STATUS_SUCCESS; sdio_mmc_card *card = NULL; t_u16 card_type = 0; ENTER(); PRINTM(MMSG, "vendor=0x%4.04X device=0x%4.04X class=%d function=%d\n", func->vendor, func->device, func->class, func->num); card = kzalloc(sizeof(sdio_mmc_card), GFP_KERNEL); if (!card) { PRINTM(MFATAL, "Failed to allocate memory in probe function!\n"); LEAVE(); return -ENOMEM; } card->func = func; #ifdef MMC_QUIRK_BLKSZ_FOR_BYTE_MODE /* The byte mode patch is available in kernel MMC driver * which fixes one issue in MP-A transfer. * bit1: use func->cur_blksize for byte mode */ func->card->quirks |= MMC_QUIRK_BLKSZ_FOR_BYTE_MODE; #endif #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 0, 0) func->card->quirks |= MMC_QUIRK_LENIENT_FN0; #endif #if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 27) /* wait for chip fully wake up */ if (!func->enable_timeout) func->enable_timeout = 200; #endif sdio_claim_host(func); ret = sdio_enable_func(func); if (ret) { sdio_release_host(func); PRINTM(MFATAL, "sdio_enable_func() failed: ret=%d\n", ret); ret = -EIO; goto err; } sdio_release_host(func); card_type = woal_update_card_type(card); if (!card_type) { PRINTM(MERROR, "sdmmc probe: woal_update_card_type() failed\n"); ret = MLAN_STATUS_FAILURE; goto err; } INIT_WORK(&card->reset_work, woal_sdiommc_work); if (NULL == woal_add_card(card, &card->func->dev, &sdiommc_ops, card_type)) { PRINTM(MMSG, "woal_add_card failed\n"); ret = MLAN_STATUS_FAILURE; goto err; } #ifdef IMX_SUPPORT woal_regist_oob_wakeup_irq(card->handle); #endif /* IMX_SUPPORT */ LEAVE(); return ret; err: kfree(card); sdio_claim_host(func); sdio_disable_func(func); sdio_release_host(func); LEAVE(); return ret; } /** @brief This function handles client driver remove. * * @param func A pointer to sdio_func structure. * @return N/A */ void woal_sdio_remove(struct sdio_func *func) { sdio_mmc_card *card; ENTER(); if (func) { PRINTM(MINFO, "SDIO func=%d\n", func->num); card = sdio_get_drvdata(func); if (card) { /* We need to advance the time to set surprise_removed * to MTRUE as fast as possible to avoid race condition * with woal_sdio_interrupt() * * @todo: Due to woal_sdio_interrupt() is called in * Linux's work queue, cannot be suspended to impact * other works. Need a lock in these two functions: * woal_sdio_remove() waits until woal_sdio_interrupt * ends. woal_sdio_interrupt() returns if * woal_sdio_remove() is running. */ if (card->handle != NULL) { card->handle->surprise_removed = MTRUE; /* check if woal_sdio_interrupt() is running */ while (card->handle->main_state != MOAL_END_MAIN_PROCESS) woal_sched_timeout(2); /* wait until woal_sdio_interrupt ends */ } #ifdef IMX_SUPPORT woal_unregist_oob_wakeup_irq(card->handle); #endif /* IMX_SUPPORT */ woal_remove_card(card); kfree(card); } } LEAVE(); } #ifdef SDIO_SUSPEND_RESUME #ifdef MMC_PM_KEEP_POWER #ifdef MMC_PM_FUNC_SUSPENDED /** @brief This function tells lower driver that WLAN is suspended * * @param handle A Pointer to the moal_handle structure * @return N/A */ void woal_wlan_is_suspended(moal_handle *handle) { ENTER(); if (handle->suspend_notify_req == MTRUE) { handle->is_suspended = MTRUE; sdio_func_suspended(((sdio_mmc_card *)handle->card)->func); } LEAVE(); } #endif /** @brief This function handles client driver shutdown * * @param dev A pointer to device structure * @return N/A */ void woal_sdio_shutdown(struct device *dev) { struct sdio_func *func = dev_to_sdio_func(dev); moal_handle *handle = NULL; sdio_mmc_card *cardp; mlan_ds_ps_info pm_info; int i, retry_num = 8; ENTER(); PRINTM(MCMND, "<--- Enter woal_sdio_shutdown --->\n"); cardp = sdio_get_drvdata(func); if (!cardp || !cardp->handle) { PRINTM(MERROR, "Card or moal_handle structure is not valid\n"); LEAVE(); return; } handle = cardp->handle; for (i = 0; i < handle->priv_num; i++) netif_device_detach(handle->priv[i]->netdev); if (moal_extflg_isset(handle, EXT_SHUTDOWN_HS)) { handle->shutdown_hs_in_process = MTRUE; memset(&pm_info, 0, sizeof(pm_info)); for (i = 0; i < retry_num; i++) { if (MLAN_STATUS_SUCCESS == woal_get_pm_info(woal_get_priv(handle, MLAN_BSS_ROLE_ANY), &pm_info)) { if (pm_info.is_suspend_allowed == MTRUE) break; else PRINTM(MMSG, "Shutdown not allowed and retry again\n"); } woal_sched_timeout(100); } if (pm_info.is_suspend_allowed == MFALSE) { PRINTM(MMSG, "Shutdown not allowed\n"); goto done; } woal_enable_hs(woal_get_priv(handle, MLAN_BSS_ROLE_ANY)); wait_event_interruptible_timeout( handle->hs_activate_wait_q, handle->hs_activate_wait_q_woken, HS_ACTIVE_TIMEOUT); if (handle->hs_activated == MTRUE) PRINTM(MMSG, "HS actived in shutdown\n"); else PRINTM(MMSG, "Fail to enable HS in shutdown\n"); } else { for (i = 0; i < MIN(handle->priv_num, MLAN_MAX_BSS_NUM); i++) { if (handle->priv[i]) { if (handle->priv[i]->media_connected == MTRUE #ifdef UAP_SUPPORT || (GET_BSS_ROLE(handle->priv[i]) == MLAN_BSS_ROLE_UAP) #endif ) { PRINTM(MIOCTL, "disconnect on suspend\n"); if (woal_disconnect( handle->priv[i], MOAL_NO_WAIT, NULL, DEF_DEAUTH_REASON_CODE)) PRINTM(MERROR, "failed to disconnect on suspend\n"); } } } } done: PRINTM(MCMND, "<--- Leave woal_sdio_shutdown --->\n"); LEAVE(); return; } /** @brief This function handles client driver suspend * * @param dev A pointer to device structure * @return MLAN_STATUS_SUCCESS or error code */ int woal_sdio_suspend(struct device *dev) { struct sdio_func *func = dev_to_sdio_func(dev); mmc_pm_flag_t pm_flags = 0; moal_handle *handle = NULL; sdio_mmc_card *cardp; int i, retry_num = 8; int ret = MLAN_STATUS_SUCCESS; int hs_actived = 0; mlan_ds_ps_info pm_info; ENTER(); PRINTM(MCMND, "<--- Enter woal_sdio_suspend --->\n"); pm_flags = sdio_get_host_pm_caps(func); PRINTM(MCMND, "%s: suspend: PM flags = 0x%x\n", sdio_func_id(func), pm_flags); cardp = sdio_get_drvdata(func); if (!cardp || !cardp->handle) { PRINTM(MERROR, "Card or moal_handle structure is not valid\n"); LEAVE(); return MLAN_STATUS_SUCCESS; } handle = cardp->handle; if (moal_extflg_isset(handle, EXT_PM_KEEP_POWER) && !(pm_flags & MMC_PM_KEEP_POWER)) { PRINTM(MERROR, "%s: cannot remain alive while host is suspended\n", sdio_func_id(func)); LEAVE(); return -ENOSYS; } if (handle->is_suspended == MTRUE) { PRINTM(MWARN, "Device already suspended\n"); LEAVE(); return MLAN_STATUS_SUCCESS; } if (handle->fw_dump) { PRINTM(MMSG, "suspend not allowed while FW dump!"); ret = -EBUSY; goto done; } #ifdef STA_SUPPORT for (i = 0; i < MIN(handle->priv_num, MLAN_MAX_BSS_NUM); i++) { if (handle->priv[i]) woal_cancel_scan(handle->priv[i], MOAL_IOCTL_WAIT); } #endif handle->suspend_fail = MFALSE; memset(&pm_info, 0, sizeof(pm_info)); for (i = 0; i < retry_num; i++) { if (MLAN_STATUS_SUCCESS == woal_get_pm_info(woal_get_priv(handle, MLAN_BSS_ROLE_ANY), &pm_info)) { if (pm_info.is_suspend_allowed == MTRUE) break; else PRINTM(MMSG, "Suspend not allowed and retry again\n"); } woal_sched_timeout(100); } if (pm_info.is_suspend_allowed == MFALSE) { PRINTM(MMSG, "Suspend not allowed\n"); ret = -EBUSY; goto done; } for (i = 0; i < handle->priv_num; i++) netif_device_detach(handle->priv[i]->netdev); if (moal_extflg_isset(handle, EXT_PM_KEEP_POWER)) { /* Enable the Host Sleep */ #ifdef MMC_PM_FUNC_SUSPENDED handle->suspend_notify_req = MTRUE; #endif /* add delay to wait kernel complete cfg80211_disconnect*/ woal_sched_timeout(200); hs_actived = woal_enable_hs( woal_get_priv(handle, MLAN_BSS_ROLE_ANY)); #ifdef MMC_PM_FUNC_SUSPENDED handle->suspend_notify_req = MFALSE; #endif if (hs_actived) { #if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 3, 4) if (pm_flags & MMC_PM_WAKE_SDIO_IRQ) { ret = sdio_set_host_pm_flags( func, MMC_PM_WAKE_SDIO_IRQ); PRINTM(MCMND, "suspend with MMC_PM_WAKE_SDIO_IRQ ret=%d\n", ret); } #endif #ifdef MMC_PM_SKIP_RESUME_PROBE PRINTM(MCMND, "suspend with MMC_PM_KEEP_POWER and MMC_PM_SKIP_RESUME_PROBE\n"); ret = sdio_set_host_pm_flags( func, MMC_PM_KEEP_POWER | MMC_PM_SKIP_RESUME_PROBE); #else PRINTM(MCMND, "suspend with MMC_PM_KEEP_POWER\n"); ret = sdio_set_host_pm_flags(func, MMC_PM_KEEP_POWER); #endif } else { PRINTM(MMSG, "HS not actived, suspend fail!"); handle->suspend_fail = MTRUE; for (i = 0; i < handle->priv_num; i++) netif_device_attach(handle->priv[i]->netdev); ret = -EBUSY; goto done; } } /* Indicate device suspended */ handle->is_suspended = MTRUE; #ifdef IMX_SUPPORT woal_enable_oob_wakeup_irq(handle); #endif /* IMX_SUPPORT */ done: PRINTM(MCMND, "<--- Leave woal_sdio_suspend --->\n"); LEAVE(); return ret; } /** @brief This function handles client driver resume * * @param dev A pointer to device structure * @return MLAN_STATUS_SUCCESS */ int woal_sdio_resume(struct device *dev) { struct sdio_func *func = dev_to_sdio_func(dev); mmc_pm_flag_t pm_flags = 0; moal_handle *handle = NULL; sdio_mmc_card *cardp; int i; ENTER(); PRINTM(MCMND, "<--- Enter woal_sdio_resume --->\n"); pm_flags = sdio_get_host_pm_caps(func); PRINTM(MCMND, "%s: resume: PM flags = 0x%x\n", sdio_func_id(func), pm_flags); cardp = sdio_get_drvdata(func); if (!cardp || !cardp->handle) { PRINTM(MERROR, "Card or moal_handle structure is not valid\n"); LEAVE(); return MLAN_STATUS_SUCCESS; } handle = cardp->handle; if (handle->is_suspended == MFALSE) { PRINTM(MWARN, "Device already resumed\n"); LEAVE(); return MLAN_STATUS_SUCCESS; } handle->is_suspended = MFALSE; if (woal_check_driver_status(handle)) { PRINTM(MERROR, "Resuem, device is in hang state\n"); LEAVE(); return MLAN_STATUS_SUCCESS; } for (i = 0; i < handle->priv_num; i++) netif_device_attach(handle->priv[i]->netdev); /* Disable Host Sleep */ woal_cancel_hs(woal_get_priv(handle, MLAN_BSS_ROLE_ANY), MOAL_NO_WAIT); #ifdef IMX_SUPPORT woal_disable_oob_wakeup_irq(handle); #endif /* IMX_SUPPORT */ PRINTM(MCMND, "<--- Leave woal_sdio_resume --->\n"); LEAVE(); return MLAN_STATUS_SUCCESS; } #endif #endif /* SDIO_SUSPEND_RESUME */ /** * @brief This function writes data into card register * * @param handle A Pointer to the moal_handle structure * @param reg Register offset * @param data Value * * @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE */ static mlan_status woal_sdiommc_write_reg(moal_handle *handle, t_u32 reg, t_u32 data) { mlan_status ret = MLAN_STATUS_FAILURE; sdio_claim_host(((sdio_mmc_card *)handle->card)->func); sdio_writeb(((sdio_mmc_card *)handle->card)->func, (t_u8)data, reg, (int *)&ret); sdio_release_host(((sdio_mmc_card *)handle->card)->func); return ret; } /** * @brief This function reads data from card register * * @param handle A Pointer to the moal_handle structure * @param reg Register offset * @param data Value * * @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE */ static mlan_status woal_sdiommc_read_reg(moal_handle *handle, t_u32 reg, t_u32 *data) { mlan_status ret = MLAN_STATUS_FAILURE; t_u8 val; sdio_claim_host(((sdio_mmc_card *)handle->card)->func); val = sdio_readb(((sdio_mmc_card *)handle->card)->func, reg, (int *)&ret); sdio_release_host(((sdio_mmc_card *)handle->card)->func); *data = val; return ret; } /** * @brief This function writes data into card register * * @param handle A Pointer to the moal_handle structure * @param reg Register offset * @param data Value * * @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE */ static mlan_status woal_sdio_writeb(moal_handle *handle, t_u32 reg, t_u8 data) { mlan_status ret = MLAN_STATUS_FAILURE; sdio_claim_host(((sdio_mmc_card *)handle->card)->func); sdio_writeb(((sdio_mmc_card *)handle->card)->func, (t_u8)data, reg, (int *)&ret); sdio_release_host(((sdio_mmc_card *)handle->card)->func); return ret; } /** * @brief This function reads data from card register * * @param handle A Pointer to the moal_handle structure * @param reg Register offset * @param data Value * * @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE */ static mlan_status woal_sdio_readb(moal_handle *handle, t_u32 reg, t_u8 *data) { mlan_status ret = MLAN_STATUS_FAILURE; t_u8 val; sdio_claim_host(((sdio_mmc_card *)handle->card)->func); val = sdio_readb(((sdio_mmc_card *)handle->card)->func, reg, (int *)&ret); sdio_release_host(((sdio_mmc_card *)handle->card)->func); *data = val; return ret; } /** * @brief This function reads data from card register FN0 * * @param handle A Pointer to the moal_handle structure * @param reg Register offset * @param data Value * * @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE */ static mlan_status woal_sdio_f0_readb(moal_handle *handle, t_u32 reg, t_u8 *data) { mlan_status ret = MLAN_STATUS_FAILURE; t_u8 val; sdio_claim_host(((sdio_mmc_card *)handle->card)->func); val = sdio_f0_readb(((sdio_mmc_card *)handle->card)->func, reg, (int *)&ret); sdio_release_host(((sdio_mmc_card *)handle->card)->func); *data = val; return ret; } /** * @brief This function use SG mode to read/write data into card memory * * @param handle A Pointer to the moal_handle structure * @param pmbuf_list Pointer to a linked list of mlan_buffer structure * @param port Port * @param write write flag * * @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE */ static mlan_status woal_sdio_rw_mb(moal_handle *handle, pmlan_buffer pmbuf_list, t_u32 port, t_u8 write) { struct scatterlist sg_list[SDIO_MP_AGGR_DEF_PKT_LIMIT_MAX]; int num_sg = pmbuf_list->use_count; int i = 0; mlan_buffer *pmbuf = NULL; struct mmc_request mmc_req; struct mmc_command mmc_cmd; struct mmc_data mmc_dat; struct sdio_func *func = ((sdio_mmc_card *)handle->card)->func; t_u32 ioport = (port & MLAN_SDIO_IO_PORT_MASK); t_u32 blkcnt = pmbuf_list->data_len / MLAN_SDIO_BLOCK_SIZE; #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 0, 0) int status; #endif if (num_sg > SDIO_MP_AGGR_DEF_PKT_LIMIT_MAX) { PRINTM(MERROR, "ERROR: num_sg=%d", num_sg); return MLAN_STATUS_FAILURE; } sg_init_table(sg_list, num_sg); pmbuf = pmbuf_list->pnext; for (i = 0; i < num_sg; i++) { if (pmbuf == pmbuf_list) break; sg_set_buf(&sg_list[i], pmbuf->pbuf + pmbuf->data_offset, pmbuf->data_len); pmbuf = pmbuf->pnext; } memset(&mmc_req, 0, sizeof(struct mmc_request)); memset(&mmc_cmd, 0, sizeof(struct mmc_command)); memset(&mmc_dat, 0, sizeof(struct mmc_data)); mmc_dat.sg = sg_list; mmc_dat.sg_len = num_sg; mmc_dat.blksz = MLAN_SDIO_BLOCK_SIZE; mmc_dat.blocks = blkcnt; mmc_dat.flags = write ? MMC_DATA_WRITE : MMC_DATA_READ; mmc_cmd.opcode = SD_IO_RW_EXTENDED; mmc_cmd.arg = write ? 1 << 31 : 0; mmc_cmd.arg |= (func->num & 0x7) << 28; mmc_cmd.arg |= 1 << 27; /* block basic */ mmc_cmd.arg |= 0; /* fix address */ mmc_cmd.arg |= (ioport & 0x1FFFF) << 9; mmc_cmd.arg |= blkcnt & 0x1FF; mmc_cmd.flags = MMC_RSP_SPI_R5 | MMC_RSP_R5 | MMC_CMD_ADTC; mmc_req.cmd = &mmc_cmd; mmc_req.data = &mmc_dat; sdio_claim_host(((sdio_mmc_card *)handle->card)->func); mmc_set_data_timeout(&mmc_dat, ((sdio_mmc_card *)handle->card)->func->card); mmc_wait_for_req(((sdio_mmc_card *)handle->card)->func->card->host, &mmc_req); if (mmc_cmd.error || mmc_dat.error) { PRINTM(MERROR, "CMD53 %s cmd_error = %d data_error=%d\n", write ? "write" : "read", mmc_cmd.error, mmc_dat.error); #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 0, 0) /* issue abort cmd52 command through F0*/ sdio_f0_writeb(((sdio_mmc_card *)handle->card)->func, 0x01, SDIO_CCCR_ABORT, &status); #endif sdio_release_host(((sdio_mmc_card *)handle->card)->func); return MLAN_STATUS_FAILURE; } sdio_release_host(((sdio_mmc_card *)handle->card)->func); return MLAN_STATUS_SUCCESS; } /** * @brief This function writes multiple bytes into card memory * * @param handle A Pointer to the moal_handle structure * @param pmbuf Pointer to mlan_buffer structure * @param port Port * @param timeout Time out value * * @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE */ static mlan_status woal_sdiommc_write_data_sync(moal_handle *handle, mlan_buffer *pmbuf, t_u32 port, t_u32 timeout) { mlan_status ret = MLAN_STATUS_FAILURE; t_u8 *buffer = (t_u8 *)(pmbuf->pbuf + pmbuf->data_offset); t_u8 blkmode = (port & MLAN_SDIO_BYTE_MODE_MASK) ? BYTE_MODE : BLOCK_MODE; t_u32 blksz = (blkmode == BLOCK_MODE) ? MLAN_SDIO_BLOCK_SIZE : 1; t_u32 blkcnt = (blkmode == BLOCK_MODE) ? (pmbuf->data_len / MLAN_SDIO_BLOCK_SIZE) : pmbuf->data_len; t_u32 ioport = (port & MLAN_SDIO_IO_PORT_MASK); int status = 0; if (pmbuf->use_count > 1) return woal_sdio_rw_mb(handle, pmbuf, port, MTRUE); #ifdef SDIO_MMC_DEBUG handle->cmd53w = 1; #endif sdio_claim_host(((sdio_mmc_card *)handle->card)->func); status = sdio_writesb(((sdio_mmc_card *)handle->card)->func, ioport, buffer, blkcnt * blksz); if (!status) ret = MLAN_STATUS_SUCCESS; else { PRINTM(MERROR, "cmd53 write error=%d\n", status); #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 0, 0) /* issue abort cmd52 command through F0*/ sdio_f0_writeb(((sdio_mmc_card *)handle->card)->func, 0x01, SDIO_CCCR_ABORT, &status); #endif } sdio_release_host(((sdio_mmc_card *)handle->card)->func); #ifdef SDIO_MMC_DEBUG handle->cmd53w = 2; #endif return ret; } /** * @brief This function reads multiple bytes from card memory * * @param handle A Pointer to the moal_handle structure * @param pmbuf Pointer to mlan_buffer structure * @param port Port * @param timeout Time out value * * @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE */ static mlan_status woal_sdiommc_read_data_sync(moal_handle *handle, mlan_buffer *pmbuf, t_u32 port, t_u32 timeout) { mlan_status ret = MLAN_STATUS_FAILURE; t_u8 *buffer = (t_u8 *)(pmbuf->pbuf + pmbuf->data_offset); t_u8 blkmode = (port & MLAN_SDIO_BYTE_MODE_MASK) ? BYTE_MODE : BLOCK_MODE; t_u32 blksz = (blkmode == BLOCK_MODE) ? MLAN_SDIO_BLOCK_SIZE : 1; t_u32 blkcnt = (blkmode == BLOCK_MODE) ? (pmbuf->data_len / MLAN_SDIO_BLOCK_SIZE) : pmbuf->data_len; t_u32 ioport = (port & MLAN_SDIO_IO_PORT_MASK); int status = 0; if (pmbuf->use_count > 1) return woal_sdio_rw_mb(handle, pmbuf, port, MFALSE); #ifdef SDIO_MMC_DEBUG handle->cmd53r = 1; #endif sdio_claim_host(((sdio_mmc_card *)handle->card)->func); status = sdio_readsb(((sdio_mmc_card *)handle->card)->func, buffer, ioport, blkcnt * blksz); if (!status) { ret = MLAN_STATUS_SUCCESS; } else { PRINTM(MERROR, "cmd53 read error=%d\n", status); #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 0, 0) /* issue abort cmd52 command through F0*/ sdio_f0_writeb(((sdio_mmc_card *)handle->card)->func, 0x01, SDIO_CCCR_ABORT, &status); #endif } sdio_release_host(((sdio_mmc_card *)handle->card)->func); #ifdef SDIO_MMC_DEBUG handle->cmd53r = 2; #endif return ret; } /** * @brief This function registers the IF module in bus driver * * @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE */ mlan_status woal_sdiommc_bus_register(void) { mlan_status ret = MLAN_STATUS_SUCCESS; ENTER(); /* SDIO Driver Registration */ if (sdio_register_driver(&wlan_sdio)) { PRINTM(MFATAL, "SDIO Driver Registration Failed \n"); LEAVE(); return MLAN_STATUS_FAILURE; } LEAVE(); return ret; } /** * @brief This function de-registers the IF module in bus driver * * @return N/A */ void woal_sdiommc_bus_unregister(void) { ENTER(); /* SDIO Driver Unregistration */ sdio_unregister_driver(&wlan_sdio); LEAVE(); } /** * @brief This function de-registers the device * * @param handle A pointer to moal_handle structure * @return N/A */ static void woal_sdiommc_unregister_dev(moal_handle *handle) { ENTER(); if (handle->card) { sdio_mmc_card *card = handle->card; #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 2, 0) struct sdio_func *func = card->func; #endif /* Release the SDIO IRQ */ sdio_claim_host(card->func); sdio_release_irq(card->func); sdio_disable_func(card->func); #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 2, 0) if (handle->driver_status) #if LINUX_VERSION_CODE >= KERNEL_VERSION(5, 18, 0) mmc_hw_reset(func->card); #else mmc_hw_reset(func->card->host); #endif #endif sdio_release_host(card->func); sdio_set_drvdata(card->func, NULL); PRINTM(MWARN, "Making the sdio dev card as NULL\n"); card->handle = NULL; } LEAVE(); } /** * @brief This function registers the device * * @param handle A pointer to moal_handle structure * @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE */ static mlan_status woal_sdiommc_register_dev(moal_handle *handle) { int ret = MLAN_STATUS_SUCCESS; sdio_mmc_card *card = handle->card; struct sdio_func *func; ENTER(); /* save adapter pointer in card */ card->handle = handle; func = card->func; sdio_claim_host(func); /* Request the SDIO IRQ */ ret = sdio_claim_irq(func, woal_sdio_interrupt); if (ret) { PRINTM(MFATAL, "sdio_claim_irq failed: ret=%d\n", ret); goto release_host; } /* Set block size */ ret = sdio_set_block_size(card->func, MLAN_SDIO_BLOCK_SIZE); if (ret) { PRINTM(MERROR, "sdio_set_block_seize(): cannot set SDIO block size\n"); ret = MLAN_STATUS_FAILURE; goto release_irq; } sdio_release_host(func); sdio_set_drvdata(func, card); LEAVE(); return MLAN_STATUS_SUCCESS; release_irq: sdio_release_irq(func); release_host: sdio_release_host(func); handle->card = NULL; LEAVE(); return MLAN_STATUS_FAILURE; } /** * @brief This function set the sdio bus width * * @param handle A pointer to moal_handle structure * @param mode 1--1 bit mode, 4--4 bit mode * @return MLAN_STATUS_SUCCESS */ int woal_sdio_set_buswidth(moal_handle *handle, t_u8 bus_width) { sdio_mmc_card *cardp = (sdio_mmc_card *)handle->card; struct mmc_host *host = cardp->func->card->host; int ret = 0; t_u8 val; ENTER(); if (bus_width != SDIO_BUS_WIDTH_1 && bus_width != SDIO_BUS_WIDTH_4) { LEAVE(); return -EFAULT; } sdio_claim_host(((sdio_mmc_card *)handle->card)->func); val = sdio_f0_readb(((sdio_mmc_card *)handle->card)->func, SDIO_CCCR_IF, (int *)&ret); val &= ~SDIO_BUS_WIDTH_MASK; if (bus_width == SDIO_BUS_WIDTH_1) host->ios.bus_width = MMC_BUS_WIDTH_1; else { host->ios.bus_width = MMC_BUS_WIDTH_4; val |= SDIO_BUS_WIDTH_4BIT; } sdio_f0_writeb(((sdio_mmc_card *)handle->card)->func, val, SDIO_CCCR_IF, &ret); sdio_release_host(cardp->func); host->ops->set_ios(host, &host->ios); LEAVE(); return ret; } /** * @brief This function set bus clock on/off * * @param handle A pointer to moal_handle structure * @param option TRUE--on , FALSE--off * @return MLAN_STATUS_SUCCESS */ int woal_sdio_set_bus_clock(moal_handle *handle, t_u8 option) { sdio_mmc_card *cardp = (sdio_mmc_card *)handle->card; struct mmc_host *host = cardp->func->card->host; ENTER(); if (option == MTRUE) { /* restore value if non-zero */ if (cardp->host_clock) host->ios.clock = cardp->host_clock; } else { /* backup value if non-zero, then clear */ if (host->ios.clock) cardp->host_clock = host->ios.clock; host->ios.clock = 0; } host->ops->set_ios(host, &host->ios); LEAVE(); return MLAN_STATUS_SUCCESS; } /** * @brief This function updates card reg based on the Cmd52 value in dev * structure * * @param handle A pointer to moal_handle structure * @param func A pointer to store func variable * @param reg A pointer to store reg variable * @param val A pointer to store val variable * @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE */ int woal_sdio_read_write_cmd52(moal_handle *handle, int func, int reg, int val) { int ret = MLAN_STATUS_SUCCESS; sdio_mmc_card *card = (sdio_mmc_card *)handle->card; ENTER(); /* Save current func and reg for read */ handle->cmd52_func = func; handle->cmd52_reg = reg; sdio_claim_host(card->func); if (val >= 0) { /* Perform actual write only if val is provided */ if (func) sdio_writeb(card->func, val, reg, &ret); else sdio_f0_writeb(card->func, val, reg, &ret); if (ret) { PRINTM(MERROR, "Cannot write value (0x%x) to func %d reg 0x%x\n", val, func, reg); } else { PRINTM(MMSG, "write value (0x%x) to func %d reg 0x%x\n", (u8)val, func, reg); handle->cmd52_val = val; } } else { if (func) val = sdio_readb(card->func, reg, &ret); else val = sdio_f0_readb(card->func, reg, &ret); if (ret) { PRINTM(MERROR, "Cannot read value from func %d reg 0x%x\n", func, reg); } else { PRINTM(MMSG, "read value (0x%x) from func %d reg 0x%x\n", (u8)val, func, reg); handle->cmd52_val = val; } } sdio_release_host(card->func); LEAVE(); return ret; } /** * @brief This function check if this is second mac * * @param handle A pointer to moal_handle structure * @return MTRUE/MFALSE * */ static t_u8 woal_sdiommc_is_second_mac(moal_handle *handle) { #if defined(SDAW693) || defined(SD9098) sdio_mmc_card *card = (sdio_mmc_card *)handle->card; #endif #ifdef SDAW693 if (card->func->device == SD_DEVICE_ID_AW693_FN2) return MTRUE; #endif #ifdef SD9098 if (card->func->device == SD_DEVICE_ID_9098_FN2) return MTRUE; #endif return MFALSE; } static mlan_status woal_sdiommc_get_fw_name(moal_handle *handle) { mlan_status ret = MLAN_STATUS_SUCCESS; #ifdef SD9098 sdio_mmc_card *card = (sdio_mmc_card *)handle->card; #endif t_u32 revision_id = 0; t_u32 rev_id_reg = handle->card_info->rev_id_reg; #if defined(SD8987) || defined(SD8997) || defined(SD9098) || \ defined(SD9097) || defined(SDIW624) || defined(SDAW693) || \ defined(SD8978) || defined(SD9177) || defined(SDIW615) t_u32 magic_reg = handle->card_info->magic_reg; t_u32 magic = 0; t_u32 host_strap_reg = handle->card_info->host_strap_reg; t_u32 strap = 0; #endif ENTER(); if (handle->params.fw_name) goto done; #ifdef SD8801 if (IS_SD8801(handle->card_type)) goto done; #endif /** Revision ID register */ woal_sdiommc_read_reg(handle, rev_id_reg, &revision_id); PRINTM(MCMND, "revision_id=0x%x\n", revision_id); #if defined(SD8987) || defined(SD8997) || defined(SD9098) || \ defined(SD9097) || defined(SDIW624) || defined(SDAW693) || \ defined(SD8978) || defined(SD9177) || defined(SDIW615) /** Revision ID register */ woal_sdiommc_read_reg(handle, magic_reg, &magic); /** Revision ID register */ woal_sdiommc_read_reg(handle, host_strap_reg, &strap); strap &= 0x1; magic &= 0xFF; /* 1 = SDSD, 0 --SD UART */ PRINTM(MCMND, "magic=0x%x strap=0x%x\n", magic, strap); #endif #if defined(SD8977) if (IS_SD8977(handle->card_type)) { switch (revision_id) { case SD8977_V0: strncpy(handle->card_info->fw_name, SD8977_V0_FW_NAME, FW_NAMW_MAX_LEN); strncpy(handle->card_info->fw_name_wlan, SD8977_WLAN_V0_FW_NAME, FW_NAMW_MAX_LEN); break; case SD8977_V1: strncpy(handle->card_info->fw_name, SD8977_V1_FW_NAME, FW_NAMW_MAX_LEN); strncpy(handle->card_info->fw_name_wlan, SD8977_WLAN_V1_FW_NAME, FW_NAMW_MAX_LEN); break; case SD8977_V2: strncpy(handle->card_info->fw_name, SD8977_V2_FW_NAME, FW_NAMW_MAX_LEN); strncpy(handle->card_info->fw_name_wlan, SD8977_WLAN_V2_FW_NAME, FW_NAMW_MAX_LEN); break; default: break; } } #endif #if defined(SD8887) if (IS_SD8887(handle->card_type)) { /* Check revision ID */ switch (revision_id) { case SD8887_A0: strncpy(handle->card_info->fw_name, SD8887_A0_FW_NAME, FW_NAMW_MAX_LEN); strncpy(handle->card_info->fw_name_wlan, SD8887_WLAN_A0_FW_NAME, FW_NAMW_MAX_LEN); break; case SD8887_A2: strncpy(handle->card_info->fw_name, SD8887_A2_FW_NAME, FW_NAMW_MAX_LEN); strncpy(handle->card_info->fw_name_wlan, SD8887_WLAN_A2_FW_NAME, FW_NAMW_MAX_LEN); break; default: break; } } #endif #ifdef SD8997 if (IS_SD8997(handle->card_type)) { if (magic == CHIP_MAGIC_VALUE) { if (strap == CARD_TYPE_SD_UART) strncpy(handle->card_info->fw_name, SDUART8997_DEFAULT_COMBO_FW_NAME, FW_NAMW_MAX_LEN); else strncpy(handle->card_info->fw_name, SDSD8997_DEFAULT_COMBO_FW_NAME, FW_NAMW_MAX_LEN); } } #endif #ifdef SD8987 if (IS_SD8987(handle->card_type)) { if (magic == CHIP_MAGIC_VALUE) { if (strap == CARD_TYPE_SD_UART) strncpy(handle->card_info->fw_name, SDUART8987_DEFAULT_COMBO_FW_NAME, FW_NAMW_MAX_LEN); else strncpy(handle->card_info->fw_name, SDSD8987_DEFAULT_COMBO_FW_NAME, FW_NAMW_MAX_LEN); } } #endif #ifdef SD8978 if (IS_SD8978(handle->card_type)) { if (magic == CHIP_MAGIC_VALUE) { if (strap == CARD_TYPE_SD_UART) strncpy(handle->card_info->fw_name, SDUART8978_DEFAULT_COMBO_FW_NAME, FW_NAMW_MAX_LEN); else strncpy(handle->card_info->fw_name, SDSD8978_DEFAULT_COMBO_FW_NAME, FW_NAMW_MAX_LEN); } } #endif #ifdef SD9098 if (IS_SD9098(handle->card_type) && (card->func->device == SD_DEVICE_ID_9098_FN1)) { switch (revision_id) { case SD9098_Z1Z2: if (magic == CHIP_MAGIC_VALUE) { if (strap == CARD_TYPE_SD_UART) strncpy(handle->card_info->fw_name, SDUART9098_DEFAULT_COMBO_FW_NAME, FW_NAMW_MAX_LEN); else strncpy(handle->card_info->fw_name, SDSD9098_DEFAULT_COMBO_FW_NAME, FW_NAMW_MAX_LEN); } strncpy(handle->card_info->fw_name_wlan, SD9098_DEFAULT_WLAN_FW_NAME, FW_NAMW_MAX_LEN); break; case SD9098_A0: case SD9098_A1: case SD9098_A2: if (magic == CHIP_MAGIC_VALUE) { if (strap == CARD_TYPE_SD_UART) strncpy(handle->card_info->fw_name, SDUART9098_COMBO_V1_FW_NAME, FW_NAMW_MAX_LEN); else strncpy(handle->card_info->fw_name, SDSD9098_COMBO_V1_FW_NAME, FW_NAMW_MAX_LEN); } strncpy(handle->card_info->fw_name_wlan, SD9098_WLAN_V1_FW_NAME, FW_NAMW_MAX_LEN); break; default: break; } } #endif #ifdef SD9097 if (IS_SD9097(handle->card_type)) { switch (revision_id) { case SD9097_B0: case SD9097_B1: if (magic == CHIP_MAGIC_VALUE) { if (strap == CARD_TYPE_SD_UART) strncpy(handle->card_info->fw_name, SDUART9097_COMBO_V1_FW_NAME, FW_NAMW_MAX_LEN); else strncpy(handle->card_info->fw_name, SDSD9097_COMBO_V1_FW_NAME, FW_NAMW_MAX_LEN); } strncpy(handle->card_info->fw_name_wlan, SD9097_WLAN_V1_FW_NAME, FW_NAMW_MAX_LEN); break; default: break; } } #endif #ifdef SDAW693 if (IS_SDAW693(handle->card_type)) { if (magic == CHIP_MAGIC_VALUE) { if (strap == CARD_TYPE_SD_UART) strncpy(handle->card_info->fw_name, SDUARTAW693_COMBO_FW_NAME, FW_NAMW_MAX_LEN); else strncpy(handle->card_info->fw_name, SDSDAW693_COMBO_FW_NAME, FW_NAMW_MAX_LEN); } } #endif #ifdef SDIW624 if (IS_SDIW624(handle->card_type)) { magic &= 0x03; if (magic == 0x03) PRINTM(MMSG, "wlan: SDIW624 in secure-boot mode\n"); if (strap == CARD_TYPE_SDIW624_UARTSPI) { if (handle->params.dual_nb) strncpy(handle->card_info->fw_name, SDUARTSPIIW624_COMBO_FW_NAME, FW_NAMW_MAX_LEN); else strncpy(handle->card_info->fw_name, SDUARTIW624_COMBO_FW_NAME, FW_NAMW_MAX_LEN); } else if (strap == CARD_TYPE_SDIW624_UARTUART) { if (handle->params.dual_nb) strncpy(handle->card_info->fw_name, SDUARTUARTIW624_COMBO_FW_NAME, FW_NAMW_MAX_LEN); else strncpy(handle->card_info->fw_name, SDUARTIW624_COMBO_FW_NAME, FW_NAMW_MAX_LEN); } else { strncpy(handle->card_info->fw_name, SDSDIW624_COMBO_FW_NAME, FW_NAMW_MAX_LEN); } } #endif #ifdef SD9177 if (IS_SD9177(handle->card_type)) { switch (revision_id) { case SD9177_A0: if (magic == CHIP_MAGIC_VALUE) { if (strap == CARD_TYPE_SD9177_UART) strncpy(handle->card_info->fw_name, SDUART9177_DEFAULT_COMBO_FW_NAME, FW_NAMW_MAX_LEN); else strncpy(handle->card_info->fw_name, SDSD9177_DEFAULT_COMBO_FW_NAME, FW_NAMW_MAX_LEN); } strncpy(handle->card_info->fw_name_wlan, SD9177_DEFAULT_WLAN_FW_NAME, FW_NAMW_MAX_LEN); break; case SD9177_A1: if (magic == CHIP_MAGIC_VALUE) { if (strap == CARD_TYPE_SD9177_UART) { if (handle->params.rf_test_mode) strncpy(handle->card_info ->fw_name, SDUART9177_DEFAULT_RFTM_COMBO_V1_FW_NAME, FW_NAMW_MAX_LEN); else strncpy(handle->card_info ->fw_name, SDUART9177_DEFAULT_COMBO_V1_FW_NAME, FW_NAMW_MAX_LEN); } else { if (handle->params.rf_test_mode) strncpy(handle->card_info ->fw_name, SDSD9177_DEFAULT_RFTM_COMBO_V1_FW_NAME, FW_NAMW_MAX_LEN); else strncpy(handle->card_info ->fw_name, SDSD9177_DEFAULT_COMBO_V1_FW_NAME, FW_NAMW_MAX_LEN); } } if (handle->params.rf_test_mode) strncpy(handle->card_info->fw_name, SD9177_DEFAULT_RFTM_WLAN_V1_FW_NAME, FW_NAMW_MAX_LEN); else strncpy(handle->card_info->fw_name_wlan, SD9177_DEFAULT_WLAN_V1_FW_NAME, FW_NAMW_MAX_LEN); break; default: break; } } #endif #ifdef SDIW615 if (IS_SDIW615(handle->card_type)) { if (magic == CHIP_MAGIC_VALUE) { if (strap == CARD_TYPE_SD_UART) strcpy(handle->card_info->fw_name, SDUARTIW615_COMBO_FW_NAME); else strcpy(handle->card_info->fw_name, SDSDIW615_COMBO_FW_NAME); } } #endif done: PRINTM(MCMND, "combo fw:%s wlan fw:%s \n", handle->card_info->fw_name, handle->card_info->fw_name_wlan); LEAVE(); return ret; } #define DEBUG_FW_DONE 0xFF #define DEBUG_MEMDUMP_FINISH 0xFE #define MAX_POLL_TRIES 100 #define HOST_TO_CARD_EVENT_REG 0x00 #define HOST_TO_CARD_EVENT MBIT(3) #define HOST_RST_EVENT MBIT(4) typedef enum { DUMP_TYPE_ITCM = 0, DUMP_TYPE_DTCM = 1, DUMP_TYPE_SQRAM = 2, DUMP_TYPE_APU_REGS = 3, DUMP_TYPE_CIU_REGS = 4, DUMP_TYPE_ICU_REGS = 5, DUMP_TYPE_MAC_REGS = 6, DUMP_TYPE_EXTEND_7 = 7, DUMP_TYPE_EXTEND_8 = 8, DUMP_TYPE_EXTEND_9 = 9, DUMP_TYPE_EXTEND_10 = 10, DUMP_TYPE_EXTEND_11 = 11, DUMP_TYPE_EXTEND_12 = 12, DUMP_TYPE_EXTEND_13 = 13, DUMP_TYPE_EXTEND_LAST = 14 } dumped_mem_type; #define MAX_NAME_LEN 8 typedef struct { t_u8 mem_name[MAX_NAME_LEN]; t_u8 *mem_Ptr; struct file *pfile_mem; t_u8 done_flag; t_u8 type; } memory_type_mapping; static memory_type_mapping mem_type_mapping_tbl[] = { {"ITCM", NULL, NULL, 0xF0, FW_DUMP_TYPE_MEM_ITCM}, {"DTCM", NULL, NULL, 0xF1, FW_DUMP_TYPE_MEM_DTCM}, {"SQRAM", NULL, NULL, 0xF2, FW_DUMP_TYPE_MEM_SQRAM}, {"APU", NULL, NULL, 0xF3, FW_DUMP_TYPE_REG_APU}, {"CIU", NULL, NULL, 0xF4, FW_DUMP_TYPE_REG_CIU}, {"ICU", NULL, NULL, 0xF5, FW_DUMP_TYPE_REG_ICU}, {"MAC", NULL, NULL, 0xF6, FW_DUMP_TYPE_REG_MAC}, {"EXT7", NULL, NULL, 0xF7, 0}, {"EXT8", NULL, NULL, 0xF8, 0}, {"EXT9", NULL, NULL, 0xF9, 0}, {"EXT10", NULL, NULL, 0xFA, 0}, {"EXT11", NULL, NULL, 0xFB, 0}, {"EXT12", NULL, NULL, 0xFC, 0}, {"EXT13", NULL, NULL, 0xFD, 0}, {"EXTLAST", NULL, NULL, 0xFE, 0}, }; static memory_type_mapping mem_type_mapping_tbl_8977_8997 = {"DUMP", NULL, NULL, 0xDD, 0}; /** * @brief This function read/write firmware via cmd52 * * @param phandle A pointer to moal_handle * @param doneflag A flag * @param trigger trigger FW dump flag * * @return MLAN_STATUS_SUCCESS */ static rdwr_status woal_cmd52_rdwr_firmware(moal_handle *phandle, t_u8 doneflag, t_u8 trigger) { int ret = 0; int tries = 0; t_u8 ctrl_data = 0; t_u8 dbg_dump_ctrl_reg = phandle->card_info->dump_fw_ctrl_reg; t_u8 debug_host_ready = phandle->card_info->dump_fw_host_ready; #ifdef SD9177 if (IS_SD9177(phandle->card_type)) { if (phandle->event_fw_dump) debug_host_ready = 0xAA; } #endif ret = woal_sdio_writeb(phandle, dbg_dump_ctrl_reg, debug_host_ready); if (ret) { PRINTM(MERROR, "SDIO Write ERR\n"); return RDWR_STATUS_FAILURE; } if (trigger) { PRINTM(MMSG, "Trigger FW dump...\n"); ret = woal_sdio_writeb(phandle, HOST_TO_CARD_EVENT_REG, HOST_TO_CARD_EVENT); if (ret) { PRINTM(MERROR, "Fail to set HOST_TO_CARD_EVENT_REG\n"); return RDWR_STATUS_FAILURE; } } #ifdef SD9177 if (IS_SD9177(phandle->card_type)) { if (phandle->event_fw_dump) return RDWR_STATUS_SUCCESS; } #endif for (tries = 0; tries < MAX_POLL_TRIES; tries++) { ret = woal_sdio_readb(phandle, dbg_dump_ctrl_reg, &ctrl_data); if (ret) { PRINTM(MERROR, "SDIO READ ERR\n"); return RDWR_STATUS_FAILURE; } if (ctrl_data == DEBUG_FW_DONE) break; if (doneflag && ctrl_data == doneflag) return RDWR_STATUS_DONE; if (ctrl_data != debug_host_ready) { PRINTM(MMSG, "The ctrl reg was changed, re-try again!\n"); ret = woal_sdio_writeb(phandle, dbg_dump_ctrl_reg, debug_host_ready); if (ret) { PRINTM(MERROR, "SDIO Write ERR\n"); return RDWR_STATUS_FAILURE; } } udelay(100); } if (ctrl_data == debug_host_ready || tries == MAX_POLL_TRIES) { PRINTM(MERROR, "Fail to pull ctrl_data\n"); return RDWR_STATUS_FAILURE; } return RDWR_STATUS_SUCCESS; } #ifdef SD8801 #define DEBUG_HOST_READY 0xEE #define DEBUG_FW_DONE 0xFF #define DEBUG_MEMDUMP_FINISH 0xFE #define MAX_POLL_TRIES 100 #define DEBUG_ITCM_DONE 0xaa #define DEBUG_DTCM_DONE 0xbb #define DEBUG_SQRAM_DONE 0xcc #define DEBUG_DUMP_CTRL_REG 0x63 #define DEBUG_DUMP_FIRST_REG 0x62 #define DEBUG_DUMP_START_REG 0x64 #define DEBUG_DUMP_END_REG 0x6a #define ITCM_SIZE 0x60000 #define SQRAM_SIZE 0x33500 #define DTCM_SIZE 0x14000 /** * @brief This function dump firmware memory to file * * @param phandle A pointer to moal_handle * * @return N/A */ void woal_dump_firmware_info(moal_handle *phandle) { int ret = 0; unsigned int reg, reg_start, reg_end; #ifndef DUMP_TO_PROC t_u8 path_name[64], file_name[32]; #endif t_u8 *ITCM_Ptr = NULL; t_u8 *DTCM_Ptr = NULL; t_u8 *SQRAM_Ptr = NULL; t_u8 *dbg_ptr = NULL; t_u32 sec, usec; t_u8 ctrl_data = 0; t_u32 dtcm_size = DTCM_SIZE; t_u32 sqram_size = SQRAM_SIZE; t_u8 *end_ptr = NULL; int tries; if (!phandle) { PRINTM(MERROR, "Could not dump firmwware info\n"); return; } #ifdef DUMP_TO_PROC if (!phandle->fw_dump_buf) { ret = moal_vmalloc(phandle, FW_DUMP_INFO_LEN, &(phandle->fw_dump_buf)); if (ret != MLAN_STATUS_SUCCESS || !phandle->fw_dump_buf) { PRINTM(MERROR, "Failed to vmalloc fw dump bufffer\n"); return; } } else { memset(phandle->fw_dump_buf, 0x00, FW_DUMP_INFO_LEN); } phandle->fw_dump_len = 0; #else #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 1, 0) /** Create dump directort*/ woal_create_dump_dir(phandle, path_name, sizeof(path_name)); #else memset(path_name, 0, sizeof(path_name)); strncpy(path_name, "/data", sizeof(path_name)); #endif PRINTM(MMSG, "Directory name is %s\n", path_name); woal_dump_drv_info(phandle, path_name); #endif #if LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 32) sdio_claim_host(((sdio_mmc_card *)phandle->card)->func); #endif /* start dump fw memory */ moal_get_system_time(phandle, &sec, &usec); PRINTM(MMSG, "==== DEBUG MODE OUTPUT START: %u.%06u ====\n", sec, usec); ret = moal_vmalloc(phandle, ITCM_SIZE + 1, (t_u8 **)&ITCM_Ptr); if ((ret != MLAN_STATUS_SUCCESS) || !ITCM_Ptr) { PRINTM(MERROR, "Error: vmalloc ITCM buffer failed!!!\n"); goto done; } PRINTM(MMSG, "DTCM_SIZE=0x%x\n", dtcm_size); ret = moal_vmalloc(phandle, dtcm_size + 1, (t_u8 **)&DTCM_Ptr); if ((ret != MLAN_STATUS_SUCCESS) || !DTCM_Ptr) { PRINTM(MERROR, "Error: vmalloc DTCM buffer failed!!!\n"); goto done; } ret = moal_vmalloc(phandle, sqram_size + 1, (t_u8 **)&SQRAM_Ptr); if ((ret != MLAN_STATUS_SUCCESS) || !SQRAM_Ptr) { PRINTM(MERROR, "Error: vmalloc SQRAM buffer failed!!!\n"); goto done; } dbg_ptr = ITCM_Ptr; end_ptr = ITCM_Ptr + ITCM_SIZE; moal_get_system_time(phandle, &sec, &usec); PRINTM(MMSG, "Start ITCM output %u.%06u, please wait...\n", sec, usec); reg_start = DEBUG_DUMP_START_REG; reg_end = DEBUG_DUMP_END_REG; do { ret = woal_sdio_writeb(phandle, DEBUG_DUMP_CTRL_REG, DEBUG_HOST_READY); if (ret) { PRINTM(MERROR, "SDIO Write ERR\n"); goto done; } for (tries = 0; tries < MAX_POLL_TRIES; tries++) { ret = woal_sdio_readb(phandle, DEBUG_DUMP_CTRL_REG, &ctrl_data); if (ret) { PRINTM(MERROR, "SDIO READ ERR\n"); goto done; } if ((ctrl_data == DEBUG_FW_DONE) || (ctrl_data == DEBUG_ITCM_DONE) || (ctrl_data == DEBUG_DTCM_DONE) || (ctrl_data == DEBUG_SQRAM_DONE)) break; if (ctrl_data != DEBUG_HOST_READY) { ret = woal_sdio_writeb(phandle, DEBUG_DUMP_CTRL_REG, DEBUG_HOST_READY); if (ret) { PRINTM(MERROR, "SDIO Write ERR\n"); goto done; } } udelay(100); } if (ctrl_data == DEBUG_HOST_READY) { PRINTM(MERROR, "Fail to pull ctrl_data\n"); goto done; } reg = DEBUG_DUMP_FIRST_REG; ret = woal_sdio_readb(phandle, reg, dbg_ptr); if (ret) { PRINTM(MMSG, "SDIO READ ERR\n"); goto done; } if (dbg_ptr < end_ptr) dbg_ptr++; else { PRINTM(MINFO, "pre-allocced buf is not enough\n"); goto done; } for (reg = reg_start; reg <= reg_end; reg++) { ret = woal_sdio_readb(phandle, reg, dbg_ptr); if (ret) { PRINTM(MMSG, "SDIO READ ERR\n"); goto done; } if (dbg_ptr < end_ptr) dbg_ptr++; else PRINTM(MINFO, "pre-allocced buf is not enough\n"); } switch (ctrl_data) { case DEBUG_ITCM_DONE: #ifdef MLAN_64BIT PRINTM(MMSG, "ITCM done: size=0x%lx\n", dbg_ptr - ITCM_Ptr); #else PRINTM(MMSG, "ITCM done: size=0x%x\n", dbg_ptr - ITCM_Ptr); #endif #ifdef DUMP_TO_PROC woal_save_dump_info_to_buf(phandle, ITCM_Ptr, ITCM_SIZE, FW_DUMP_TYPE_MEM_ITCM); #else memset(file_name, 0, sizeof(file_name)); snprintf(file_name, sizeof(file_name), "%s", "file_sdio_ITCM"); if (MLAN_STATUS_SUCCESS != woal_save_dump_info_to_file(path_name, file_name, ITCM_Ptr, ITCM_SIZE)) PRINTM(MMSG, "Can't save dump file %s in %s\n", file_name, path_name); #endif dbg_ptr = DTCM_Ptr; end_ptr = DTCM_Ptr + dtcm_size; moal_get_system_time(phandle, &sec, &usec); PRINTM(MMSG, "Start DTCM output %u.%06u, please wait...\n", sec, usec); break; case DEBUG_DTCM_DONE: #ifdef MLAN_64BIT PRINTM(MMSG, "DTCM done: size=0x%lx\n", dbg_ptr - DTCM_Ptr); #else PRINTM(MMSG, "DTCM done: size=0x%x\n", dbg_ptr - DTCM_Ptr); #endif #ifdef DUMP_TO_PROC woal_save_dump_info_to_buf(phandle, ITCM_Ptr, dtcm_size, FW_DUMP_TYPE_MEM_DTCM); #else memset(file_name, 0, sizeof(file_name)); snprintf(file_name, sizeof(file_name), "%s", "file_sdio_DTCM"); if (MLAN_STATUS_SUCCESS != woal_save_dump_info_to_file(path_name, file_name, DTCM_Ptr, dtcm_size)) PRINTM(MMSG, "Can't save dump file %s in %s\n", file_name, path_name); #endif dbg_ptr = SQRAM_Ptr; end_ptr = SQRAM_Ptr + sqram_size; moal_get_system_time(phandle, &sec, &usec); PRINTM(MMSG, "Start SQRAM output %u.%06u, please wait...\n", sec, usec); break; case DEBUG_SQRAM_DONE: #ifdef MLAN_64BIT PRINTM(MMSG, "SQRAM done: size=0x%lx\n", dbg_ptr - SQRAM_Ptr); #else PRINTM(MMSG, "SQRAM done: size=0x%x\n", dbg_ptr - SQRAM_Ptr); #endif #ifdef DUMP_TO_PROC woal_save_dump_info_to_buf(phandle, SQRAM_Ptr, sqram_size, FW_DUMP_TYPE_MEM_SQRAM); #else memset(file_name, 0, sizeof(file_name)); snprintf(file_name, sizeof(file_name), "%s", "file_sdio_SQRAM"); if (MLAN_STATUS_SUCCESS != woal_save_dump_info_to_file(path_name, file_name, SQRAM_Ptr, sqram_size)) PRINTM(MMSG, "Can't save dump file %s in %s\n", file_name, path_name); #endif PRINTM(MMSG, "End output!\n"); break; default: break; } } while (ctrl_data != DEBUG_SQRAM_DONE); #ifdef DUMP_TO_PROC woal_append_end_block(phandle); #endif PRINTM(MMSG, "The output ITCM/DTCM/SQRAM have been saved to files successfully!\n"); moal_get_system_time(phandle, &sec, &usec); PRINTM(MMSG, "==== DEBUG MODE OUTPUT END: %u.%06u ====\n", sec, usec); /* end dump fw memory */ done: #if LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 32) sdio_release_host(((sdio_mmc_card *)phandle->card)->func); #endif if (ITCM_Ptr) moal_vfree(phandle, ITCM_Ptr); if (DTCM_Ptr) moal_vfree(phandle, DTCM_Ptr); if (SQRAM_Ptr) moal_vfree(phandle, SQRAM_Ptr); PRINTM(MMSG, "==== DEBUG MODE END ====\n"); return; } #endif /** * @brief This function dump firmware memory to file * * @param phandle A pointer to moal_handle * * @return N/A */ void woal_dump_firmware_info_v2(moal_handle *phandle) { int ret = 0; unsigned int reg, reg_start, reg_end; t_u8 *dbg_ptr = NULL; t_u32 sec, usec; t_u8 dump_num = 0; t_u8 idx = 0; t_u8 doneflag = 0; rdwr_status stat; t_u8 i = 0; t_u8 read_reg = 0; t_u32 memory_size = 0; #ifndef DUMP_TO_PROC t_u8 path_name[64], file_name[32], firmware_dump_file[128]; #endif t_u8 *end_ptr = NULL; t_u8 dbg_dump_start_reg = 0; t_u8 dbg_dump_end_reg = 0; t_u8 dbg_dump_ctrl_reg = 0; if (!phandle) { PRINTM(MERROR, "Could not dump firmwware info\n"); return; } dbg_dump_start_reg = phandle->card_info->dump_fw_start_reg; dbg_dump_end_reg = phandle->card_info->dump_fw_end_reg; dbg_dump_ctrl_reg = phandle->card_info->dump_fw_ctrl_reg; #ifdef DUMP_TO_PROC if (!phandle->fw_dump_buf) { ret = moal_vmalloc(phandle, FW_DUMP_INFO_LEN, &(phandle->fw_dump_buf)); if (ret != MLAN_STATUS_SUCCESS || !phandle->fw_dump_buf) { PRINTM(MERROR, "Failed to vmalloc fw dump bufffer\n"); return; } } else { memset(phandle->fw_dump_buf, 0x00, FW_DUMP_INFO_LEN); } phandle->fw_dump_len = 0; #else #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 1, 0) /** Create dump directort*/ woal_create_dump_dir(phandle, path_name, sizeof(path_name)); #else memset(path_name, 0, sizeof(path_name)); strncpy(path_name, "/data", sizeof(path_name)); #endif PRINTM(MMSG, "Directory name is %s\n", path_name); woal_dump_drv_info(phandle, path_name); #endif /* start dump fw memory */ moal_get_system_time(phandle, &sec, &usec); PRINTM(MMSG, "==== DEBUG MODE OUTPUT START: %u.%06u ====\n", sec, usec); /* read the number of the memories which will dump */ if (RDWR_STATUS_FAILURE == woal_cmd52_rdwr_firmware(phandle, doneflag, MFALSE)) { if (RDWR_STATUS_FAILURE == woal_cmd52_rdwr_firmware(phandle, doneflag, MTRUE)) goto done; } reg = dbg_dump_start_reg; ret = woal_sdio_readb(phandle, reg, &dump_num); if (ret) { PRINTM(MMSG, "SDIO READ MEM NUM ERR\n"); goto done; } /* read the length of every memory which will dump */ for (idx = 0; idx < dump_num; idx++) { if (RDWR_STATUS_FAILURE == woal_cmd52_rdwr_firmware(phandle, doneflag, MFALSE)) goto done; memory_size = 0; reg = dbg_dump_start_reg; for (i = 0; i < 4; i++) { ret = woal_sdio_readb(phandle, reg, &read_reg); if (ret) { PRINTM(MMSG, "SDIO READ ERR\n"); goto done; } memory_size |= (read_reg << i * 8); reg++; } if (memory_size == 0) { PRINTM(MMSG, "Firmware Dump Finished!\n"); ret = woal_sdiommc_write_reg(phandle, dbg_dump_ctrl_reg, DEBUG_MEMDUMP_FINISH); if (ret) { PRINTM(MERROR, "SDIO Write MEMDUMP_FINISH ERR\n"); goto done; } break; } else { PRINTM(MMSG, "%s_SIZE=0x%x\n", mem_type_mapping_tbl[idx].mem_name, memory_size); ret = moal_vmalloc( phandle, memory_size + 1, (t_u8 **)&mem_type_mapping_tbl[idx].mem_Ptr); if ((ret != MLAN_STATUS_SUCCESS) || !mem_type_mapping_tbl[idx].mem_Ptr) { PRINTM(MERROR, "Error: vmalloc %s buffer failed!!!\n", mem_type_mapping_tbl[idx].mem_name); goto done; } dbg_ptr = mem_type_mapping_tbl[idx].mem_Ptr; end_ptr = dbg_ptr + memory_size; } doneflag = mem_type_mapping_tbl[idx].done_flag; moal_get_system_time(phandle, &sec, &usec); PRINTM(MMSG, "Start %s output %u.%06u, please wait...\n", mem_type_mapping_tbl[idx].mem_name, sec, usec); do { stat = woal_cmd52_rdwr_firmware(phandle, doneflag, MFALSE); if (RDWR_STATUS_FAILURE == stat) goto done; reg_start = dbg_dump_start_reg; reg_end = dbg_dump_end_reg; for (reg = reg_start; reg <= reg_end; reg++) { ret = woal_sdio_readb(phandle, reg, dbg_ptr); if (ret) { PRINTM(MMSG, "SDIO READ ERR\n"); goto done; } if (dbg_ptr < end_ptr) dbg_ptr++; else PRINTM(MINFO, "pre-allocced buf is not enough\n"); } if (RDWR_STATUS_DONE == stat) { #ifdef MLAN_64BIT PRINTM(MMSG, "%s done:" "size = 0x%lx\n", mem_type_mapping_tbl[idx].mem_name, dbg_ptr - mem_type_mapping_tbl[idx] .mem_Ptr); #else PRINTM(MMSG, "%s done:" "size = 0x%x\n", mem_type_mapping_tbl[idx].mem_name, dbg_ptr - mem_type_mapping_tbl[idx] .mem_Ptr); #endif #ifdef DUMP_TO_PROC woal_save_dump_info_to_buf( phandle, mem_type_mapping_tbl[idx].mem_Ptr, memory_size, mem_type_mapping_tbl[idx].type); #else memset(file_name, 0, sizeof(file_name)); snprintf(file_name, sizeof(file_name), "%s%s", "file_sdio_", mem_type_mapping_tbl[idx].mem_name); if (MLAN_STATUS_SUCCESS != woal_save_dump_info_to_file( path_name, file_name, mem_type_mapping_tbl[idx].mem_Ptr, memory_size)) PRINTM(MERROR, "Can't save dump file %s in %s\n", file_name, path_name); #endif moal_vfree(phandle, mem_type_mapping_tbl[idx].mem_Ptr); mem_type_mapping_tbl[idx].mem_Ptr = NULL; break; } } while (1); } #ifdef DUMP_TO_PROC woal_append_end_block(phandle); #endif moal_get_system_time(phandle, &sec, &usec); PRINTM(MMSG, "==== DEBUG MODE OUTPUT END: %u.%06u ====\n", sec, usec); /* end dump fw memory */ #ifndef DUMP_TO_PROC memset(firmware_dump_file, 0, sizeof(firmware_dump_file)); snprintf(firmware_dump_file, sizeof(firmware_dump_file), "%s/%s", path_name, file_name); moal_memcpy_ext(phandle, phandle->firmware_dump_file, firmware_dump_file, sizeof(firmware_dump_file), sizeof(phandle->firmware_dump_file)); #endif done: for (idx = 0; idx < dump_num; idx++) { if (mem_type_mapping_tbl[idx].mem_Ptr) { moal_vfree(phandle, mem_type_mapping_tbl[idx].mem_Ptr); mem_type_mapping_tbl[idx].mem_Ptr = NULL; } } PRINTM(MMSG, "==== DEBUG MODE END ====\n"); return; } static void woal_trigger_nmi_on_no_dump_event(moal_handle *phandle) { int ret = 0; t_u8 ctrl_data = 0; t_u8 dbg_dump_ctrl_reg = phandle->card_info->dump_fw_ctrl_reg; phandle->init_wait_q_woken = MFALSE; ret = woal_sdio_readb(phandle, dbg_dump_ctrl_reg, &ctrl_data); if (ctrl_data == 0xAA) { ret = wait_event_timeout(phandle->init_wait_q, phandle->init_wait_q_woken, 2 * HZ); if (!ret) { PRINTM(MMSG, "Trigger NMI FW dump...\n"); ret = woal_sdio_writeb(phandle, HOST_TO_CARD_EVENT_REG, HOST_RST_EVENT); } } } /** * @brief This function dump firmware memory to file * * @param phandle A pointer to moal_handle * * @return N/A */ void woal_dump_firmware_info_v3(moal_handle *phandle) { int ret = 0; int tries = 0; unsigned int reg, reg_start, reg_end; t_u8 *dbg_ptr = NULL; t_u8 *temp_Ptr = NULL; t_u32 sec, usec; t_u8 start_flag = 0; t_u8 doneflag = 0; rdwr_status stat; t_u32 memory_size = 0; #ifndef DUMP_TO_PROC t_u8 path_name[64], file_name[32], firmware_dump_file[128]; moal_handle *ref_handle; #endif t_u8 *end_ptr = NULL; t_u8 dbg_dump_start_reg = 0; t_u8 dbg_dump_end_reg = 0; memory_type_mapping *pmem_type_mapping_tbl = &mem_type_mapping_tbl_8977_8997; if (!phandle) { PRINTM(MERROR, "Could not dump firmwware info\n"); return; } #ifdef SD9177 if (IS_SD9177(phandle->card_type)) { if (phandle->event_fw_dump) { if (RDWR_STATUS_FAILURE != woal_cmd52_rdwr_firmware(phandle, doneflag, MFALSE)) { PRINTM(MMSG, "====SDIO FW DUMP EVENT MODE START ====\n"); return; } if (RDWR_STATUS_FAILURE != woal_cmd52_rdwr_firmware(phandle, doneflag, MTRUE)) { PRINTM(MMSG, "====SDIO FW DUMP EVENT MODE START ====\n"); return; } } } #endif dbg_dump_start_reg = phandle->card_info->dump_fw_start_reg; dbg_dump_end_reg = phandle->card_info->dump_fw_end_reg; #ifndef DUMP_TO_PROC #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 1, 0) /** Create dump directort*/ woal_create_dump_dir(phandle, path_name, sizeof(path_name)); #else memset(path_name, 0, sizeof(path_name)); strncpy(path_name, "/data", sizeof(path_name)); #endif PRINTM(MMSG, "Directory name is %s\n", path_name); ref_handle = (moal_handle *)phandle->pref_mac; if (ref_handle) woal_dump_drv_info(ref_handle, path_name); woal_dump_drv_info(phandle, path_name); #endif /* start dump fw memory */ moal_get_system_time(phandle, &sec, &usec); PRINTM(MMSG, "==== DEBUG MODE OUTPUT START: %u.%06u ====\n", sec, usec); /* read the number of the memories which will dump */ if (RDWR_STATUS_FAILURE == woal_cmd52_rdwr_firmware(phandle, doneflag, MFALSE)) { if (RDWR_STATUS_FAILURE == woal_cmd52_rdwr_firmware(phandle, doneflag, MTRUE)) goto done; } /** check the reg which indicate dump starting */ for (reg = dbg_dump_start_reg; reg <= dbg_dump_end_reg; reg++) { for (tries = 0; tries < MAX_POLL_TRIES; tries++) { ret = woal_sdio_readb(phandle, reg, &start_flag); if (ret) { PRINTM(MMSG, "SDIO READ ERR\n"); goto done; } /** 0 means dump starting*/ if (start_flag == 0) break; udelay(100); } if (tries == MAX_POLL_TRIES) { PRINTM(MMSG, "FW not ready to dump\n"); goto done; } } memory_size = 0xF0000; PRINTM(MMSG, "%s_SIZE=0x%x\n", pmem_type_mapping_tbl->mem_name, memory_size); ret = moal_vmalloc(phandle, memory_size + 1, (t_u8 **)&pmem_type_mapping_tbl->mem_Ptr); if ((ret != MLAN_STATUS_SUCCESS) || !pmem_type_mapping_tbl->mem_Ptr) { PRINTM(MERROR, "Error: vmalloc %s buffer failed!!!\n", pmem_type_mapping_tbl->mem_name); goto done; } dbg_ptr = pmem_type_mapping_tbl->mem_Ptr; end_ptr = dbg_ptr + memory_size; doneflag = pmem_type_mapping_tbl->done_flag; moal_get_system_time(phandle, &sec, &usec); PRINTM(MMSG, "Start %s output %u.%06u, please wait...\n", pmem_type_mapping_tbl->mem_name, sec, usec); do { stat = woal_cmd52_rdwr_firmware(phandle, doneflag, MFALSE); if (RDWR_STATUS_FAILURE == stat) goto done; reg_start = dbg_dump_start_reg; reg_end = dbg_dump_end_reg; for (reg = reg_start; reg <= reg_end; reg++) { ret = woal_sdio_readb(phandle, reg, dbg_ptr); if (ret) { PRINTM(MMSG, "SDIO READ ERR\n"); goto done; } dbg_ptr++; if (dbg_ptr >= end_ptr) { PRINTM(MINFO, "pre-allocced buf is not enough\n"); ret = moal_vmalloc(phandle, memory_size + 0x2000 + 1, (t_u8 **)&temp_Ptr); if ((ret != MLAN_STATUS_SUCCESS) || !temp_Ptr) { PRINTM(MERROR, "Error: vmalloc buffer failed!!!\n"); goto done; } moal_memcpy_ext(phandle, temp_Ptr, pmem_type_mapping_tbl->mem_Ptr, memory_size, memory_size + 0x2000); moal_vfree(phandle, pmem_type_mapping_tbl->mem_Ptr); pmem_type_mapping_tbl->mem_Ptr = temp_Ptr; temp_Ptr = NULL; dbg_ptr = pmem_type_mapping_tbl->mem_Ptr + memory_size; memory_size += 0x2000; end_ptr = pmem_type_mapping_tbl->mem_Ptr + memory_size; } } if (RDWR_STATUS_DONE == stat) { #ifdef MLAN_64BIT PRINTM(MMSG, "%s done:" "size = 0x%lx\n", pmem_type_mapping_tbl->mem_name, dbg_ptr - pmem_type_mapping_tbl->mem_Ptr); #else PRINTM(MMSG, "%s done:" "size = 0x%x\n", pmem_type_mapping_tbl->mem_name, dbg_ptr - pmem_type_mapping_tbl->mem_Ptr); #endif #ifdef DUMP_TO_PROC if (phandle->fw_dump_buf) { moal_vfree(phandle, phandle->fw_dump_buf); phandle->fw_dump_buf = NULL; phandle->fw_dump_len = 0; } phandle->fw_dump_buf = pmem_type_mapping_tbl->mem_Ptr; phandle->fw_dump_len = dbg_ptr - pmem_type_mapping_tbl->mem_Ptr; pmem_type_mapping_tbl->mem_Ptr = NULL; #else memset(file_name, 0, sizeof(file_name)); snprintf(file_name, sizeof(file_name), "%s%s", "file_sdio_", pmem_type_mapping_tbl->mem_name); if (MLAN_STATUS_SUCCESS != woal_save_dump_info_to_file( path_name, file_name, pmem_type_mapping_tbl->mem_Ptr, dbg_ptr - pmem_type_mapping_tbl->mem_Ptr)) PRINTM(MERROR, "Can't save dump file %s in %s\n", file_name, path_name); moal_vfree(phandle, pmem_type_mapping_tbl->mem_Ptr); pmem_type_mapping_tbl->mem_Ptr = NULL; #endif break; } } while (1); moal_get_system_time(phandle, &sec, &usec); PRINTM(MMSG, "==== DEBUG MODE OUTPUT END: %u.%06u ====\n", sec, usec); /* end dump fw memory */ #ifndef DUMP_TO_PROC memset(firmware_dump_file, 0, sizeof(firmware_dump_file)); snprintf(firmware_dump_file, sizeof(firmware_dump_file), "%s/%s", path_name, file_name); moal_memcpy_ext(phandle, phandle->firmware_dump_file, firmware_dump_file, sizeof(firmware_dump_file), sizeof(phandle->firmware_dump_file)); #endif done: if (pmem_type_mapping_tbl->mem_Ptr) { moal_vfree(phandle, pmem_type_mapping_tbl->mem_Ptr); pmem_type_mapping_tbl->mem_Ptr = NULL; } PRINTM(MMSG, "==== DEBUG MODE END ====\n"); return; } /** * @brief This function reads and displays SDIO registers for debugging * * @param phandle A pointer to moal_handle * * @return N/A */ static void woal_sdiommc_reg_dbg(moal_handle *phandle) { int ret = 0; t_u8 loop, index = 0, func, data; unsigned int reg, reg_start, reg_end; unsigned int scratch_reg = phandle->card_info->scratch_reg; t_u8 *reg_table = phandle->card_info->dump_reg.reg_table; t_u8 reg_table_size = phandle->card_info->dump_reg.reg_table_size; char buf[256], *ptr; mlan_pm_wakeup_card(phandle->pmlan_adapter, MTRUE); reg = phandle->card_info->fw_stuck_code_reg; if (reg != 0) { ret = woal_sdio_readb(phandle, reg, &data); if (!ret && data) { PRINTM(MERROR, "FW in debug mode (0x%x)\n", data); } } for (loop = 0; loop < 5; loop++) { memset(buf, 0, sizeof(buf)); ptr = buf; if (loop == 0) { /* Read the registers of SDIO function0 */ func = loop; reg_start = 0; reg_end = 9; } else if (loop == 1) { /* Read the registers of SDIO function1 */ func = loop; reg_start = phandle->card_info->func1_reg_start; reg_end = phandle->card_info->func1_reg_end; } else if (loop == 2) { /* Read specific registers of SDIO function1 */ index = 0; func = 1; reg_start = reg_table[index++]; reg_end = reg_table[reg_table_size - 1]; } else { /* Read the scratch registers of SDIO function1 */ if (loop == 4) mdelay(100); func = 1; reg_start = scratch_reg; reg_end = scratch_reg + 23; } if (loop != 2) ptr += snprintf(ptr, sizeof(buf), "SDIO Func%d (%#x-%#x): ", func, reg_start, reg_end); else ptr += snprintf(ptr, sizeof(buf), "SDIO Func%d: ", func); for (reg = reg_start; reg <= reg_end;) { if (func == 0) ret = woal_sdio_f0_readb(phandle, reg, &data); else ret = woal_sdio_readb(phandle, reg, &data); if (loop == 2) ptr += snprintf(ptr, sizeof(buf), "(%#x) ", reg); if (!ret) ptr += snprintf(ptr, sizeof(buf), "%02x ", data); else { ptr += snprintf(ptr, sizeof(buf), "ERR"); break; } if (loop == 2 && reg < reg_end) reg = reg_table[index++]; else reg++; } PRINTM(MMSG, "%s\n", buf); } mlan_pm_wakeup_card(phandle->pmlan_adapter, MFALSE); } /** * @brief This function dump firmware memory to file * * @param phandle A pointer to moal_handle * * @return N/A */ static void woal_sdiommc_dump_fw_info(moal_handle *phandle) { if (!phandle) { PRINTM(MERROR, "Could not dump firmwware info\n"); return; } #ifdef DUMP_TO_PROC if (phandle->fw_dump_buf) { PRINTM(MERROR, "FW dump already exist\n"); return; } #endif /** cancel all pending commands */ mlan_ioctl(phandle->pmlan_adapter, NULL); mlan_pm_wakeup_card(phandle->pmlan_adapter, MTRUE); phandle->fw_dump = MTRUE; if (phandle->card_info->dump_fw_info == DUMP_FW_SDIO_V2) { woal_dump_firmware_info_v2(phandle); } else if (phandle->card_info->dump_fw_info == DUMP_FW_SDIO_V3) { woal_dump_firmware_info_v3(phandle); if (phandle->event_fw_dump) { woal_trigger_nmi_on_no_dump_event(phandle); queue_work(phandle->workqueue, &phandle->main_work); phandle->is_fw_dump_timer_set = MTRUE; woal_mod_timer(&phandle->fw_dump_timer, MOAL_TIMER_5S); return; } } #ifdef SD8801 else { woal_dump_firmware_info(phandle); } #endif phandle->fw_dump = MFALSE; woal_sdiommc_reg_dbg(phandle); if (!phandle->priv_num) return; woal_send_fw_dump_complete_event( woal_get_priv(phandle, MLAN_BSS_ROLE_ANY)); mlan_pm_wakeup_card(phandle->pmlan_adapter, MFALSE); queue_work(phandle->workqueue, &phandle->main_work); woal_process_hang(phandle); return; } /** * @brief This function save sdio reg info * * @param phandle A pointer to moal_handle * @param buf A pointer buffer saving log * * @return The length of this log */ static int woal_sdiommc_dump_reg_info(moal_handle *phandle, t_u8 *drv_buf) { char *drv_ptr = (char *)drv_buf; int ret = 0; t_u8 loop, index = 0, func, data; unsigned int reg, reg_start, reg_end; unsigned int scratch_reg = 0; t_u8 *reg_table = NULL; t_u8 reg_table_size = 0; char buf[256], *ptr; ENTER(); if (!phandle || !drv_buf) { PRINTM(MMSG, "%s: can't retreive info\n", __func__); LEAVE(); return 0; } scratch_reg = phandle->card_info->scratch_reg; reg_table = phandle->card_info->dump_reg.reg_table; reg_table_size = phandle->card_info->dump_reg.reg_table_size; mlan_pm_wakeup_card(phandle->pmlan_adapter, MTRUE); reg = phandle->card_info->fw_stuck_code_reg; if (reg != 0) { ret = woal_sdio_readb(phandle, reg, &data); if (!ret && data) { PRINTM(MERROR, "FW in debug mode (0x%x)\n", data); drv_ptr += snprintf(drv_ptr, MAX_BUF_LEN, "FW in debug mode (0x%x)\n", data); } } drv_ptr += snprintf(drv_ptr, MAX_BUF_LEN, "--------sdio_reg_debug_info---------\n"); for (loop = 0; loop < 5; loop++) { memset(buf, 0, sizeof(buf)); ptr = buf; if (loop == 0) { /* Read the registers of SDIO function0 */ func = loop; reg_start = 0; reg_end = 9; } else if (loop == 1) { /* Read the registers of SDIO function1 */ func = loop; reg_start = phandle->card_info->func1_reg_start; reg_end = phandle->card_info->func1_reg_end; } else if (loop == 2) { /* Read specific registers of SDIO function1 */ index = 0; func = 1; reg_start = reg_table[index++]; reg_end = reg_table[reg_table_size - 1]; } else { /* Read the scratch registers of SDIO function1 */ if (loop == 4) mdelay(100); func = 1; reg_start = scratch_reg; reg_end = scratch_reg + 10; } if (loop != 2) ptr += snprintf(ptr, MAX_BUF_LEN, "SDIO Func%d (%#x-%#x): ", func, reg_start, reg_end); else ptr += snprintf(ptr, MAX_BUF_LEN, "SDIO Func%d: ", func); for (reg = reg_start; reg <= reg_end;) { if (func == 0) ret = woal_sdio_f0_readb(phandle, reg, &data); else ret = woal_sdio_readb(phandle, reg, &data); if (loop == 2) ptr += snprintf(ptr, MAX_BUF_LEN, "(%#x) ", reg); if (!ret) ptr += snprintf(ptr, MAX_BUF_LEN, "%02x ", data); else { ptr += snprintf(ptr, MAX_BUF_LEN, "ERR"); break; } if (loop == 2 && reg < reg_end) reg = reg_table[index++]; else reg++; } drv_ptr += snprintf(drv_ptr, MAX_BUF_LEN, "%s\n", buf); } drv_ptr += snprintf(drv_ptr, MAX_BUF_LEN, "--------sdio_reg_debug_info End---------\n"); mlan_pm_wakeup_card(phandle->pmlan_adapter, MFALSE); LEAVE(); return (int)(drv_ptr - (char *)drv_buf); } /** * @brief This function reset sdio through sdio bus driver * * @param phandle A pointer to moal_handle * * @return N/A */ void woal_sdio_reset_hw(moal_handle *handle) { sdio_mmc_card *card = handle->card; struct sdio_func *func = card->func; ENTER(); sdio_claim_host(func); sdio_release_irq(card->func); sdio_disable_func(card->func); #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 2, 0) #if LINUX_VERSION_CODE >= KERNEL_VERSION(5, 18, 0) mmc_hw_reset(func->card); #else mmc_hw_reset(func->card->host); #endif #endif #ifdef MMC_QUIRK_BLKSZ_FOR_BYTE_MODE /* The byte mode patch is available in kernel MMC driver * which fixes one issue in MP-A transfer. * bit1: use func->cur_blksize for byte mode */ func->card->quirks |= MMC_QUIRK_BLKSZ_FOR_BYTE_MODE; #endif #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 0, 0) func->card->quirks |= MMC_QUIRK_LENIENT_FN0; #endif #if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 27) /* wait for chip fully wake up */ if (!func->enable_timeout) func->enable_timeout = 200; #endif sdio_enable_func(func); sdio_claim_irq(func, woal_sdio_interrupt); sdio_set_block_size(card->func, MLAN_SDIO_BLOCK_SIZE); sdio_release_host(func); LEAVE(); return; } /** * @brief This function reload fw * * @param handle A pointer to moal_handle structure * * @return 0--success, otherwise failure */ static int woal_sdiommc_reset_fw(moal_handle *handle) { int ret = 0, tries = 0; t_u32 value = 1; t_u32 reset_reg = handle->card_info->fw_reset_reg; t_u8 reset_val = handle->card_info->fw_reset_val; t_u32 wakeup_reg = handle->card_info->fw_wakeup_reg; t_u8 wakeup_val = handle->card_info->fw_wakeup_val; ENTER(); /** write power up bit to wakeu device */ if (handle->ops.write_reg(handle, wakeup_reg, wakeup_val) != MLAN_STATUS_SUCCESS) { PRINTM(MERROR, "Failed to write wakeup register.\n"); ret = -EFAULT; goto done; } /** wait SOC fully wake up */ for (tries = 0; tries < MAX_POLL_TRIES; ++tries) { ret = handle->ops.write_reg(handle, reset_reg, 0xba); if (ret == MLAN_STATUS_SUCCESS) { handle->ops.read_reg(handle, reset_reg, &value); if (value == 0xba) { PRINTM(MMSG, "FW wake up\n"); break; } } udelay(1000); } /* Write register to notify FW */ if (handle->ops.write_reg(handle, reset_reg, reset_val) != MLAN_STATUS_SUCCESS) { PRINTM(MERROR, "Failed to write reset register.\n"); ret = -EFAULT; goto done; } #if defined(SD9098) || defined(SD9097) || defined(SDIW624) || \ defined(SDAW693) || defined(SD9177) || defined(SDIW615) if (IS_SD9098(handle->card_type) || IS_SD9097(handle->card_type) || IS_SDIW624(handle->card_type) || IS_SD9177(handle->card_type) || IS_SDIW615(handle->card_type) || IS_SDAW693(handle->card_type)) handle->ops.write_reg(handle, 0x00, 0x10); #endif /* Poll register around 100 ms */ for (tries = 0; tries < MAX_POLL_TRIES; ++tries) { handle->ops.read_reg(handle, reset_reg, &value); if (value == 0) /* FW is ready */ break; udelay(1000); } if (value) { PRINTM(MERROR, "Failed to poll FW reset register %X=0x%x\n", reset_reg, value); ret = -EFAULT; goto done; } PRINTM(MMSG, "SDIO Trigger FW In-band Reset success"); done: LEAVE(); return ret; } /** * @brief Function to process pre/post SDIO function level reset * * @param handle A pointer to moal_handle structure * @param prepare True :- its a prepare reset * False :- its a post reset * @param flr True: flag for wifi-only firmware * * Note: This function is mix of woal_switch_drv_mode() and * remove_card(). Idea is to cleanup the software only. * post reset: add_card() * * @return MLAN_STATUS_SUCCESS or MLAN_STATUS_FAILURE */ static mlan_status woal_do_sdiommc_flr(moal_handle *handle, bool prepare, bool flr_flag) { unsigned int i; int index = 0; mlan_status status = MLAN_STATUS_SUCCESS; moal_private *priv = NULL; int fw_serial_bkp = 0; sdio_mmc_card *card = NULL; ENTER(); if (!handle) { PRINTM(MINFO, "\n Handle null during prepare=%d\n", prepare); LEAVE(); return status; } card = (sdio_mmc_card *)handle->card; if (MOAL_ACQ_SEMAPHORE_BLOCK(&AddRemoveCardSem)) goto exit_sem_err; if (!prepare) goto perform_init; /* Reset all interfaces */ priv = woal_get_priv(handle, MLAN_BSS_ROLE_ANY); mlan_disable_host_int(handle->pmlan_adapter); woal_reset_intf(priv, MOAL_IOCTL_WAIT, MTRUE); woal_clean_up(handle); mlan_ioctl(handle->pmlan_adapter, NULL); /* Shutdown firmware */ handle->init_wait_q_woken = MFALSE; status = mlan_shutdown_fw(handle->pmlan_adapter); if (status == MLAN_STATUS_PENDING) wait_event_interruptible(handle->init_wait_q, handle->init_wait_q_woken); if (atomic_read(&handle->rx_pending) || atomic_read(&handle->tx_pending) || atomic_read(&handle->ioctl_pending)) { PRINTM(MERROR, "ERR: rx_pending=%d,tx_pending=%d,ioctl_pending=%d\n", atomic_read(&handle->rx_pending), atomic_read(&handle->tx_pending), atomic_read(&handle->ioctl_pending)); } unregister_inetaddr_notifier(&handle->woal_notifier); #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 1, 0) #if IS_ENABLED(CONFIG_IPV6) unregister_inet6addr_notifier(&handle->woal_inet6_notifier); #endif #endif #ifdef WIFI_DIRECT_SUPPORT #if defined(STA_CFG80211) && defined(UAP_CFG80211) #if CFG80211_VERSION_CODE >= WIFI_DIRECT_KERNEL_VERSION /* Remove virtual interface */ woal_remove_virtual_interface(handle); #endif #endif #endif /* Remove interface */ for (i = 0; i < handle->priv_num; i++) woal_remove_interface(handle, i); handle->priv_num = 0; #if defined(STA_CFG80211) || defined(UAP_CFG80211) /* Unregister wiphy device and free */ if (handle->wiphy) { wiphy_unregister(handle->wiphy); woal_cfg80211_free_bands(handle->wiphy); wiphy_free(handle->wiphy); handle->wiphy = NULL; } #endif /* Unregister mlan */ if (handle->pmlan_adapter) { mlan_unregister(handle->pmlan_adapter); if (atomic_read(&handle->lock_count) || atomic_read(&handle->malloc_count) || atomic_read(&handle->mbufalloc_count)) { PRINTM(MERROR, "mlan has memory leak: lock_count=%d," " malloc_count=%d, mbufalloc_count=%d\n", atomic_read(&handle->lock_count), atomic_read(&handle->malloc_count), atomic_read(&handle->mbufalloc_count)); } handle->pmlan_adapter = NULL; } handle->fw_dump = MFALSE; goto exit; perform_init: handle->priv_num = 0; /* Init SW */ if (woal_init_sw(handle)) { PRINTM(MFATAL, "Software Init Failed\n"); goto err_init_fw; } /* Init FW and HW */ /* Load wlan only binary */ if (flr_flag) { fw_serial_bkp = moal_extflg_isset(handle, EXT_FW_SERIAL); moal_extflg_clear(handle, EXT_FW_SERIAL); woal_update_firmware_name(handle); } if (woal_init_fw(handle)) { PRINTM(MFATAL, "Firmware Init Failed\n"); woal_sdiommc_reg_dbg(handle); if (fw_serial_bkp) moal_extflg_set(handle, EXT_FW_SERIAL); goto err_init_fw; } if (flr_flag && fw_serial_bkp) moal_extflg_set(handle, EXT_FW_SERIAL); exit: MOAL_REL_SEMAPHORE(&AddRemoveCardSem); exit_sem_err: LEAVE(); return status; err_init_fw: if (handle->is_fw_dump_timer_set) { woal_cancel_timer(&handle->fw_dump_timer); handle->is_fw_dump_timer_set = MFALSE; } if ((handle->hardware_status == HardwareStatusFwReady) || (handle->hardware_status == HardwareStatusReady)) { PRINTM(MINFO, "shutdown mlan\n"); handle->init_wait_q_woken = MFALSE; status = mlan_shutdown_fw(handle->pmlan_adapter); if (status == MLAN_STATUS_PENDING) wait_event_interruptible(handle->init_wait_q, handle->init_wait_q_woken); } #ifdef ANDROID_KERNEL #if LINUX_VERSION_CODE >= KERNEL_VERSION(4, 1, 0) wakeup_source_trash(&handle->ws); #else wake_lock_destroy(&handle->wake_lock); #endif #endif #ifdef CONFIG_PROC_FS woal_proc_exit(handle); #endif /* Unregister device */ PRINTM(MINFO, "unregister device\n"); woal_sdiommc_unregister_dev(handle); handle->surprise_removed = MTRUE; #ifdef REASSOCIATION if (handle->reassoc_thread.pid) wake_up_interruptible(&handle->reassoc_thread.wait_q); /* waiting for main thread quit */ while (handle->reassoc_thread.pid) woal_sched_timeout(2); #endif /* REASSOCIATION */ if (moal_extflg_isset(handle, EXT_NAPI)) netif_napi_del(&handle->napi_rx); woal_terminate_workqueue(handle); woal_free_moal_handle(handle); for (index = 0; index < MAX_MLAN_ADAPTER; index++) { if (m_handle[index] == handle) break; } if (index < MAX_MLAN_ADAPTER) m_handle[index] = NULL; card->handle = NULL; MOAL_REL_SEMAPHORE(&AddRemoveCardSem); LEAVE(); return (mlan_status)MLAN_STATUS_FAILURE; } /** * @brief This function handle the sdio work * * @param WORK A pointer to work_struct * @return N/A * */ static void woal_sdiommc_work(struct work_struct *work) { sdio_mmc_card *card = container_of(work, sdio_mmc_card, reset_work); moal_handle *handle = NULL; moal_handle *ref_handle = NULL; PRINTM(MMSG, "========START IN-BAND RESET===========\n"); handle = card->handle; // handle-> mac0 , ref_handle->second mac if (handle->pref_mac) { if (handle->second_mac) { handle = (moal_handle *)handle->pref_mac; ref_handle = (moal_handle *)handle->pref_mac; } else { ref_handle = (moal_handle *)handle->pref_mac; } if (ref_handle) { ref_handle->surprise_removed = MTRUE; woal_clean_up(ref_handle); mlan_ioctl(ref_handle->pmlan_adapter, NULL); } } handle->surprise_removed = MTRUE; handle->fw_reseting = MTRUE; woal_do_sdiommc_flr(handle, true, true); if (ref_handle) { ref_handle->surprise_removed = MTRUE; ref_handle->fw_reseting = MTRUE; woal_do_sdiommc_flr(ref_handle, true, true); } if (woal_sdiommc_reset_fw(handle)) { PRINTM(MERROR, "SDIO In-band Reset Fail\n"); goto done; } handle->surprise_removed = MFALSE; if (MLAN_STATUS_SUCCESS == woal_do_sdiommc_flr(handle, false, true)) handle->fw_reseting = MFALSE; else handle = NULL; if (ref_handle) { ref_handle->surprise_removed = MFALSE; if (MLAN_STATUS_SUCCESS == woal_do_sdiommc_flr(ref_handle, false, true)) ref_handle->fw_reseting = MFALSE; } card->work_flags = MFALSE; done: wifi_status = WIFI_STATUS_OK; if (handle) woal_send_auto_recovery_complete_event(handle); PRINTM(MMSG, "========END IN-BAND RESET===========\n"); return; } /** * @brief This function start reset_work * * @param handle A pointer to moal_handle structure * @return MTRUE/MFALSE * */ static void woal_sdiommc_card_reset(moal_handle *handle) { sdio_mmc_card *card = handle->card; if (!card->work_flags) { card->work_flags = MTRUE; schedule_work(&card->reset_work); } } static moal_if_ops sdiommc_ops = { .register_dev = woal_sdiommc_register_dev, .unregister_dev = woal_sdiommc_unregister_dev, .read_reg = woal_sdiommc_read_reg, .write_reg = woal_sdiommc_write_reg, .read_data_sync = woal_sdiommc_read_data_sync, .write_data_sync = woal_sdiommc_write_data_sync, .get_fw_name = woal_sdiommc_get_fw_name, .dump_fw_info = woal_sdiommc_dump_fw_info, .dump_reg_info = woal_sdiommc_dump_reg_info, .reg_dbg = woal_sdiommc_reg_dbg, .card_reset = woal_sdiommc_card_reset, .is_second_mac = woal_sdiommc_is_second_mac, };