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;
 | |
| }
 | 
