//------------------------------------------------------------------------------ // ISC License (ISC) // // Copyright (c) 2004-2010, The Linux Foundation // All rights reserved. // Software was previously licensed under ISC license by Qualcomm Atheros, Inc. // // // Permission to use, copy, modify, and/or distribute this software for any // purpose with or without fee is hereby granted, provided that the above // copyright notice and this permission notice appear in all copies. // // THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES // WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF // MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR // ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES // WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN // ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF // OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. // // //------------------------------------------------------------------------------ //============================================================================== // Author(s): ="Atheros" //============================================================================== #include "a_config.h" #include "athdefs.h" #include "a_types.h" #include "a_osapi.h" #include "targaddrs.h" #include "hif.h" #include "htc_api.h" #include "wmi.h" #include "bmi.h" #include "bmi_msg.h" #include "common_drv.h" #define ATH_MODULE_NAME misc #include "a_debug.h" #include "ar6000_diag.h" #include "target_reg_table.h" #include "host_reg_table.h" static ATH_DEBUG_MODULE_DBG_INFO *g_pModuleInfoHead = NULL; static A_MUTEX_T g_ModuleListLock; static A_BOOL g_ModuleDebugInit = FALSE; #define DUMP2STR(a) (a)[0], (a)[1], (a)[2], (a)[3] #define DUMPSTR "0x%8.8X 0x%8.8X 0x%8.8X 0x%8.8X" #ifdef DEBUG ATH_DEBUG_INSTANTIATE_MODULE_VAR(misc, "misc", "Common and misc APIs", ATH_DEBUG_MASK_DEFAULTS, 0, NULL); #endif #define AR6001_LOCAL_COUNT_ADDRESS 0x0c014080 #define AR6002_LOCAL_COUNT_ADDRESS 0x00018080 #define AR6003_LOCAL_COUNT_ADDRESS 0x00018080 static A_UINT8 custDataAR6002[AR6002_CUST_DATA_SIZE]; static A_UINT8 custDataAR6003[AR6003_CUST_DATA_SIZE]; static A_UINT8 custDataMCKINLEY[MCKINLEY_CUST_DATA_SIZE]; /* * Read from the AR6000 through its diagnostic window. * No cooperation from the Target is required for this. */ A_STATUS ar6000_ReadRegDiag(HIF_DEVICE *hifDevice, A_UINT32 *address, A_UINT32 *data) { return HIFDiagReadAccess(hifDevice, *address, data); } /* * Write to the AR6000 through its diagnostic window. * No cooperation from the Target is required for this. */ A_STATUS ar6000_WriteRegDiag(HIF_DEVICE *hifDevice, A_UINT32 *address, A_UINT32 *data) { return HIFDiagWriteAccess(hifDevice, *address, *data); } A_STATUS ar6000_ReadDataDiag(HIF_DEVICE *hifDevice, A_UINT32 address, A_UCHAR *data, A_UINT32 length) { A_UINT32 count; A_STATUS status = A_OK; for (count = 0; count < length; count += 4, address += 4) { if ((status = HIFDiagReadAccess(hifDevice, address, (A_UINT32 *)&data[count])) != A_OK) { break; } } return status; } A_STATUS ar6000_WriteDataDiag(HIF_DEVICE *hifDevice, A_UINT32 address, A_UCHAR *data, A_UINT32 length) { A_UINT32 count; A_STATUS status = A_OK; for (count = 0; count < length; count += 4, address += 4) { if ((status = HIFDiagWriteAccess(hifDevice, address, *((A_UINT32 *)&data[count]))) != A_OK) { break; } } return status; } #ifdef HIF_MESSAGE_BASED /* TODO : for message based interfaces there are no HOST registers. These are just * stub functions for now */ A_STATUS ar6k_ReadTargetRegister(HIF_DEVICE *hifDevice, int regsel, A_UINT32 *regval) { /* TODO */ return A_ERROR; } void ar6k_FetchTargetRegs(HIF_DEVICE *hifDevice, A_UINT32 *targregs) { int i; for (i=0; i REGISTER_DUMP_LEN_MAX #error "REG_DUMP_COUNT_AR6001 too large" #endif #if REG_DUMP_COUNT_AR6002 > REGISTER_DUMP_LEN_MAX #error "REG_DUMP_COUNT_AR6002 too large" #endif #if REG_DUMP_COUNT_AR6003 > REGISTER_DUMP_LEN_MAX #error "REG_DUMP_COUNT_AR6003 too large" #endif #if REG_DUMP_COUNT_MCKINLEY > REGISTER_DUMP_LEN_MAX #error "REG_DUMP_COUNT_MCKINLEY too large" #endif void ar6000_dump_target_assert_info(HIF_DEVICE *hifDevice, A_UINT32 TargetType) { A_UINT32 address; A_UINT32 regDumpArea = 0; A_STATUS status; A_UINT32 regDumpValues[REGISTER_DUMP_LEN_MAX]; A_UINT32 regDumpCount = 0; A_UINT32 i; do { /* the reg dump pointer is copied to the host interest area */ address = HOST_INTEREST_ITEM_ADDRESS(TargetType, hi_failure_state); address = TARG_VTOP(TargetType, address); if (TargetType == TARGET_TYPE_AR6002) { regDumpCount = REG_DUMP_COUNT_AR6002; } else if (TargetType == TARGET_TYPE_AR6003) { regDumpCount = REG_DUMP_COUNT_AR6003; } else if (TargetType == TARGET_TYPE_MCKINLEY) { regDumpCount = REG_DUMP_COUNT_MCKINLEY; } else { A_ASSERT(0); } /* read RAM location through diagnostic window */ status = ar6000_ReadRegDiag(hifDevice, &address, ®DumpArea); if (A_FAILED(status)) { AR_DEBUG_PRINTF(ATH_DEBUG_ERR,("AR6K: Failed to get ptr to register dump area \n")); break; } AR_DEBUG_PRINTF(ATH_DEBUG_ERR,("AR6K: Location of register dump data: 0x%X \n",regDumpArea)); if (regDumpArea == 0) { /* no reg dump */ break; } regDumpArea = TARG_VTOP(TargetType, regDumpArea); /* fetch register dump data */ status = ar6000_ReadDataDiag(hifDevice, regDumpArea, (A_UCHAR *)®DumpValues[0], regDumpCount * (sizeof(A_UINT32))); if (A_FAILED(status)) { AR_DEBUG_PRINTF(ATH_DEBUG_ERR,("AR6K: Failed to get register dump \n")); break; } AR_DEBUG_PRINTF(ATH_DEBUG_ERR,("AR6K: Register Dump: \n")); for (i = 0; i < regDumpCount; i = i + 4) { AR_DEBUG_PRINTF (ATH_DEBUG_ERR,(" %2.2d : "DUMPSTR"\n", i, DUMP2STR(®DumpValues[i]))); #ifdef UNDER_CE /* * For Every logPrintf() Open the File so that in case of Crashes * We will have until the Last Message Flushed on to the File * So use logPrintf Sparingly..!! */ tgtassertPrintf (ATH_DEBUG_TRC," %2.2d: "DUMPSTR"\n",i, DUMP2STR(®DumpValues[i])); #endif } } while (FALSE); } /* set HTC/Mbox operational parameters, this can only be called when the target is in the * BMI phase */ A_STATUS ar6000_set_htc_params(HIF_DEVICE *hifDevice, A_UINT32 TargetType, A_UINT32 MboxIsrYieldValue, A_UINT8 HtcControlBuffers) { A_STATUS status; A_UINT32 blocksizes[HTC_MAILBOX_NUM_MAX]; do { /* get the block sizes */ status = HIFConfigureDevice(hifDevice, HIF_DEVICE_GET_MBOX_BLOCK_SIZE, blocksizes, sizeof(blocksizes)); if (A_FAILED(status)) { AR_DEBUG_PRINTF(ATH_LOG_ERR,("Failed to get block size info from HIF layer...\n")); break; } /* note: we actually get the block size for mailbox 1, for SDIO the block * size on mailbox 0 is artificially set to 1 */ /* must be a power of 2 */ A_ASSERT((blocksizes[1] & (blocksizes[1] - 1)) == 0); if (HtcControlBuffers != 0) { /* set override for number of control buffers to use */ blocksizes[1] |= ((A_UINT32)HtcControlBuffers) << 16; } /* set the host interest area for the block size */ status = BMIWriteMemory(hifDevice, HOST_INTEREST_ITEM_ADDRESS(TargetType, hi_mbox_io_block_sz), (A_UCHAR *)&blocksizes[1], 4); if (A_FAILED(status)) { AR_DEBUG_PRINTF(ATH_LOG_ERR,("BMIWriteMemory for IO block size failed \n")); break; } AR_DEBUG_PRINTF(ATH_LOG_INF,("Block Size Set: %d (target address:0x%X)\n", blocksizes[1], HOST_INTEREST_ITEM_ADDRESS(TargetType, hi_mbox_io_block_sz))); if (MboxIsrYieldValue != 0) { /* set the host interest area for the mbox ISR yield limit */ status = BMIWriteMemory(hifDevice, HOST_INTEREST_ITEM_ADDRESS(TargetType, hi_mbox_isr_yield_limit), (A_UCHAR *)&MboxIsrYieldValue, 4); if (A_FAILED(status)) { AR_DEBUG_PRINTF(ATH_LOG_ERR,("BMIWriteMemory for yield limit failed \n")); break; } } } while (FALSE); return status; } static A_STATUS prepare_ar6002(HIF_DEVICE *hifDevice, A_UINT32 TargetVersion) { A_STATUS status = A_OK; /* placeholder */ return status; } static A_STATUS prepare_ar6003(HIF_DEVICE *hifDevice, A_UINT32 TargetVersion) { A_STATUS status = A_OK; /* placeholder */ return status; } static A_STATUS prepare_MCKINLEY(HIF_DEVICE *hifDevice, A_UINT32 TargetVersion) { A_STATUS status = A_OK; A_UINT32 value = 0; /* force the setting to disable sleep for Bringup FIXME_MK */ value |= WLAN_SYSTEM_SLEEP_DISABLE_MASK; status = BMIWriteMemory(hifDevice, HOST_INTEREST_ITEM_ADDRESS(TARGET_TYPE_AR6002, hi_system_sleep_setting), (A_UCHAR *)&value, 4); return status; } /* this function assumes the caller has already initialized the BMI APIs */ A_STATUS ar6000_prepare_target(HIF_DEVICE *hifDevice, A_UINT32 TargetType, A_UINT32 TargetVersion) { if (TargetType == TARGET_TYPE_AR6002) { /* do any preparations for AR6002 devices */ return prepare_ar6002(hifDevice,TargetVersion); } else if (TargetType == TARGET_TYPE_AR6003) { return prepare_ar6003(hifDevice,TargetVersion); } else if (TargetType == TARGET_TYPE_MCKINLEY) { return prepare_MCKINLEY(hifDevice,TargetVersion); } return A_OK; } #if defined(CONFIG_AR6002_REV1_FORCE_HOST) /* * Call this function just before the call to BMIInit * in order to force* AR6002 rev 1.x firmware to detect a Host. * THIS IS FOR USE ONLY WITH AR6002 REV 1.x. * TBDXXX: Remove this function when REV 1.x is desupported. */ A_STATUS ar6002_REV1_reset_force_host (HIF_DEVICE *hifDevice) { A_INT32 i; struct forceROM_s { A_UINT32 addr; A_UINT32 data; }; struct forceROM_s *ForceROM; A_INT32 szForceROM; A_STATUS status = A_OK; A_UINT32 address; A_UINT32 data; /* Force AR6002 REV1.x to recognize Host presence. * * Note: Use RAM at 0x52df80..0x52dfa0 with ROM Remap entry 0 * so that this workaround functions with AR6002.war1.sh. We * could fold that entire workaround into this one, but it's not * worth the effort at this point. This workaround cannot be * merged into the other workaround because this must be done * before BMI. */ static struct forceROM_s ForceROM_NEW[] = { {0x52df80, 0x20f31c07}, {0x52df84, 0x92374420}, {0x52df88, 0x1d120c03}, {0x52df8c, 0xff8216f0}, {0x52df90, 0xf01d120c}, {0x52df94, 0x81004136}, {0x52df98, 0xbc9100bd}, {0x52df9c, 0x00bba100}, {0x00008000|MC_TCAM_TARGET_ADDRESS, 0x0012dfe0}, /* Use remap entry 0 */ {0x00008000|MC_TCAM_COMPARE_ADDRESS, 0x000e2380}, {0x00008000|MC_TCAM_MASK_ADDRESS, 0x00000000}, {0x00008000|MC_TCAM_VALID_ADDRESS, 0x00000001}, {0x00018000|(LOCAL_COUNT_ADDRESS+0x10), 0}, /* clear BMI credit counter */ {0x00004000|AR6002_RESET_CONTROL_ADDRESS, RESET_CONTROL_WARM_RST_MASK}, }; address = 0x004ed4b0; /* REV1 target software ID is stored here */ status = ar6000_ReadRegDiag(hifDevice, &address, &data); if (A_FAILED(status) || (data != AR6002_VERSION_REV1)) { return A_ERROR; /* Not AR6002 REV1 */ } ForceROM = ForceROM_NEW; szForceROM = sizeof(ForceROM_NEW)/sizeof(*ForceROM); ATH_DEBUG_PRINTF (DBG_MISC_DRV, ATH_DEBUG_TRC, ("Force Target to recognize Host....\n")); for (i = 0; i < szForceROM; i++) { if (ar6000_WriteRegDiag(hifDevice, &ForceROM[i].addr, &ForceROM[i].data) != A_OK) { ATH_DEBUG_PRINTF (DBG_MISC_DRV, ATH_DEBUG_TRC, ("Cannot force Target to recognize Host!\n")); return A_ERROR; } } A_MDELAY(1000); return A_OK; } #endif /* CONFIG_AR6002_REV1_FORCE_HOST */ void DebugDumpBytes(A_UCHAR *buffer, A_UINT16 length, char *pDescription) { A_CHAR stream[60]; A_CHAR byteOffsetStr[10]; A_UINT32 i; A_UINT16 offset, count, byteOffset; A_PRINTF("<---------Dumping %d Bytes : %s ------>\n", length, pDescription); count = 0; offset = 0; byteOffset = 0; for(i = 0; i < length; i++) { A_SPRINTF(stream + offset, "%2.2X ", buffer[i]); count ++; offset += 3; if(count == 16) { count = 0; offset = 0; A_SPRINTF(byteOffsetStr,"%4.4X",byteOffset); A_PRINTF("[%s]: %s\n", byteOffsetStr, stream); A_MEMZERO(stream, 60); byteOffset += 16; } } if(offset != 0) { A_SPRINTF(byteOffsetStr,"%4.4X",byteOffset); A_PRINTF("[%s]: %s\n", byteOffsetStr, stream); } A_PRINTF("<------------------------------------------------->\n"); } void a_dump_module_debug_info(ATH_DEBUG_MODULE_DBG_INFO *pInfo) { int i; ATH_DEBUG_MASK_DESCRIPTION *pDesc; if (pInfo == NULL) { return; } pDesc = pInfo->pMaskDescriptions; A_PRINTF("========================================================\n\n"); A_PRINTF("Module Debug Info => Name : %s \n", pInfo->ModuleName); A_PRINTF(" => Descr. : %s \n", pInfo->ModuleDescription); A_PRINTF("\n Current mask => 0x%8.8X \n", pInfo->CurrentMask); A_PRINTF("\n Avail. Debug Masks :\n\n"); for (i = 0; i < pInfo->MaxDescriptions; i++,pDesc++) { A_PRINTF(" => 0x%8.8X -- %s \n", pDesc->Mask, pDesc->Description); } if (0 == i) { A_PRINTF(" => * none defined * \n"); } A_PRINTF("\n Standard Debug Masks :\n\n"); /* print standard masks */ A_PRINTF(" => 0x%8.8X -- Errors \n", ATH_DEBUG_ERR); A_PRINTF(" => 0x%8.8X -- Warnings \n", ATH_DEBUG_WARN); A_PRINTF(" => 0x%8.8X -- Informational \n", ATH_DEBUG_INFO); A_PRINTF(" => 0x%8.8X -- Tracing \n", ATH_DEBUG_TRC); A_PRINTF("\n========================================================\n"); } static ATH_DEBUG_MODULE_DBG_INFO *FindModule(A_CHAR *module_name) { ATH_DEBUG_MODULE_DBG_INFO *pInfo = g_pModuleInfoHead; if (!g_ModuleDebugInit) { return NULL; } while (pInfo != NULL) { /* TODO: need to use something other than strlen */ if (A_MEMCMP(pInfo->ModuleName,module_name,strlen(module_name)) == 0) { break; } pInfo = pInfo->pNext; } return pInfo; } void a_register_module_debug_info(ATH_DEBUG_MODULE_DBG_INFO *pInfo) { if (!g_ModuleDebugInit) { return; } A_MUTEX_LOCK(&g_ModuleListLock); if (!(pInfo->Flags & ATH_DEBUG_INFO_FLAGS_REGISTERED)) { if (g_pModuleInfoHead == NULL) { g_pModuleInfoHead = pInfo; } else { pInfo->pNext = g_pModuleInfoHead; g_pModuleInfoHead = pInfo; } pInfo->Flags |= ATH_DEBUG_INFO_FLAGS_REGISTERED; } A_MUTEX_UNLOCK(&g_ModuleListLock); } void a_dump_module_debug_info_by_name(A_CHAR *module_name) { ATH_DEBUG_MODULE_DBG_INFO *pInfo = g_pModuleInfoHead; if (!g_ModuleDebugInit) { return; } if (A_MEMCMP(module_name,"all",3) == 0) { /* dump all */ while (pInfo != NULL) { a_dump_module_debug_info(pInfo); pInfo = pInfo->pNext; } return; } pInfo = FindModule(module_name); if (pInfo != NULL) { a_dump_module_debug_info(pInfo); } } A_STATUS a_get_module_mask(A_CHAR *module_name, A_UINT32 *pMask) { ATH_DEBUG_MODULE_DBG_INFO *pInfo = FindModule(module_name); if (NULL == pInfo) { return A_ERROR; } *pMask = pInfo->CurrentMask; return A_OK; } A_STATUS a_set_module_mask(A_CHAR *module_name, A_UINT32 Mask) { ATH_DEBUG_MODULE_DBG_INFO *pInfo = FindModule(module_name); if (NULL == pInfo) { return A_ERROR; } pInfo->CurrentMask = Mask; A_PRINTF("Module %s, new mask: 0x%8.8X \n",module_name,pInfo->CurrentMask); return A_OK; } void a_module_debug_support_init(void) { if (g_ModuleDebugInit) { return; } A_MUTEX_INIT(&g_ModuleListLock); g_pModuleInfoHead = NULL; g_ModuleDebugInit = TRUE; A_REGISTER_MODULE_DEBUG_INFO(misc); } void a_module_debug_support_cleanup(void) { ATH_DEBUG_MODULE_DBG_INFO *pInfo = g_pModuleInfoHead; ATH_DEBUG_MODULE_DBG_INFO *pCur; if (!g_ModuleDebugInit) { return; } g_ModuleDebugInit = FALSE; A_MUTEX_LOCK(&g_ModuleListLock); while (pInfo != NULL) { pCur = pInfo; pInfo = pInfo->pNext; pCur->pNext = NULL; /* clear registered flag */ pCur->Flags &= ~ATH_DEBUG_INFO_FLAGS_REGISTERED; } A_MUTEX_UNLOCK(&g_ModuleListLock); A_MUTEX_DELETE(&g_ModuleListLock); g_pModuleInfoHead = NULL; } /* can only be called during bmi init stage */ A_STATUS ar6000_set_hci_bridge_flags(HIF_DEVICE *hifDevice, A_UINT32 TargetType, A_UINT32 Flags) { A_STATUS status = A_OK; do { if ((TargetType != TARGET_TYPE_AR6003) && (TargetType != TARGET_TYPE_MCKINLEY)) { AR_DEBUG_PRINTF(ATH_DEBUG_WARN, ("Target Type:%d, does not support HCI bridging! \n", TargetType)); break; } /* set hci bridge flags */ status = BMIWriteMemory(hifDevice, HOST_INTEREST_ITEM_ADDRESS(TargetType, hi_hci_bridge_flags), (A_UCHAR *)&Flags, 4); } while (FALSE); return status; }