430 lines
11 KiB
C
Executable File
430 lines
11 KiB
C
Executable File
/*
|
|
* Utility routines for configuring different memories in Broadcom chips.
|
|
*
|
|
* Copyright (C) 1999-2019, Broadcom.
|
|
*
|
|
* 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;
|
|
}
|