123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431 |
- /*
- * Utility routines for configuring different memories in Broadcom chips.
- *
- * Portions of this code are copyright (c) 2020 Cypress Semiconductor Corporation
- *
- * Copyright (C) 1999-2020, Broadcom Corporation
- *
- * Unless you and Broadcom execute a separate written software license
- * agreement governing use of this software, this software is licensed to you
- * under the terms of the GNU General Public License version 2 (the "GPL"),
- * available at http://www.broadcom.com/licenses/GPLv2.php, with the
- * following added to such license:
- *
- * As a special exception, the copyright holders of this software give you
- * permission to link this software with independent modules, and to copy and
- * distribute the resulting executable under terms of your choice, provided that
- * you also meet, for each linked independent module, the terms and conditions of
- * the license of that module. An independent module is a module which is not
- * derived from this software. The special exception does not apply to any
- * modifications of the software.
- *
- * Notwithstanding the above, under no circumstances may you combine this
- * software in any way with any other Broadcom software provided under a license
- * other than the GPL, without Broadcom's express prior written consent.
- *
- *
- * <<Broadcom-WL-IPTag/Open:>>
- *
- * $Id: $
- */
- #include <typedefs.h>
- #include <sbchipc.h>
- #include <hndsoc.h>
- #include <bcmdevs.h>
- #include <osl.h>
- #include <sbgci.h>
- #include <siutils.h>
- #include <bcmutils.h>
- #include <hndmem.h>
- #define IS_MEMTYPE_VALID(mem) ((mem >= MEM_SOCRAM) && (mem < MEM_MAX))
- #define IS_MEMCONFIG_VALID(cfg) ((cfg >= PDA_CONFIG_CLEAR) && (cfg < PDA_CONFIG_MAX))
- /* Returns the number of banks in a given memory */
- int
- hndmem_num_banks(si_t *sih, int mem)
- {
- uint32 savecore, mem_info;
- int num_banks = 0;
- gciregs_t *gciregs;
- osl_t *osh = si_osh(sih);
- if (!IS_MEMTYPE_VALID(mem)) {
- goto exit;
- }
- savecore = si_coreidx(sih);
- /* TODO: Check whether SOCRAM core is present or not. If not, bail out */
- /* In future we need to add code for TCM based chips as well */
- if (!si_setcore(sih, SOCRAM_CORE_ID, 0)) {
- goto exit;
- }
- if (sih->gcirev >= 9) {
- gciregs = si_setcore(sih, GCI_CORE_ID, 0);
- mem_info = R_REG(osh, &gciregs->wlan_mem_info);
- switch (mem) {
- case MEM_SOCRAM:
- num_banks = (mem_info & WLAN_MEM_INFO_REG_NUMSOCRAMBANKS_MASK) >>
- WLAN_MEM_INFO_REG_NUMSOCRAMBANKS_SHIFT;
- break;
- case MEM_BM:
- num_banks = (mem_info & WLAN_MEM_INFO_REG_NUMD11MACBM_MASK) >>
- WLAN_MEM_INFO_REG_NUMD11MACBM_SHIFT;
- break;
- case MEM_UCM:
- num_banks = (mem_info & WLAN_MEM_INFO_REG_NUMD11MACUCM_MASK) >>
- WLAN_MEM_INFO_REG_NUMD11MACUCM_SHIFT;
- break;
- case MEM_SHM:
- num_banks = (mem_info & WLAN_MEM_INFO_REG_NUMD11MACSHM_MASK) >>
- WLAN_MEM_INFO_REG_NUMD11MACSHM_SHIFT;
- break;
- default:
- ASSERT(0);
- break;
- }
- } else {
- /* TODO: Figure out bank information using SOCRAM registers */
- }
- si_setcoreidx(sih, savecore);
- exit:
- return num_banks;
- }
- /* Returns the size of a give bank in a given memory */
- int
- hndmem_bank_size(si_t *sih, hndmem_type_t mem, int bank_num)
- {
- uint32 savecore, bank_info, reg_data;
- int bank_sz = 0;
- gciregs_t *gciregs;
- osl_t *osh = si_osh(sih);
- if (!IS_MEMTYPE_VALID(mem)) {
- goto exit;
- }
- savecore = si_coreidx(sih);
- /* TODO: Check whether SOCRAM core is present or not. If not, bail out */
- /* In future we need to add code for TCM based chips as well */
- if (!si_setcore(sih, SOCRAM_CORE_ID, 0)) {
- goto exit;
- }
- if (sih->gcirev >= 9) {
- gciregs = si_setcore(sih, GCI_CORE_ID, 0);
- reg_data = ((mem &
- GCI_INDIRECT_ADDRESS_REG_GPIOINDEX_MASK) <<
- GCI_INDIRECT_ADDRESS_REG_GPIOINDEX_SHIFT) |
- ((bank_num & GCI_INDIRECT_ADDRESS_REG_REGINDEX_MASK)
- << GCI_INDIRECT_ADDRESS_REG_REGINDEX_SHIFT);
- W_REG(osh, &gciregs->gci_indirect_addr, reg_data);
- bank_info = R_REG(osh, &gciregs->wlan_bankxinfo);
- bank_sz = (bank_info & WLAN_BANKXINFO_BANK_SIZE_MASK) >>
- WLAN_BANKXINFO_BANK_SIZE_SHIFT;
- } else {
- /* TODO: Figure out bank size using SOCRAM registers */
- }
- si_setcoreidx(sih, savecore);
- exit:
- return bank_sz;
- }
- /* Returns the start address of given memory */
- uint32
- hndmem_mem_base(si_t *sih, hndmem_type_t mem)
- {
- uint32 savecore, base_addr = 0;
- /* Currently only support of SOCRAM is available in hardware */
- if (mem != MEM_SOCRAM) {
- goto exit;
- }
- savecore = si_coreidx(sih);
- if (si_setcore(sih, SOCRAM_CORE_ID, 0))
- {
- base_addr = si_get_slaveport_addr(sih, CORE_SLAVE_PORT_1,
- CORE_BASE_ADDR_0, SOCRAM_CORE_ID, 0);
- } else {
- /* TODO: Add code to get the base address of TCM */
- base_addr = 0;
- }
- si_setcoreidx(sih, savecore);
- exit:
- return base_addr;
- }
- #ifdef BCMDEBUG
- char *hndmem_type_str[] =
- {
- "SOCRAM", /* 0 */
- "BM", /* 1 */
- "UCM", /* 2 */
- "SHM", /* 3 */
- };
- /* Dumps the complete memory information */
- void
- hndmem_dump_meminfo_all(si_t *sih)
- {
- int mem, bank, bank_cnt, bank_sz;
- for (mem = MEM_SOCRAM; mem < MEM_MAX; mem++) {
- bank_cnt = hndmem_num_banks(sih, mem);
- printf("\nMemtype: %s\n", hndmem_type_str[mem]);
- for (bank = 0; bank < bank_cnt; bank++) {
- bank_sz = hndmem_bank_size(sih, mem, bank);
- printf("Bank-%d: %d KB\n", bank, bank_sz);
- }
- }
- }
- #endif /* BCMDEBUG */
- /* Configures the Sleep PDA for a particular bank for a given memory type */
- int
- hndmem_sleeppda_bank_config(si_t *sih, hndmem_type_t mem, int bank_num,
- hndmem_config_t config, uint32 pda)
- {
- uint32 savecore, reg_data;
- gciregs_t *gciregs;
- int err = BCME_OK;
- osl_t *osh = si_osh(sih);
- /* TODO: Check whether SOCRAM core is present or not. If not, bail out */
- /* In future we need to add code for TCM based chips as well */
- if (!si_setcore(sih, SOCRAM_CORE_ID, 0)) {
- err = BCME_UNSUPPORTED;
- goto exit;
- }
- /* Sleep PDA is supported only by GCI rev >= 9 */
- if (sih->gcirev < 9) {
- err = BCME_UNSUPPORTED;
- goto exit;
- }
- if (!IS_MEMTYPE_VALID(mem)) {
- err = BCME_BADOPTION;
- goto exit;
- }
- if (!IS_MEMCONFIG_VALID(config)) {
- err = BCME_BADOPTION;
- goto exit;
- }
- savecore = si_coreidx(sih);
- gciregs = si_setcore(sih, GCI_CORE_ID, 0);
- reg_data = ((mem &
- GCI_INDIRECT_ADDRESS_REG_GPIOINDEX_MASK) <<
- GCI_INDIRECT_ADDRESS_REG_GPIOINDEX_SHIFT) |
- ((bank_num & GCI_INDIRECT_ADDRESS_REG_REGINDEX_MASK)
- << GCI_INDIRECT_ADDRESS_REG_REGINDEX_SHIFT);
- W_REG(osh, &gciregs->gci_indirect_addr, reg_data);
- if (config == PDA_CONFIG_SET_PARTIAL) {
- W_REG(osh, &gciregs->wlan_bankxsleeppda, pda);
- W_REG(osh, &gciregs->wlan_bankxkill, 0);
- }
- else if (config == PDA_CONFIG_SET_FULL) {
- W_REG(osh, &gciregs->wlan_bankxsleeppda, WLAN_BANKX_SLEEPPDA_REG_SLEEPPDA_MASK);
- W_REG(osh, &gciregs->wlan_bankxkill, WLAN_BANKX_PKILL_REG_SLEEPPDA_MASK);
- } else {
- W_REG(osh, &gciregs->wlan_bankxsleeppda, 0);
- W_REG(osh, &gciregs->wlan_bankxkill, 0);
- }
- si_setcoreidx(sih, savecore);
- exit:
- return err;
- }
- /* Configures the Active PDA for a particular bank for a given memory type */
- int
- hndmem_activepda_bank_config(si_t *sih, hndmem_type_t mem,
- int bank_num, hndmem_config_t config, uint32 pda)
- {
- uint32 savecore, reg_data;
- gciregs_t *gciregs;
- int err = BCME_OK;
- osl_t *osh = si_osh(sih);
- if (!IS_MEMTYPE_VALID(mem)) {
- err = BCME_BADOPTION;
- goto exit;
- }
- if (!IS_MEMCONFIG_VALID(config)) {
- err = BCME_BADOPTION;
- goto exit;
- }
- savecore = si_coreidx(sih);
- /* TODO: Check whether SOCRAM core is present or not. If not, bail out */
- /* In future we need to add code for TCM based chips as well */
- if (!si_setcore(sih, SOCRAM_CORE_ID, 0)) {
- err = BCME_UNSUPPORTED;
- goto exit;
- }
- if (sih->gcirev >= 9) {
- gciregs = si_setcore(sih, GCI_CORE_ID, 0);
- reg_data = ((mem &
- GCI_INDIRECT_ADDRESS_REG_GPIOINDEX_MASK) <<
- GCI_INDIRECT_ADDRESS_REG_GPIOINDEX_SHIFT) |
- ((bank_num & GCI_INDIRECT_ADDRESS_REG_REGINDEX_MASK)
- << GCI_INDIRECT_ADDRESS_REG_REGINDEX_SHIFT);
- W_REG(osh, &gciregs->gci_indirect_addr, reg_data);
- if (config == PDA_CONFIG_SET_PARTIAL) {
- W_REG(osh, &gciregs->wlan_bankxactivepda, pda);
- }
- else if (config == PDA_CONFIG_SET_FULL) {
- W_REG(osh, &gciregs->wlan_bankxactivepda,
- WLAN_BANKX_SLEEPPDA_REG_SLEEPPDA_MASK);
- } else {
- W_REG(osh, &gciregs->wlan_bankxactivepda, 0);
- }
- } else {
- /* TODO: Configure SOCRAM PDA using SOCRAM registers */
- err = BCME_UNSUPPORTED;
- }
- si_setcoreidx(sih, savecore);
- exit:
- return err;
- }
- /* Configures the Sleep PDA for all the banks for a given memory type */
- int
- hndmem_sleeppda_config(si_t *sih, hndmem_type_t mem, hndmem_config_t config)
- {
- int bank;
- int num_banks = hndmem_num_banks(sih, mem);
- int err = BCME_OK;
- /* Sleep PDA is supported only by GCI rev >= 9 */
- if (sih->gcirev < 9) {
- err = BCME_UNSUPPORTED;
- goto exit;
- }
- if (!IS_MEMTYPE_VALID(mem)) {
- err = BCME_BADOPTION;
- goto exit;
- }
- if (!IS_MEMCONFIG_VALID(config)) {
- err = BCME_BADOPTION;
- goto exit;
- }
- for (bank = 0; bank < num_banks; bank++)
- {
- err = hndmem_sleeppda_bank_config(sih, mem, bank, config, 0);
- }
- exit:
- return err;
- }
- /* Configures the Active PDA for all the banks for a given memory type */
- int
- hndmem_activepda_config(si_t *sih, hndmem_type_t mem, hndmem_config_t config)
- {
- int bank;
- int num_banks = hndmem_num_banks(sih, mem);
- int err = BCME_OK;
- if (!IS_MEMTYPE_VALID(mem)) {
- err = BCME_BADOPTION;
- goto exit;
- }
- if (!IS_MEMCONFIG_VALID(config)) {
- err = BCME_BADOPTION;
- goto exit;
- }
- for (bank = 0; bank < num_banks; bank++)
- {
- err = hndmem_activepda_bank_config(sih, mem, bank, config, 0);
- }
- exit:
- return err;
- }
- /* Turn off/on all the possible banks in a given memory range.
- * Currently this works only for SOCRAM as this is restricted by HW.
- */
- int
- hndmem_activepda_mem_config(si_t *sih, hndmem_type_t mem, uint32 mem_start,
- uint32 size, hndmem_config_t config)
- {
- int bank, bank_sz, num_banks;
- int mem_end;
- int bank_start_addr, bank_end_addr;
- int err = BCME_OK;
- /* We can get bank size for only SOCRAM/TCM only. Support is not avilable
- * for other memories (BM, UCM and SHM)
- */
- if (mem != MEM_SOCRAM) {
- err = BCME_UNSUPPORTED;
- goto exit;
- }
- num_banks = hndmem_num_banks(sih, mem);
- bank_start_addr = hndmem_mem_base(sih, mem);
- mem_end = mem_start + size - 1;
- for (bank = 0; bank < num_banks; bank++)
- {
- /* Bank size is spcified in bankXinfo register in terms on KBs */
- bank_sz = 1024 * hndmem_bank_size(sih, mem, bank);
- bank_end_addr = bank_start_addr + bank_sz - 1;
- if (config == PDA_CONFIG_SET_FULL) {
- /* Check if the bank is completely overlapping with the given mem range */
- if ((mem_start <= bank_start_addr) && (mem_end >= bank_end_addr)) {
- err = hndmem_activepda_bank_config(sih, mem, bank, config, 0);
- }
- } else {
- /* Check if the bank is completely overlaped with the given mem range */
- if (((mem_start <= bank_start_addr) && (mem_end >= bank_end_addr)) ||
- /* Check if the bank is partially overlaped with the given range */
- ((mem_start <= bank_end_addr) && (mem_end >= bank_start_addr))) {
- err = hndmem_activepda_bank_config(sih, mem, bank, config, 0);
- }
- }
- bank_start_addr += bank_sz;
- }
- exit:
- return err;
- }
|