mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	net: wireless: add brcm80211 drivers
Add the brcm80211 tree to drivers/net/wireless, and disable the version that's in drivers/staging. This version includes the sources currently in staging, plus any changes that have been sent out for review. Sources in staging will be deleted in a followup patch. Signed-off-by: Arend van Spriel <arend@broadcom.com> Signed-off-by: John W. Linville <linville@tuxdriver.com>
This commit is contained in:
		
							parent
							
								
									5f68a2b0a8
								
							
						
					
					
						commit
						5b435de0d7
					
				| @ -271,6 +271,7 @@ config MWL8K | |||||||
| source "drivers/net/wireless/ath/Kconfig" | source "drivers/net/wireless/ath/Kconfig" | ||||||
| source "drivers/net/wireless/b43/Kconfig" | source "drivers/net/wireless/b43/Kconfig" | ||||||
| source "drivers/net/wireless/b43legacy/Kconfig" | source "drivers/net/wireless/b43legacy/Kconfig" | ||||||
|  | source "drivers/net/wireless/brcm80211/Kconfig" | ||||||
| source "drivers/net/wireless/hostap/Kconfig" | source "drivers/net/wireless/hostap/Kconfig" | ||||||
| source "drivers/net/wireless/ipw2x00/Kconfig" | source "drivers/net/wireless/ipw2x00/Kconfig" | ||||||
| source "drivers/net/wireless/iwlwifi/Kconfig" | source "drivers/net/wireless/iwlwifi/Kconfig" | ||||||
|  | |||||||
| @ -58,3 +58,6 @@ obj-$(CONFIG_WL12XX_PLATFORM_DATA)	+= wl12xx/ | |||||||
| obj-$(CONFIG_IWM)	+= iwmc3200wifi/ | obj-$(CONFIG_IWM)	+= iwmc3200wifi/ | ||||||
| 
 | 
 | ||||||
| obj-$(CONFIG_MWIFIEX)	+= mwifiex/ | obj-$(CONFIG_MWIFIEX)	+= mwifiex/ | ||||||
|  | obj-$(CONFIG_BRCMFMAC) += brcm80211/ | ||||||
|  | obj-$(CONFIG_BRCMUMAC) += brcm80211/ | ||||||
|  | obj-$(CONFIG_BRCMSMAC) += brcm80211/ | ||||||
|  | |||||||
							
								
								
									
										35
									
								
								drivers/net/wireless/brcm80211/Kconfig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								drivers/net/wireless/brcm80211/Kconfig
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,35 @@ | |||||||
|  | config BRCMUTIL | ||||||
|  | 	tristate | ||||||
|  | 
 | ||||||
|  | config BRCMSMAC | ||||||
|  | 	tristate "Broadcom IEEE802.11n PCIe SoftMAC WLAN driver" | ||||||
|  | 	depends on PCI | ||||||
|  | 	depends on MAC80211 | ||||||
|  | 	depends on BCMA=n | ||||||
|  | 	select BRCMUTIL | ||||||
|  | 	select FW_LOADER | ||||||
|  | 	select CRC_CCITT | ||||||
|  | 	select CRC8 | ||||||
|  | 	select CORDIC | ||||||
|  | 	---help--- | ||||||
|  | 	  This module adds support for PCIe wireless adapters based on Broadcom | ||||||
|  | 	  IEEE802.11n SoftMAC chipsets.  If you choose to build a module, it'll | ||||||
|  | 	  be called brcmsmac.ko. | ||||||
|  | 
 | ||||||
|  | config BRCMFMAC | ||||||
|  | 	tristate "Broadcom IEEE802.11n embedded FullMAC WLAN driver" | ||||||
|  | 	depends on MMC | ||||||
|  | 	depends on CFG80211 | ||||||
|  | 	select BRCMUTIL | ||||||
|  | 	select FW_LOADER | ||||||
|  | 	---help--- | ||||||
|  | 	  This module adds support for embedded wireless adapters based on | ||||||
|  | 	  Broadcom IEEE802.11n FullMAC chipsets.  This driver uses the kernel's | ||||||
|  | 	  wireless extensions subsystem.  If you choose to build a module, | ||||||
|  | 	  it'll be called brcmfmac.ko. | ||||||
|  | 
 | ||||||
|  | config BRCMDBG | ||||||
|  | 	bool "Broadcom driver debug functions" | ||||||
|  | 	depends on BRCMSMAC || BRCMFMAC | ||||||
|  | 	---help--- | ||||||
|  | 	  Selecting this enables additional code for debug purposes. | ||||||
							
								
								
									
										23
									
								
								drivers/net/wireless/brcm80211/Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								drivers/net/wireless/brcm80211/Makefile
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,23 @@ | |||||||
|  | #
 | ||||||
|  | # Makefile fragment for Broadcom 802.11n Networking Device Driver
 | ||||||
|  | #
 | ||||||
|  | # Copyright (c) 2010 Broadcom Corporation
 | ||||||
|  | #
 | ||||||
|  | # 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.
 | ||||||
|  | 
 | ||||||
|  | # common flags
 | ||||||
|  | subdir-ccflags-$(CONFIG_BRCMDBG)	+= -DBCMDBG | ||||||
|  | 
 | ||||||
|  | obj-$(CONFIG_BRCMUTIL)	+= brcmutil/ | ||||||
|  | obj-$(CONFIG_BRCMFMAC)	+= brcmfmac/ | ||||||
|  | obj-$(CONFIG_BRCMSMAC)	+= brcmsmac/ | ||||||
							
								
								
									
										33
									
								
								drivers/net/wireless/brcm80211/brcmfmac/Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								drivers/net/wireless/brcm80211/brcmfmac/Makefile
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,33 @@ | |||||||
|  | #
 | ||||||
|  | # Makefile fragment for Broadcom 802.11n Networking Device Driver
 | ||||||
|  | #
 | ||||||
|  | # Copyright (c) 2010 Broadcom Corporation
 | ||||||
|  | #
 | ||||||
|  | # 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.
 | ||||||
|  | 
 | ||||||
|  | ccflags-y += \
 | ||||||
|  | 	-Idrivers/net/wireless/brcm80211/brcmfmac	\
 | ||||||
|  | 	-Idrivers/net/wireless/brcm80211/include | ||||||
|  | 
 | ||||||
|  | DHDOFILES = \
 | ||||||
|  | 	wl_cfg80211.o \
 | ||||||
|  | 	dhd_cdc.o \
 | ||||||
|  | 	dhd_common.o \
 | ||||||
|  | 	dhd_sdio.o	\
 | ||||||
|  | 	dhd_linux.o \
 | ||||||
|  | 	bcmsdh.o \
 | ||||||
|  | 	bcmsdh_sdmmc.o | ||||||
|  | 
 | ||||||
|  | obj-$(CONFIG_BRCMFMAC) += brcmfmac.o | ||||||
|  | brcmfmac-objs += $(DHDOFILES) | ||||||
|  | ccflags-y += -D__CHECK_ENDIAN__ | ||||||
							
								
								
									
										32
									
								
								drivers/net/wireless/brcm80211/brcmfmac/bcmchip.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								drivers/net/wireless/brcm80211/brcmfmac/bcmchip.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,32 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2011 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _bcmchip_h_ | ||||||
|  | #define _bcmchip_h_ | ||||||
|  | 
 | ||||||
|  | /* bcm4329 */ | ||||||
|  | /* SDIO device core, ID 0x829 */ | ||||||
|  | #define BCM4329_CORE_BUS_BASE		0x18011000 | ||||||
|  | /* internal memory core, ID 0x80e */ | ||||||
|  | #define BCM4329_CORE_SOCRAM_BASE	0x18003000 | ||||||
|  | /* ARM Cortex M3 core, ID 0x82a */ | ||||||
|  | #define BCM4329_CORE_ARM_BASE		0x18002000 | ||||||
|  | #define BCM4329_RAMSIZE			0x48000 | ||||||
|  | /* firmware name */ | ||||||
|  | #define BCM4329_FW_NAME			"brcm/bcm4329-fullmac-4.bin" | ||||||
|  | #define BCM4329_NV_NAME			"brcm/bcm4329-fullmac-4.txt" | ||||||
|  | 
 | ||||||
|  | #endif				/* _bcmchip_h_ */ | ||||||
							
								
								
									
										371
									
								
								drivers/net/wireless/brcm80211/brcmfmac/bcmsdh.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										371
									
								
								drivers/net/wireless/brcm80211/brcmfmac/bcmsdh.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,371 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | /* ****************** SDIO CARD Interface Functions **************************/ | ||||||
|  | 
 | ||||||
|  | #include <linux/types.h> | ||||||
|  | #include <linux/netdevice.h> | ||||||
|  | #include <linux/pci.h> | ||||||
|  | #include <linux/pci_ids.h> | ||||||
|  | #include <linux/sched.h> | ||||||
|  | #include <linux/completion.h> | ||||||
|  | #include <linux/mmc/sdio.h> | ||||||
|  | #include <linux/mmc/sdio_func.h> | ||||||
|  | #include <linux/mmc/card.h> | ||||||
|  | 
 | ||||||
|  | #include <defs.h> | ||||||
|  | #include <brcm_hw_ids.h> | ||||||
|  | #include <brcmu_utils.h> | ||||||
|  | #include <brcmu_wifi.h> | ||||||
|  | #include <soc.h> | ||||||
|  | #include "dhd.h" | ||||||
|  | #include "dhd_bus.h" | ||||||
|  | #include "dhd_dbg.h" | ||||||
|  | #include "sdio_host.h" | ||||||
|  | 
 | ||||||
|  | #define SDIOH_API_ACCESS_RETRY_LIMIT	2 | ||||||
|  | 
 | ||||||
|  | static void brcmf_sdioh_irqhandler(struct sdio_func *func) | ||||||
|  | { | ||||||
|  | 	struct brcmf_sdio_dev *sdiodev = dev_get_drvdata(&func->card->dev); | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(TRACE, "***IRQHandler\n"); | ||||||
|  | 
 | ||||||
|  | 	sdio_release_host(func); | ||||||
|  | 
 | ||||||
|  | 	brcmf_sdbrcm_isr(sdiodev->bus); | ||||||
|  | 
 | ||||||
|  | 	sdio_claim_host(func); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int brcmf_sdcard_intr_reg(struct brcmf_sdio_dev *sdiodev) | ||||||
|  | { | ||||||
|  | 	brcmf_dbg(TRACE, "Entering\n"); | ||||||
|  | 
 | ||||||
|  | 	sdio_claim_host(sdiodev->func[1]); | ||||||
|  | 	sdio_claim_irq(sdiodev->func[1], brcmf_sdioh_irqhandler); | ||||||
|  | 	sdio_release_host(sdiodev->func[1]); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int brcmf_sdcard_intr_dereg(struct brcmf_sdio_dev *sdiodev) | ||||||
|  | { | ||||||
|  | 	brcmf_dbg(TRACE, "Entering\n"); | ||||||
|  | 
 | ||||||
|  | 	sdio_claim_host(sdiodev->func[1]); | ||||||
|  | 	sdio_release_irq(sdiodev->func[1]); | ||||||
|  | 	sdio_release_host(sdiodev->func[1]); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | u8 brcmf_sdcard_cfg_read(struct brcmf_sdio_dev *sdiodev, uint fnc_num, u32 addr, | ||||||
|  | 			 int *err) | ||||||
|  | { | ||||||
|  | 	int status; | ||||||
|  | 	s32 retry = 0; | ||||||
|  | 	u8 data = 0; | ||||||
|  | 
 | ||||||
|  | 	do { | ||||||
|  | 		if (retry)	/* wait for 1 ms till bus get settled down */ | ||||||
|  | 			udelay(1000); | ||||||
|  | 		status = brcmf_sdioh_request_byte(sdiodev, SDIOH_READ, fnc_num, | ||||||
|  | 						  addr, (u8 *) &data); | ||||||
|  | 	} while (status != 0 | ||||||
|  | 		 && (retry++ < SDIOH_API_ACCESS_RETRY_LIMIT)); | ||||||
|  | 	if (err) | ||||||
|  | 		*err = status; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(INFO, "fun = %d, addr = 0x%x, u8data = 0x%x\n", | ||||||
|  | 		  fnc_num, addr, data); | ||||||
|  | 
 | ||||||
|  | 	return data; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | brcmf_sdcard_cfg_write(struct brcmf_sdio_dev *sdiodev, uint fnc_num, u32 addr, | ||||||
|  | 		       u8 data, int *err) | ||||||
|  | { | ||||||
|  | 	int status; | ||||||
|  | 	s32 retry = 0; | ||||||
|  | 
 | ||||||
|  | 	do { | ||||||
|  | 		if (retry)	/* wait for 1 ms till bus get settled down */ | ||||||
|  | 			udelay(1000); | ||||||
|  | 		status = brcmf_sdioh_request_byte(sdiodev, SDIOH_WRITE, fnc_num, | ||||||
|  | 						  addr, (u8 *) &data); | ||||||
|  | 	} while (status != 0 | ||||||
|  | 		 && (retry++ < SDIOH_API_ACCESS_RETRY_LIMIT)); | ||||||
|  | 	if (err) | ||||||
|  | 		*err = status; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(INFO, "fun = %d, addr = 0x%x, u8data = 0x%x\n", | ||||||
|  | 		  fnc_num, addr, data); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | brcmf_sdcard_set_sbaddr_window(struct brcmf_sdio_dev *sdiodev, u32 address) | ||||||
|  | { | ||||||
|  | 	int err = 0; | ||||||
|  | 	brcmf_sdcard_cfg_write(sdiodev, SDIO_FUNC_1, SBSDIO_FUNC1_SBADDRLOW, | ||||||
|  | 			 (address >> 8) & SBSDIO_SBADDRLOW_MASK, &err); | ||||||
|  | 	if (!err) | ||||||
|  | 		brcmf_sdcard_cfg_write(sdiodev, SDIO_FUNC_1, | ||||||
|  | 				       SBSDIO_FUNC1_SBADDRMID, | ||||||
|  | 				       (address >> 16) & SBSDIO_SBADDRMID_MASK, | ||||||
|  | 				       &err); | ||||||
|  | 	if (!err) | ||||||
|  | 		brcmf_sdcard_cfg_write(sdiodev, SDIO_FUNC_1, | ||||||
|  | 				       SBSDIO_FUNC1_SBADDRHIGH, | ||||||
|  | 				       (address >> 24) & SBSDIO_SBADDRHIGH_MASK, | ||||||
|  | 				       &err); | ||||||
|  | 
 | ||||||
|  | 	return err; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | u32 brcmf_sdcard_reg_read(struct brcmf_sdio_dev *sdiodev, u32 addr, uint size) | ||||||
|  | { | ||||||
|  | 	int status; | ||||||
|  | 	u32 word = 0; | ||||||
|  | 	uint bar0 = addr & ~SBSDIO_SB_OFT_ADDR_MASK; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(INFO, "fun = 1, addr = 0x%x\n", addr); | ||||||
|  | 
 | ||||||
|  | 	if (bar0 != sdiodev->sbwad) { | ||||||
|  | 		if (brcmf_sdcard_set_sbaddr_window(sdiodev, bar0)) | ||||||
|  | 			return 0xFFFFFFFF; | ||||||
|  | 
 | ||||||
|  | 		sdiodev->sbwad = bar0; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	addr &= SBSDIO_SB_OFT_ADDR_MASK; | ||||||
|  | 	if (size == 4) | ||||||
|  | 		addr |= SBSDIO_SB_ACCESS_2_4B_FLAG; | ||||||
|  | 
 | ||||||
|  | 	status = brcmf_sdioh_request_word(sdiodev, SDIOH_READ, SDIO_FUNC_1, | ||||||
|  | 					  addr, &word, size); | ||||||
|  | 
 | ||||||
|  | 	sdiodev->regfail = (status != 0); | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(INFO, "u32data = 0x%x\n", word); | ||||||
|  | 
 | ||||||
|  | 	/* if ok, return appropriately masked word */ | ||||||
|  | 	if (status == 0) { | ||||||
|  | 		switch (size) { | ||||||
|  | 		case sizeof(u8): | ||||||
|  | 			return word & 0xff; | ||||||
|  | 		case sizeof(u16): | ||||||
|  | 			return word & 0xffff; | ||||||
|  | 		case sizeof(u32): | ||||||
|  | 			return word; | ||||||
|  | 		default: | ||||||
|  | 			sdiodev->regfail = true; | ||||||
|  | 
 | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* otherwise, bad sdio access or invalid size */ | ||||||
|  | 	brcmf_dbg(ERROR, "error reading addr 0x%04x size %d\n", addr, size); | ||||||
|  | 	return 0xFFFFFFFF; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | u32 brcmf_sdcard_reg_write(struct brcmf_sdio_dev *sdiodev, u32 addr, uint size, | ||||||
|  | 			   u32 data) | ||||||
|  | { | ||||||
|  | 	int status; | ||||||
|  | 	uint bar0 = addr & ~SBSDIO_SB_OFT_ADDR_MASK; | ||||||
|  | 	int err = 0; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(INFO, "fun = 1, addr = 0x%x, uint%ddata = 0x%x\n", | ||||||
|  | 		  addr, size * 8, data); | ||||||
|  | 
 | ||||||
|  | 	if (bar0 != sdiodev->sbwad) { | ||||||
|  | 		err = brcmf_sdcard_set_sbaddr_window(sdiodev, bar0); | ||||||
|  | 		if (err) | ||||||
|  | 			return err; | ||||||
|  | 
 | ||||||
|  | 		sdiodev->sbwad = bar0; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	addr &= SBSDIO_SB_OFT_ADDR_MASK; | ||||||
|  | 	if (size == 4) | ||||||
|  | 		addr |= SBSDIO_SB_ACCESS_2_4B_FLAG; | ||||||
|  | 	status = | ||||||
|  | 	    brcmf_sdioh_request_word(sdiodev, SDIOH_WRITE, SDIO_FUNC_1, | ||||||
|  | 				     addr, &data, size); | ||||||
|  | 	sdiodev->regfail = (status != 0); | ||||||
|  | 
 | ||||||
|  | 	if (status == 0) | ||||||
|  | 		return 0; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(ERROR, "error writing 0x%08x to addr 0x%04x size %d\n", | ||||||
|  | 		  data, addr, size); | ||||||
|  | 	return 0xFFFFFFFF; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool brcmf_sdcard_regfail(struct brcmf_sdio_dev *sdiodev) | ||||||
|  | { | ||||||
|  | 	return sdiodev->regfail; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | brcmf_sdcard_recv_buf(struct brcmf_sdio_dev *sdiodev, u32 addr, uint fn, | ||||||
|  | 		      uint flags, | ||||||
|  | 		      u8 *buf, uint nbytes, struct sk_buff *pkt) | ||||||
|  | { | ||||||
|  | 	int status; | ||||||
|  | 	uint incr_fix; | ||||||
|  | 	uint width; | ||||||
|  | 	uint bar0 = addr & ~SBSDIO_SB_OFT_ADDR_MASK; | ||||||
|  | 	int err = 0; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(INFO, "fun = %d, addr = 0x%x, size = %d\n", fn, addr, nbytes); | ||||||
|  | 
 | ||||||
|  | 	/* Async not implemented yet */ | ||||||
|  | 	if (flags & SDIO_REQ_ASYNC) | ||||||
|  | 		return -ENOTSUPP; | ||||||
|  | 
 | ||||||
|  | 	if (bar0 != sdiodev->sbwad) { | ||||||
|  | 		err = brcmf_sdcard_set_sbaddr_window(sdiodev, bar0); | ||||||
|  | 		if (err) | ||||||
|  | 			return err; | ||||||
|  | 
 | ||||||
|  | 		sdiodev->sbwad = bar0; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	addr &= SBSDIO_SB_OFT_ADDR_MASK; | ||||||
|  | 
 | ||||||
|  | 	incr_fix = (flags & SDIO_REQ_FIXED) ? SDIOH_DATA_FIX : SDIOH_DATA_INC; | ||||||
|  | 	width = (flags & SDIO_REQ_4BYTE) ? 4 : 2; | ||||||
|  | 	if (width == 4) | ||||||
|  | 		addr |= SBSDIO_SB_ACCESS_2_4B_FLAG; | ||||||
|  | 
 | ||||||
|  | 	status = brcmf_sdioh_request_buffer(sdiodev, incr_fix, SDIOH_READ, | ||||||
|  | 					    fn, addr, width, nbytes, buf, pkt); | ||||||
|  | 
 | ||||||
|  | 	return status; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | brcmf_sdcard_send_buf(struct brcmf_sdio_dev *sdiodev, u32 addr, uint fn, | ||||||
|  | 		      uint flags, u8 *buf, uint nbytes, struct sk_buff *pkt) | ||||||
|  | { | ||||||
|  | 	uint incr_fix; | ||||||
|  | 	uint width; | ||||||
|  | 	uint bar0 = addr & ~SBSDIO_SB_OFT_ADDR_MASK; | ||||||
|  | 	int err = 0; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(INFO, "fun = %d, addr = 0x%x, size = %d\n", fn, addr, nbytes); | ||||||
|  | 
 | ||||||
|  | 	/* Async not implemented yet */ | ||||||
|  | 	if (flags & SDIO_REQ_ASYNC) | ||||||
|  | 		return -ENOTSUPP; | ||||||
|  | 
 | ||||||
|  | 	if (bar0 != sdiodev->sbwad) { | ||||||
|  | 		err = brcmf_sdcard_set_sbaddr_window(sdiodev, bar0); | ||||||
|  | 		if (err) | ||||||
|  | 			return err; | ||||||
|  | 
 | ||||||
|  | 		sdiodev->sbwad = bar0; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	addr &= SBSDIO_SB_OFT_ADDR_MASK; | ||||||
|  | 
 | ||||||
|  | 	incr_fix = (flags & SDIO_REQ_FIXED) ? SDIOH_DATA_FIX : SDIOH_DATA_INC; | ||||||
|  | 	width = (flags & SDIO_REQ_4BYTE) ? 4 : 2; | ||||||
|  | 	if (width == 4) | ||||||
|  | 		addr |= SBSDIO_SB_ACCESS_2_4B_FLAG; | ||||||
|  | 
 | ||||||
|  | 	return brcmf_sdioh_request_buffer(sdiodev, incr_fix, SDIOH_WRITE, fn, | ||||||
|  | 					  addr, width, nbytes, buf, pkt); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int brcmf_sdcard_rwdata(struct brcmf_sdio_dev *sdiodev, uint rw, u32 addr, | ||||||
|  | 			u8 *buf, uint nbytes) | ||||||
|  | { | ||||||
|  | 	addr &= SBSDIO_SB_OFT_ADDR_MASK; | ||||||
|  | 	addr |= SBSDIO_SB_ACCESS_2_4B_FLAG; | ||||||
|  | 
 | ||||||
|  | 	return brcmf_sdioh_request_buffer(sdiodev, SDIOH_DATA_INC, | ||||||
|  | 		(rw ? SDIOH_WRITE : SDIOH_READ), SDIO_FUNC_1, | ||||||
|  | 		addr, 4, nbytes, buf, NULL); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int brcmf_sdcard_abort(struct brcmf_sdio_dev *sdiodev, uint fn) | ||||||
|  | { | ||||||
|  | 	char t_func = (char)fn; | ||||||
|  | 	brcmf_dbg(TRACE, "Enter\n"); | ||||||
|  | 
 | ||||||
|  | 	/* issue abort cmd52 command through F0 */ | ||||||
|  | 	brcmf_sdioh_request_byte(sdiodev, SDIOH_WRITE, SDIO_FUNC_0, | ||||||
|  | 				 SDIO_CCCR_ABORT, &t_func); | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(TRACE, "Exit\n"); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int brcmf_sdio_probe(struct brcmf_sdio_dev *sdiodev) | ||||||
|  | { | ||||||
|  | 	u32 regs = 0; | ||||||
|  | 	int ret = 0; | ||||||
|  | 
 | ||||||
|  | 	ret = brcmf_sdioh_attach(sdiodev); | ||||||
|  | 	if (ret) | ||||||
|  | 		goto out; | ||||||
|  | 
 | ||||||
|  | 	regs = SI_ENUM_BASE; | ||||||
|  | 
 | ||||||
|  | 	/* Report the BAR, to fix if needed */ | ||||||
|  | 	sdiodev->sbwad = SI_ENUM_BASE; | ||||||
|  | 
 | ||||||
|  | 	/* try to attach to the target device */ | ||||||
|  | 	sdiodev->bus = brcmf_sdbrcm_probe(0, 0, 0, 0, regs, sdiodev); | ||||||
|  | 	if (!sdiodev->bus) { | ||||||
|  | 		brcmf_dbg(ERROR, "device attach failed\n"); | ||||||
|  | 		ret = -ENODEV; | ||||||
|  | 		goto out; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | out: | ||||||
|  | 	if (ret) | ||||||
|  | 		brcmf_sdio_remove(sdiodev); | ||||||
|  | 
 | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmf_sdio_probe); | ||||||
|  | 
 | ||||||
|  | int brcmf_sdio_remove(struct brcmf_sdio_dev *sdiodev) | ||||||
|  | { | ||||||
|  | 	if (sdiodev->bus) { | ||||||
|  | 		brcmf_sdbrcm_disconnect(sdiodev->bus); | ||||||
|  | 		sdiodev->bus = NULL; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	brcmf_sdioh_detach(sdiodev); | ||||||
|  | 
 | ||||||
|  | 	sdiodev->sbwad = 0; | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmf_sdio_remove); | ||||||
|  | 
 | ||||||
|  | void brcmf_sdio_wdtmr_enable(struct brcmf_sdio_dev *sdiodev, bool enable) | ||||||
|  | { | ||||||
|  | 	if (enable) | ||||||
|  | 		brcmf_sdbrcm_wd_timer(sdiodev->bus, BRCMF_WD_POLL_MS); | ||||||
|  | 	else | ||||||
|  | 		brcmf_sdbrcm_wd_timer(sdiodev->bus, 0); | ||||||
|  | } | ||||||
							
								
								
									
										625
									
								
								drivers/net/wireless/brcm80211/brcmfmac/bcmsdh_sdmmc.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										625
									
								
								drivers/net/wireless/brcm80211/brcmfmac/bcmsdh_sdmmc.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,625 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | #include <linux/types.h> | ||||||
|  | #include <linux/netdevice.h> | ||||||
|  | #include <linux/mmc/sdio.h> | ||||||
|  | #include <linux/mmc/core.h> | ||||||
|  | #include <linux/mmc/sdio_func.h> | ||||||
|  | #include <linux/mmc/sdio_ids.h> | ||||||
|  | #include <linux/mmc/card.h> | ||||||
|  | #include <linux/suspend.h> | ||||||
|  | #include <linux/errno.h> | ||||||
|  | #include <linux/sched.h>	/* request_irq() */ | ||||||
|  | #include <net/cfg80211.h> | ||||||
|  | 
 | ||||||
|  | #include <defs.h> | ||||||
|  | #include <brcm_hw_ids.h> | ||||||
|  | #include <brcmu_utils.h> | ||||||
|  | #include <brcmu_wifi.h> | ||||||
|  | #include "sdio_host.h" | ||||||
|  | #include "dhd.h" | ||||||
|  | #include "dhd_dbg.h" | ||||||
|  | #include "wl_cfg80211.h" | ||||||
|  | 
 | ||||||
|  | #define SDIO_VENDOR_ID_BROADCOM		0x02d0 | ||||||
|  | 
 | ||||||
|  | #define DMA_ALIGN_MASK	0x03 | ||||||
|  | 
 | ||||||
|  | #define SDIO_DEVICE_ID_BROADCOM_4329	0x4329 | ||||||
|  | 
 | ||||||
|  | #define SDIO_FUNC1_BLOCKSIZE		64 | ||||||
|  | #define SDIO_FUNC2_BLOCKSIZE		512 | ||||||
|  | 
 | ||||||
|  | /* devices we support, null terminated */ | ||||||
|  | static const struct sdio_device_id brcmf_sdmmc_ids[] = { | ||||||
|  | 	{SDIO_DEVICE(SDIO_VENDOR_ID_BROADCOM, SDIO_DEVICE_ID_BROADCOM_4329)}, | ||||||
|  | 	{ /* end: all zeroes */ }, | ||||||
|  | }; | ||||||
|  | MODULE_DEVICE_TABLE(sdio, brcmf_sdmmc_ids); | ||||||
|  | 
 | ||||||
|  | static bool | ||||||
|  | brcmf_pm_resume_error(struct brcmf_sdio_dev *sdiodev) | ||||||
|  | { | ||||||
|  | 	bool is_err = false; | ||||||
|  | #ifdef CONFIG_PM_SLEEP | ||||||
|  | 	is_err = atomic_read(&sdiodev->suspend); | ||||||
|  | #endif | ||||||
|  | 	return is_err; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void | ||||||
|  | brcmf_pm_resume_wait(struct brcmf_sdio_dev *sdiodev, wait_queue_head_t *wq) | ||||||
|  | { | ||||||
|  | #ifdef CONFIG_PM_SLEEP | ||||||
|  | 	int retry = 0; | ||||||
|  | 	while (atomic_read(&sdiodev->suspend) && retry++ != 30) | ||||||
|  | 		wait_event_timeout(*wq, false, HZ/100); | ||||||
|  | #endif | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline int brcmf_sdioh_f0_write_byte(struct brcmf_sdio_dev *sdiodev, | ||||||
|  | 					    uint regaddr, u8 *byte) | ||||||
|  | { | ||||||
|  | 	struct sdio_func *sdfunc = sdiodev->func[0]; | ||||||
|  | 	int err_ret; | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * Can only directly write to some F0 registers. | ||||||
|  | 	 * Handle F2 enable/disable and Abort command | ||||||
|  | 	 * as a special case. | ||||||
|  | 	 */ | ||||||
|  | 	if (regaddr == SDIO_CCCR_IOEx) { | ||||||
|  | 		sdfunc = sdiodev->func[2]; | ||||||
|  | 		if (sdfunc) { | ||||||
|  | 			sdio_claim_host(sdfunc); | ||||||
|  | 			if (*byte & SDIO_FUNC_ENABLE_2) { | ||||||
|  | 				/* Enable Function 2 */ | ||||||
|  | 				err_ret = sdio_enable_func(sdfunc); | ||||||
|  | 				if (err_ret) | ||||||
|  | 					brcmf_dbg(ERROR, | ||||||
|  | 						  "enable F2 failed:%d\n", | ||||||
|  | 						  err_ret); | ||||||
|  | 			} else { | ||||||
|  | 				/* Disable Function 2 */ | ||||||
|  | 				err_ret = sdio_disable_func(sdfunc); | ||||||
|  | 				if (err_ret) | ||||||
|  | 					brcmf_dbg(ERROR, | ||||||
|  | 						  "Disable F2 failed:%d\n", | ||||||
|  | 						  err_ret); | ||||||
|  | 			} | ||||||
|  | 			sdio_release_host(sdfunc); | ||||||
|  | 		} | ||||||
|  | 	} else if (regaddr == SDIO_CCCR_ABORT) { | ||||||
|  | 		sdio_claim_host(sdfunc); | ||||||
|  | 		sdio_writeb(sdfunc, *byte, regaddr, &err_ret); | ||||||
|  | 		sdio_release_host(sdfunc); | ||||||
|  | 	} else if (regaddr < 0xF0) { | ||||||
|  | 		brcmf_dbg(ERROR, "F0 Wr:0x%02x: write disallowed\n", regaddr); | ||||||
|  | 		err_ret = -EPERM; | ||||||
|  | 	} else { | ||||||
|  | 		sdio_claim_host(sdfunc); | ||||||
|  | 		sdio_f0_writeb(sdfunc, *byte, regaddr, &err_ret); | ||||||
|  | 		sdio_release_host(sdfunc); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return err_ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int brcmf_sdioh_request_byte(struct brcmf_sdio_dev *sdiodev, uint rw, uint func, | ||||||
|  | 			     uint regaddr, u8 *byte) | ||||||
|  | { | ||||||
|  | 	int err_ret; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(INFO, "rw=%d, func=%d, addr=0x%05x\n", rw, func, regaddr); | ||||||
|  | 
 | ||||||
|  | 	brcmf_pm_resume_wait(sdiodev, &sdiodev->request_byte_wait); | ||||||
|  | 	if (brcmf_pm_resume_error(sdiodev)) | ||||||
|  | 		return -EIO; | ||||||
|  | 
 | ||||||
|  | 	if (rw && func == 0) { | ||||||
|  | 		/* handle F0 separately */ | ||||||
|  | 		err_ret = brcmf_sdioh_f0_write_byte(sdiodev, regaddr, byte); | ||||||
|  | 	} else { | ||||||
|  | 		sdio_claim_host(sdiodev->func[func]); | ||||||
|  | 		if (rw) /* CMD52 Write */ | ||||||
|  | 			sdio_writeb(sdiodev->func[func], *byte, regaddr, | ||||||
|  | 				    &err_ret); | ||||||
|  | 		else if (func == 0) { | ||||||
|  | 			*byte = sdio_f0_readb(sdiodev->func[func], regaddr, | ||||||
|  | 					      &err_ret); | ||||||
|  | 		} else { | ||||||
|  | 			*byte = sdio_readb(sdiodev->func[func], regaddr, | ||||||
|  | 					   &err_ret); | ||||||
|  | 		} | ||||||
|  | 		sdio_release_host(sdiodev->func[func]); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (err_ret) | ||||||
|  | 		brcmf_dbg(ERROR, "Failed to %s byte F%d:@0x%05x=%02x, Err: %d\n", | ||||||
|  | 			  rw ? "write" : "read", func, regaddr, *byte, err_ret); | ||||||
|  | 
 | ||||||
|  | 	return err_ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int brcmf_sdioh_request_word(struct brcmf_sdio_dev *sdiodev, | ||||||
|  | 			     uint rw, uint func, uint addr, u32 *word, | ||||||
|  | 			     uint nbytes) | ||||||
|  | { | ||||||
|  | 	int err_ret = -EIO; | ||||||
|  | 
 | ||||||
|  | 	if (func == 0) { | ||||||
|  | 		brcmf_dbg(ERROR, "Only CMD52 allowed to F0\n"); | ||||||
|  | 		return -EINVAL; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(INFO, "rw=%d, func=%d, addr=0x%05x, nbytes=%d\n", | ||||||
|  | 		  rw, func, addr, nbytes); | ||||||
|  | 
 | ||||||
|  | 	brcmf_pm_resume_wait(sdiodev, &sdiodev->request_word_wait); | ||||||
|  | 	if (brcmf_pm_resume_error(sdiodev)) | ||||||
|  | 		return -EIO; | ||||||
|  | 	/* Claim host controller */ | ||||||
|  | 	sdio_claim_host(sdiodev->func[func]); | ||||||
|  | 
 | ||||||
|  | 	if (rw) {		/* CMD52 Write */ | ||||||
|  | 		if (nbytes == 4) | ||||||
|  | 			sdio_writel(sdiodev->func[func], *word, addr, | ||||||
|  | 				    &err_ret); | ||||||
|  | 		else if (nbytes == 2) | ||||||
|  | 			sdio_writew(sdiodev->func[func], (*word & 0xFFFF), | ||||||
|  | 				    addr, &err_ret); | ||||||
|  | 		else | ||||||
|  | 			brcmf_dbg(ERROR, "Invalid nbytes: %d\n", nbytes); | ||||||
|  | 	} else {		/* CMD52 Read */ | ||||||
|  | 		if (nbytes == 4) | ||||||
|  | 			*word = sdio_readl(sdiodev->func[func], addr, &err_ret); | ||||||
|  | 		else if (nbytes == 2) | ||||||
|  | 			*word = sdio_readw(sdiodev->func[func], addr, | ||||||
|  | 					   &err_ret) & 0xFFFF; | ||||||
|  | 		else | ||||||
|  | 			brcmf_dbg(ERROR, "Invalid nbytes: %d\n", nbytes); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Release host controller */ | ||||||
|  | 	sdio_release_host(sdiodev->func[func]); | ||||||
|  | 
 | ||||||
|  | 	if (err_ret) | ||||||
|  | 		brcmf_dbg(ERROR, "Failed to %s word, Err: 0x%08x\n", | ||||||
|  | 			  rw ? "write" : "read", err_ret); | ||||||
|  | 
 | ||||||
|  | 	return err_ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | brcmf_sdioh_request_packet(struct brcmf_sdio_dev *sdiodev, uint fix_inc, | ||||||
|  | 			   uint write, uint func, uint addr, | ||||||
|  | 			   struct sk_buff *pkt) | ||||||
|  | { | ||||||
|  | 	bool fifo = (fix_inc == SDIOH_DATA_FIX); | ||||||
|  | 	u32 SGCount = 0; | ||||||
|  | 	int err_ret = 0; | ||||||
|  | 
 | ||||||
|  | 	struct sk_buff *pnext; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(TRACE, "Enter\n"); | ||||||
|  | 
 | ||||||
|  | 	brcmf_pm_resume_wait(sdiodev, &sdiodev->request_packet_wait); | ||||||
|  | 	if (brcmf_pm_resume_error(sdiodev)) | ||||||
|  | 		return -EIO; | ||||||
|  | 
 | ||||||
|  | 	/* Claim host controller */ | ||||||
|  | 	sdio_claim_host(sdiodev->func[func]); | ||||||
|  | 	for (pnext = pkt; pnext; pnext = pnext->next) { | ||||||
|  | 		uint pkt_len = pnext->len; | ||||||
|  | 		pkt_len += 3; | ||||||
|  | 		pkt_len &= 0xFFFFFFFC; | ||||||
|  | 
 | ||||||
|  | 		if ((write) && (!fifo)) { | ||||||
|  | 			err_ret = sdio_memcpy_toio(sdiodev->func[func], addr, | ||||||
|  | 						   ((u8 *) (pnext->data)), | ||||||
|  | 						   pkt_len); | ||||||
|  | 		} else if (write) { | ||||||
|  | 			err_ret = sdio_memcpy_toio(sdiodev->func[func], addr, | ||||||
|  | 						   ((u8 *) (pnext->data)), | ||||||
|  | 						   pkt_len); | ||||||
|  | 		} else if (fifo) { | ||||||
|  | 			err_ret = sdio_readsb(sdiodev->func[func], | ||||||
|  | 					      ((u8 *) (pnext->data)), | ||||||
|  | 					      addr, pkt_len); | ||||||
|  | 		} else { | ||||||
|  | 			err_ret = sdio_memcpy_fromio(sdiodev->func[func], | ||||||
|  | 						     ((u8 *) (pnext->data)), | ||||||
|  | 						     addr, pkt_len); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if (err_ret) { | ||||||
|  | 			brcmf_dbg(ERROR, "%s FAILED %p[%d], addr=0x%05x, pkt_len=%d, ERR=0x%08x\n", | ||||||
|  | 				  write ? "TX" : "RX", pnext, SGCount, addr, | ||||||
|  | 				  pkt_len, err_ret); | ||||||
|  | 		} else { | ||||||
|  | 			brcmf_dbg(TRACE, "%s xfr'd %p[%d], addr=0x%05x, len=%d\n", | ||||||
|  | 				  write ? "TX" : "RX", pnext, SGCount, addr, | ||||||
|  | 				  pkt_len); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if (!fifo) | ||||||
|  | 			addr += pkt_len; | ||||||
|  | 		SGCount++; | ||||||
|  | 
 | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Release host controller */ | ||||||
|  | 	sdio_release_host(sdiodev->func[func]); | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(TRACE, "Exit\n"); | ||||||
|  | 	return err_ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * This function takes a buffer or packet, and fixes everything up | ||||||
|  |  * so that in the end, a DMA-able packet is created. | ||||||
|  |  * | ||||||
|  |  * A buffer does not have an associated packet pointer, | ||||||
|  |  * and may or may not be aligned. | ||||||
|  |  * A packet may consist of a single packet, or a packet chain. | ||||||
|  |  * If it is a packet chain, then all the packets in the chain | ||||||
|  |  * must be properly aligned. | ||||||
|  |  * | ||||||
|  |  * If the packet data is not aligned, then there may only be | ||||||
|  |  * one packet, and in this case,  it is copied to a new | ||||||
|  |  * aligned packet. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | int brcmf_sdioh_request_buffer(struct brcmf_sdio_dev *sdiodev, | ||||||
|  | 			       uint fix_inc, uint write, uint func, uint addr, | ||||||
|  | 			       uint reg_width, uint buflen_u, u8 *buffer, | ||||||
|  | 			       struct sk_buff *pkt) | ||||||
|  | { | ||||||
|  | 	int Status; | ||||||
|  | 	struct sk_buff *mypkt = NULL; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(TRACE, "Enter\n"); | ||||||
|  | 
 | ||||||
|  | 	brcmf_pm_resume_wait(sdiodev, &sdiodev->request_buffer_wait); | ||||||
|  | 	if (brcmf_pm_resume_error(sdiodev)) | ||||||
|  | 		return -EIO; | ||||||
|  | 	/* Case 1: we don't have a packet. */ | ||||||
|  | 	if (pkt == NULL) { | ||||||
|  | 		brcmf_dbg(DATA, "Creating new %s Packet, len=%d\n", | ||||||
|  | 			  write ? "TX" : "RX", buflen_u); | ||||||
|  | 		mypkt = brcmu_pkt_buf_get_skb(buflen_u); | ||||||
|  | 		if (!mypkt) { | ||||||
|  | 			brcmf_dbg(ERROR, "brcmu_pkt_buf_get_skb failed: len %d\n", | ||||||
|  | 				  buflen_u); | ||||||
|  | 			return -EIO; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		/* For a write, copy the buffer data into the packet. */ | ||||||
|  | 		if (write) | ||||||
|  | 			memcpy(mypkt->data, buffer, buflen_u); | ||||||
|  | 
 | ||||||
|  | 		Status = brcmf_sdioh_request_packet(sdiodev, fix_inc, write, | ||||||
|  | 						    func, addr, mypkt); | ||||||
|  | 
 | ||||||
|  | 		/* For a read, copy the packet data back to the buffer. */ | ||||||
|  | 		if (!write) | ||||||
|  | 			memcpy(buffer, mypkt->data, buflen_u); | ||||||
|  | 
 | ||||||
|  | 		brcmu_pkt_buf_free_skb(mypkt); | ||||||
|  | 	} else if (((ulong) (pkt->data) & DMA_ALIGN_MASK) != 0) { | ||||||
|  | 		/*
 | ||||||
|  | 		 * Case 2: We have a packet, but it is unaligned. | ||||||
|  | 		 * In this case, we cannot have a chain (pkt->next == NULL) | ||||||
|  | 		 */ | ||||||
|  | 		brcmf_dbg(DATA, "Creating aligned %s Packet, len=%d\n", | ||||||
|  | 			  write ? "TX" : "RX", pkt->len); | ||||||
|  | 		mypkt = brcmu_pkt_buf_get_skb(pkt->len); | ||||||
|  | 		if (!mypkt) { | ||||||
|  | 			brcmf_dbg(ERROR, "brcmu_pkt_buf_get_skb failed: len %d\n", | ||||||
|  | 				  pkt->len); | ||||||
|  | 			return -EIO; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		/* For a write, copy the buffer data into the packet. */ | ||||||
|  | 		if (write) | ||||||
|  | 			memcpy(mypkt->data, pkt->data, pkt->len); | ||||||
|  | 
 | ||||||
|  | 		Status = brcmf_sdioh_request_packet(sdiodev, fix_inc, write, | ||||||
|  | 						    func, addr, mypkt); | ||||||
|  | 
 | ||||||
|  | 		/* For a read, copy the packet data back to the buffer. */ | ||||||
|  | 		if (!write) | ||||||
|  | 			memcpy(pkt->data, mypkt->data, mypkt->len); | ||||||
|  | 
 | ||||||
|  | 		brcmu_pkt_buf_free_skb(mypkt); | ||||||
|  | 	} else {		/* case 3: We have a packet and
 | ||||||
|  | 				 it is aligned. */ | ||||||
|  | 		brcmf_dbg(DATA, "Aligned %s Packet, direct DMA\n", | ||||||
|  | 			  write ? "Tx" : "Rx"); | ||||||
|  | 		Status = brcmf_sdioh_request_packet(sdiodev, fix_inc, write, | ||||||
|  | 						    func, addr, pkt); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return Status; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Read client card reg */ | ||||||
|  | static int | ||||||
|  | brcmf_sdioh_card_regread(struct brcmf_sdio_dev *sdiodev, int func, u32 regaddr, | ||||||
|  | 			 int regsize, u32 *data) | ||||||
|  | { | ||||||
|  | 
 | ||||||
|  | 	if ((func == 0) || (regsize == 1)) { | ||||||
|  | 		u8 temp = 0; | ||||||
|  | 
 | ||||||
|  | 		brcmf_sdioh_request_byte(sdiodev, SDIOH_READ, func, regaddr, | ||||||
|  | 					 &temp); | ||||||
|  | 		*data = temp; | ||||||
|  | 		*data &= 0xff; | ||||||
|  | 		brcmf_dbg(DATA, "byte read data=0x%02x\n", *data); | ||||||
|  | 	} else { | ||||||
|  | 		brcmf_sdioh_request_word(sdiodev, SDIOH_READ, func, regaddr, | ||||||
|  | 					 data, regsize); | ||||||
|  | 		if (regsize == 2) | ||||||
|  | 			*data &= 0xffff; | ||||||
|  | 
 | ||||||
|  | 		brcmf_dbg(DATA, "word read data=0x%08x\n", *data); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return SUCCESS; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int brcmf_sdioh_get_cisaddr(struct brcmf_sdio_dev *sdiodev, u32 regaddr) | ||||||
|  | { | ||||||
|  | 	/* read 24 bits and return valid 17 bit addr */ | ||||||
|  | 	int i; | ||||||
|  | 	u32 scratch, regdata; | ||||||
|  | 	__le32 scratch_le; | ||||||
|  | 	u8 *ptr = (u8 *)&scratch_le; | ||||||
|  | 
 | ||||||
|  | 	for (i = 0; i < 3; i++) { | ||||||
|  | 		if ((brcmf_sdioh_card_regread(sdiodev, 0, regaddr, 1, | ||||||
|  | 				®data)) != SUCCESS) | ||||||
|  | 			brcmf_dbg(ERROR, "Can't read!\n"); | ||||||
|  | 
 | ||||||
|  | 		*ptr++ = (u8) regdata; | ||||||
|  | 		regaddr++; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Only the lower 17-bits are valid */ | ||||||
|  | 	scratch = le32_to_cpu(scratch_le); | ||||||
|  | 	scratch &= 0x0001FFFF; | ||||||
|  | 	return scratch; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int brcmf_sdioh_enablefuncs(struct brcmf_sdio_dev *sdiodev) | ||||||
|  | { | ||||||
|  | 	int err_ret; | ||||||
|  | 	u32 fbraddr; | ||||||
|  | 	u8 func; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(TRACE, "\n"); | ||||||
|  | 
 | ||||||
|  | 	/* Get the Card's common CIS address */ | ||||||
|  | 	sdiodev->func_cis_ptr[0] = brcmf_sdioh_get_cisaddr(sdiodev, | ||||||
|  | 							   SDIO_CCCR_CIS); | ||||||
|  | 	brcmf_dbg(INFO, "Card's Common CIS Ptr = 0x%x\n", | ||||||
|  | 		  sdiodev->func_cis_ptr[0]); | ||||||
|  | 
 | ||||||
|  | 	/* Get the Card's function CIS (for each function) */ | ||||||
|  | 	for (fbraddr = SDIO_FBR_BASE(1), func = 1; | ||||||
|  | 	     func <= sdiodev->num_funcs; func++, fbraddr += SDIOD_FBR_SIZE) { | ||||||
|  | 		sdiodev->func_cis_ptr[func] = | ||||||
|  | 		    brcmf_sdioh_get_cisaddr(sdiodev, SDIO_FBR_CIS + fbraddr); | ||||||
|  | 		brcmf_dbg(INFO, "Function %d CIS Ptr = 0x%x\n", | ||||||
|  | 			  func, sdiodev->func_cis_ptr[func]); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Enable Function 1 */ | ||||||
|  | 	sdio_claim_host(sdiodev->func[1]); | ||||||
|  | 	err_ret = sdio_enable_func(sdiodev->func[1]); | ||||||
|  | 	sdio_release_host(sdiodev->func[1]); | ||||||
|  | 	if (err_ret) | ||||||
|  | 		brcmf_dbg(ERROR, "Failed to enable F1 Err: 0x%08x\n", err_ret); | ||||||
|  | 
 | ||||||
|  | 	return false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  *	Public entry points & extern's | ||||||
|  |  */ | ||||||
|  | int brcmf_sdioh_attach(struct brcmf_sdio_dev *sdiodev) | ||||||
|  | { | ||||||
|  | 	int err_ret = 0; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(TRACE, "\n"); | ||||||
|  | 
 | ||||||
|  | 	sdiodev->num_funcs = 2; | ||||||
|  | 
 | ||||||
|  | 	sdio_claim_host(sdiodev->func[1]); | ||||||
|  | 	err_ret = sdio_set_block_size(sdiodev->func[1], SDIO_FUNC1_BLOCKSIZE); | ||||||
|  | 	sdio_release_host(sdiodev->func[1]); | ||||||
|  | 	if (err_ret) { | ||||||
|  | 		brcmf_dbg(ERROR, "Failed to set F1 blocksize\n"); | ||||||
|  | 		goto out; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	sdio_claim_host(sdiodev->func[2]); | ||||||
|  | 	err_ret = sdio_set_block_size(sdiodev->func[2], SDIO_FUNC2_BLOCKSIZE); | ||||||
|  | 	sdio_release_host(sdiodev->func[2]); | ||||||
|  | 	if (err_ret) { | ||||||
|  | 		brcmf_dbg(ERROR, "Failed to set F2 blocksize\n"); | ||||||
|  | 		goto out; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	brcmf_sdioh_enablefuncs(sdiodev); | ||||||
|  | 
 | ||||||
|  | out: | ||||||
|  | 	brcmf_dbg(TRACE, "Done\n"); | ||||||
|  | 	return err_ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void brcmf_sdioh_detach(struct brcmf_sdio_dev *sdiodev) | ||||||
|  | { | ||||||
|  | 	brcmf_dbg(TRACE, "\n"); | ||||||
|  | 
 | ||||||
|  | 	/* Disable Function 2 */ | ||||||
|  | 	sdio_claim_host(sdiodev->func[2]); | ||||||
|  | 	sdio_disable_func(sdiodev->func[2]); | ||||||
|  | 	sdio_release_host(sdiodev->func[2]); | ||||||
|  | 
 | ||||||
|  | 	/* Disable Function 1 */ | ||||||
|  | 	sdio_claim_host(sdiodev->func[1]); | ||||||
|  | 	sdio_disable_func(sdiodev->func[1]); | ||||||
|  | 	sdio_release_host(sdiodev->func[1]); | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int brcmf_ops_sdio_probe(struct sdio_func *func, | ||||||
|  | 			      const struct sdio_device_id *id) | ||||||
|  | { | ||||||
|  | 	int ret = 0; | ||||||
|  | 	struct brcmf_sdio_dev *sdiodev; | ||||||
|  | 	brcmf_dbg(TRACE, "Enter\n"); | ||||||
|  | 	brcmf_dbg(TRACE, "func->class=%x\n", func->class); | ||||||
|  | 	brcmf_dbg(TRACE, "sdio_vendor: 0x%04x\n", func->vendor); | ||||||
|  | 	brcmf_dbg(TRACE, "sdio_device: 0x%04x\n", func->device); | ||||||
|  | 	brcmf_dbg(TRACE, "Function#: 0x%04x\n", func->num); | ||||||
|  | 
 | ||||||
|  | 	if (func->num == 1) { | ||||||
|  | 		if (dev_get_drvdata(&func->card->dev)) { | ||||||
|  | 			brcmf_dbg(ERROR, "card private drvdata occupied\n"); | ||||||
|  | 			return -ENXIO; | ||||||
|  | 		} | ||||||
|  | 		sdiodev = kzalloc(sizeof(struct brcmf_sdio_dev), GFP_KERNEL); | ||||||
|  | 		if (!sdiodev) | ||||||
|  | 			return -ENOMEM; | ||||||
|  | 		sdiodev->func[0] = func->card->sdio_func[0]; | ||||||
|  | 		sdiodev->func[1] = func; | ||||||
|  | 		dev_set_drvdata(&func->card->dev, sdiodev); | ||||||
|  | 
 | ||||||
|  | 		atomic_set(&sdiodev->suspend, false); | ||||||
|  | 		init_waitqueue_head(&sdiodev->request_byte_wait); | ||||||
|  | 		init_waitqueue_head(&sdiodev->request_word_wait); | ||||||
|  | 		init_waitqueue_head(&sdiodev->request_packet_wait); | ||||||
|  | 		init_waitqueue_head(&sdiodev->request_buffer_wait); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (func->num == 2) { | ||||||
|  | 		sdiodev = dev_get_drvdata(&func->card->dev); | ||||||
|  | 		if ((!sdiodev) || (sdiodev->func[1]->card != func->card)) | ||||||
|  | 			return -ENODEV; | ||||||
|  | 		sdiodev->func[2] = func; | ||||||
|  | 
 | ||||||
|  | 		brcmf_dbg(TRACE, "F2 found, calling brcmf_sdio_probe...\n"); | ||||||
|  | 		ret = brcmf_sdio_probe(sdiodev); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void brcmf_ops_sdio_remove(struct sdio_func *func) | ||||||
|  | { | ||||||
|  | 	struct brcmf_sdio_dev *sdiodev; | ||||||
|  | 	brcmf_dbg(TRACE, "Enter\n"); | ||||||
|  | 	brcmf_dbg(INFO, "func->class=%x\n", func->class); | ||||||
|  | 	brcmf_dbg(INFO, "sdio_vendor: 0x%04x\n", func->vendor); | ||||||
|  | 	brcmf_dbg(INFO, "sdio_device: 0x%04x\n", func->device); | ||||||
|  | 	brcmf_dbg(INFO, "Function#: 0x%04x\n", func->num); | ||||||
|  | 
 | ||||||
|  | 	if (func->num == 2) { | ||||||
|  | 		sdiodev = dev_get_drvdata(&func->card->dev); | ||||||
|  | 		brcmf_dbg(TRACE, "F2 found, calling brcmf_sdio_remove...\n"); | ||||||
|  | 		brcmf_sdio_remove(sdiodev); | ||||||
|  | 		dev_set_drvdata(&func->card->dev, NULL); | ||||||
|  | 		kfree(sdiodev); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #ifdef CONFIG_PM_SLEEP | ||||||
|  | static int brcmf_sdio_suspend(struct device *dev) | ||||||
|  | { | ||||||
|  | 	mmc_pm_flag_t sdio_flags; | ||||||
|  | 	struct brcmf_sdio_dev *sdiodev; | ||||||
|  | 	struct sdio_func *func = dev_to_sdio_func(dev); | ||||||
|  | 	int ret = 0; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(TRACE, "\n"); | ||||||
|  | 
 | ||||||
|  | 	sdiodev = dev_get_drvdata(&func->card->dev); | ||||||
|  | 
 | ||||||
|  | 	atomic_set(&sdiodev->suspend, true); | ||||||
|  | 
 | ||||||
|  | 	sdio_flags = sdio_get_host_pm_caps(sdiodev->func[1]); | ||||||
|  | 	if (!(sdio_flags & MMC_PM_KEEP_POWER)) { | ||||||
|  | 		brcmf_dbg(ERROR, "Host can't keep power while suspended\n"); | ||||||
|  | 		return -EINVAL; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	ret = sdio_set_host_pm_flags(sdiodev->func[1], MMC_PM_KEEP_POWER); | ||||||
|  | 	if (ret) { | ||||||
|  | 		brcmf_dbg(ERROR, "Failed to set pm_flags\n"); | ||||||
|  | 		return ret; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	brcmf_sdio_wdtmr_enable(sdiodev, false); | ||||||
|  | 
 | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int brcmf_sdio_resume(struct device *dev) | ||||||
|  | { | ||||||
|  | 	struct brcmf_sdio_dev *sdiodev; | ||||||
|  | 	struct sdio_func *func = dev_to_sdio_func(dev); | ||||||
|  | 
 | ||||||
|  | 	sdiodev = dev_get_drvdata(&func->card->dev); | ||||||
|  | 	brcmf_sdio_wdtmr_enable(sdiodev, true); | ||||||
|  | 	atomic_set(&sdiodev->suspend, false); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static const struct dev_pm_ops brcmf_sdio_pm_ops = { | ||||||
|  | 	.suspend	= brcmf_sdio_suspend, | ||||||
|  | 	.resume		= brcmf_sdio_resume, | ||||||
|  | }; | ||||||
|  | #endif	/* CONFIG_PM_SLEEP */ | ||||||
|  | 
 | ||||||
|  | static struct sdio_driver brcmf_sdmmc_driver = { | ||||||
|  | 	.probe = brcmf_ops_sdio_probe, | ||||||
|  | 	.remove = brcmf_ops_sdio_remove, | ||||||
|  | 	.name = "brcmfmac", | ||||||
|  | 	.id_table = brcmf_sdmmc_ids, | ||||||
|  | #ifdef CONFIG_PM_SLEEP | ||||||
|  | 	.drv = { | ||||||
|  | 		.pm = &brcmf_sdio_pm_ops, | ||||||
|  | 	}, | ||||||
|  | #endif	/* CONFIG_PM_SLEEP */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* bus register interface */ | ||||||
|  | int brcmf_bus_register(void) | ||||||
|  | { | ||||||
|  | 	brcmf_dbg(TRACE, "Enter\n"); | ||||||
|  | 
 | ||||||
|  | 	return sdio_register_driver(&brcmf_sdmmc_driver); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void brcmf_bus_unregister(void) | ||||||
|  | { | ||||||
|  | 	brcmf_dbg(TRACE, "Enter\n"); | ||||||
|  | 
 | ||||||
|  | 	sdio_unregister_driver(&brcmf_sdmmc_driver); | ||||||
|  | } | ||||||
							
								
								
									
										773
									
								
								drivers/net/wireless/brcm80211/brcmfmac/dhd.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										773
									
								
								drivers/net/wireless/brcm80211/brcmfmac/dhd.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,773 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /****************
 | ||||||
|  |  * Common types * | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCMF_H_ | ||||||
|  | #define _BRCMF_H_ | ||||||
|  | 
 | ||||||
|  | #define BRCMF_VERSION_STR		"4.218.248.5" | ||||||
|  | 
 | ||||||
|  | /*******************************************************************************
 | ||||||
|  |  * IO codes that are interpreted by dongle firmware | ||||||
|  |  ******************************************************************************/ | ||||||
|  | #define BRCMF_C_UP				2 | ||||||
|  | #define BRCMF_C_SET_PROMISC			10 | ||||||
|  | #define BRCMF_C_GET_RATE			12 | ||||||
|  | #define BRCMF_C_GET_INFRA			19 | ||||||
|  | #define BRCMF_C_SET_INFRA			20 | ||||||
|  | #define BRCMF_C_GET_AUTH			21 | ||||||
|  | #define BRCMF_C_SET_AUTH			22 | ||||||
|  | #define BRCMF_C_GET_BSSID			23 | ||||||
|  | #define BRCMF_C_GET_SSID			25 | ||||||
|  | #define BRCMF_C_SET_SSID			26 | ||||||
|  | #define BRCMF_C_GET_CHANNEL			29 | ||||||
|  | #define BRCMF_C_GET_SRL				31 | ||||||
|  | #define BRCMF_C_GET_LRL				33 | ||||||
|  | #define BRCMF_C_GET_RADIO			37 | ||||||
|  | #define BRCMF_C_SET_RADIO			38 | ||||||
|  | #define BRCMF_C_GET_PHYTYPE			39 | ||||||
|  | #define BRCMF_C_SET_KEY				45 | ||||||
|  | #define BRCMF_C_SET_PASSIVE_SCAN		49 | ||||||
|  | #define BRCMF_C_SCAN				50 | ||||||
|  | #define BRCMF_C_SCAN_RESULTS			51 | ||||||
|  | #define BRCMF_C_DISASSOC			52 | ||||||
|  | #define BRCMF_C_REASSOC				53 | ||||||
|  | #define BRCMF_C_SET_ROAM_TRIGGER		55 | ||||||
|  | #define BRCMF_C_SET_ROAM_DELTA			57 | ||||||
|  | #define BRCMF_C_GET_DTIMPRD			77 | ||||||
|  | #define BRCMF_C_SET_COUNTRY			84 | ||||||
|  | #define BRCMF_C_GET_PM				85 | ||||||
|  | #define BRCMF_C_SET_PM				86 | ||||||
|  | #define BRCMF_C_GET_AP				117 | ||||||
|  | #define BRCMF_C_SET_AP				118 | ||||||
|  | #define BRCMF_C_GET_RSSI			127 | ||||||
|  | #define BRCMF_C_GET_WSEC			133 | ||||||
|  | #define BRCMF_C_SET_WSEC			134 | ||||||
|  | #define BRCMF_C_GET_PHY_NOISE			135 | ||||||
|  | #define BRCMF_C_GET_BSS_INFO			136 | ||||||
|  | #define BRCMF_C_SET_SCAN_CHANNEL_TIME		185 | ||||||
|  | #define BRCMF_C_SET_SCAN_UNASSOC_TIME		187 | ||||||
|  | #define BRCMF_C_SCB_DEAUTHENTICATE_FOR_REASON	201 | ||||||
|  | #define BRCMF_C_GET_VALID_CHANNELS		217 | ||||||
|  | #define BRCMF_C_GET_KEY_PRIMARY			235 | ||||||
|  | #define BRCMF_C_SET_KEY_PRIMARY			236 | ||||||
|  | #define BRCMF_C_SET_SCAN_PASSIVE_TIME		258 | ||||||
|  | #define BRCMF_C_GET_VAR				262 | ||||||
|  | #define BRCMF_C_SET_VAR				263 | ||||||
|  | 
 | ||||||
|  | /* phy types (returned by WLC_GET_PHYTPE) */ | ||||||
|  | #define	WLC_PHY_TYPE_A		0 | ||||||
|  | #define	WLC_PHY_TYPE_B		1 | ||||||
|  | #define	WLC_PHY_TYPE_G		2 | ||||||
|  | #define	WLC_PHY_TYPE_N		4 | ||||||
|  | #define	WLC_PHY_TYPE_LP		5 | ||||||
|  | #define	WLC_PHY_TYPE_SSN	6 | ||||||
|  | #define	WLC_PHY_TYPE_HT		7 | ||||||
|  | #define	WLC_PHY_TYPE_LCN	8 | ||||||
|  | #define	WLC_PHY_TYPE_NULL	0xf | ||||||
|  | 
 | ||||||
|  | #define BRCMF_EVENTING_MASK_LEN	16 | ||||||
|  | 
 | ||||||
|  | #define TOE_TX_CSUM_OL		0x00000001 | ||||||
|  | #define TOE_RX_CSUM_OL		0x00000002 | ||||||
|  | 
 | ||||||
|  | #define	BRCMF_BSS_INFO_VERSION	108 /* current ver of brcmf_bss_info struct */ | ||||||
|  | 
 | ||||||
|  | /* size of brcmf_scan_params not including variable length array */ | ||||||
|  | #define BRCMF_SCAN_PARAMS_FIXED_SIZE 64 | ||||||
|  | 
 | ||||||
|  | /* masks for channel and ssid count */ | ||||||
|  | #define BRCMF_SCAN_PARAMS_COUNT_MASK 0x0000ffff | ||||||
|  | #define BRCMF_SCAN_PARAMS_NSSID_SHIFT 16 | ||||||
|  | 
 | ||||||
|  | #define BRCMF_SCAN_ACTION_START      1 | ||||||
|  | #define BRCMF_SCAN_ACTION_CONTINUE   2 | ||||||
|  | #define WL_SCAN_ACTION_ABORT      3 | ||||||
|  | 
 | ||||||
|  | #define BRCMF_ISCAN_REQ_VERSION 1 | ||||||
|  | 
 | ||||||
|  | /* brcmf_iscan_results status values */ | ||||||
|  | #define BRCMF_SCAN_RESULTS_SUCCESS	0 | ||||||
|  | #define BRCMF_SCAN_RESULTS_PARTIAL	1 | ||||||
|  | #define BRCMF_SCAN_RESULTS_PENDING	2 | ||||||
|  | #define BRCMF_SCAN_RESULTS_ABORTED	3 | ||||||
|  | #define BRCMF_SCAN_RESULTS_NO_MEM	4 | ||||||
|  | 
 | ||||||
|  | /* Indicates this key is using soft encrypt */ | ||||||
|  | #define WL_SOFT_KEY	(1 << 0) | ||||||
|  | /* primary (ie tx) key */ | ||||||
|  | #define BRCMF_PRIMARY_KEY	(1 << 1) | ||||||
|  | /* Reserved for backward compat */ | ||||||
|  | #define WL_KF_RES_4	(1 << 4) | ||||||
|  | /* Reserved for backward compat */ | ||||||
|  | #define WL_KF_RES_5	(1 << 5) | ||||||
|  | /* Indicates a group key for a IBSS PEER */ | ||||||
|  | #define WL_IBSS_PEER_GROUP_KEY	(1 << 6) | ||||||
|  | 
 | ||||||
|  | /* For supporting multiple interfaces */ | ||||||
|  | #define BRCMF_MAX_IFS	16 | ||||||
|  | #define BRCMF_DEL_IF	-0xe | ||||||
|  | #define BRCMF_BAD_IF	-0xf | ||||||
|  | 
 | ||||||
|  | #define DOT11_BSSTYPE_ANY			2 | ||||||
|  | #define DOT11_MAX_DEFAULT_KEYS	4 | ||||||
|  | 
 | ||||||
|  | #define BRCMF_EVENT_MSG_LINK		0x01 | ||||||
|  | #define BRCMF_EVENT_MSG_FLUSHTXQ	0x02 | ||||||
|  | #define BRCMF_EVENT_MSG_GROUP		0x04 | ||||||
|  | 
 | ||||||
|  | struct brcmf_event_msg { | ||||||
|  | 	__be16 version; | ||||||
|  | 	__be16 flags; | ||||||
|  | 	__be32 event_type; | ||||||
|  | 	__be32 status; | ||||||
|  | 	__be32 reason; | ||||||
|  | 	__be32 auth_type; | ||||||
|  | 	__be32 datalen; | ||||||
|  | 	u8 addr[ETH_ALEN]; | ||||||
|  | 	char ifname[IFNAMSIZ]; | ||||||
|  | } __packed; | ||||||
|  | 
 | ||||||
|  | struct brcm_ethhdr { | ||||||
|  | 	u16 subtype; | ||||||
|  | 	u16 length; | ||||||
|  | 	u8 version; | ||||||
|  | 	u8 oui[3]; | ||||||
|  | 	u16 usr_subtype; | ||||||
|  | } __packed; | ||||||
|  | 
 | ||||||
|  | struct brcmf_event { | ||||||
|  | 	struct ethhdr eth; | ||||||
|  | 	struct brcm_ethhdr hdr; | ||||||
|  | 	struct brcmf_event_msg msg; | ||||||
|  | } __packed; | ||||||
|  | 
 | ||||||
|  | struct dngl_stats { | ||||||
|  | 	unsigned long rx_packets;	/* total packets received */ | ||||||
|  | 	unsigned long tx_packets;	/* total packets transmitted */ | ||||||
|  | 	unsigned long rx_bytes;	/* total bytes received */ | ||||||
|  | 	unsigned long tx_bytes;	/* total bytes transmitted */ | ||||||
|  | 	unsigned long rx_errors;	/* bad packets received */ | ||||||
|  | 	unsigned long tx_errors;	/* packet transmit problems */ | ||||||
|  | 	unsigned long rx_dropped;	/* packets dropped by dongle */ | ||||||
|  | 	unsigned long tx_dropped;	/* packets dropped by dongle */ | ||||||
|  | 	unsigned long multicast;	/* multicast packets received */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* event codes sent by the dongle to this driver */ | ||||||
|  | #define BRCMF_E_SET_SSID			0 | ||||||
|  | #define BRCMF_E_JOIN				1 | ||||||
|  | #define BRCMF_E_START				2 | ||||||
|  | #define BRCMF_E_AUTH				3 | ||||||
|  | #define BRCMF_E_AUTH_IND			4 | ||||||
|  | #define BRCMF_E_DEAUTH				5 | ||||||
|  | #define BRCMF_E_DEAUTH_IND			6 | ||||||
|  | #define BRCMF_E_ASSOC				7 | ||||||
|  | #define BRCMF_E_ASSOC_IND			8 | ||||||
|  | #define BRCMF_E_REASSOC				9 | ||||||
|  | #define BRCMF_E_REASSOC_IND			10 | ||||||
|  | #define BRCMF_E_DISASSOC			11 | ||||||
|  | #define BRCMF_E_DISASSOC_IND			12 | ||||||
|  | #define BRCMF_E_QUIET_START			13 | ||||||
|  | #define BRCMF_E_QUIET_END			14 | ||||||
|  | #define BRCMF_E_BEACON_RX			15 | ||||||
|  | #define BRCMF_E_LINK				16 | ||||||
|  | #define BRCMF_E_MIC_ERROR			17 | ||||||
|  | #define BRCMF_E_NDIS_LINK			18 | ||||||
|  | #define BRCMF_E_ROAM				19 | ||||||
|  | #define BRCMF_E_TXFAIL				20 | ||||||
|  | #define BRCMF_E_PMKID_CACHE			21 | ||||||
|  | #define BRCMF_E_RETROGRADE_TSF			22 | ||||||
|  | #define BRCMF_E_PRUNE				23 | ||||||
|  | #define BRCMF_E_AUTOAUTH			24 | ||||||
|  | #define BRCMF_E_EAPOL_MSG			25 | ||||||
|  | #define BRCMF_E_SCAN_COMPLETE			26 | ||||||
|  | #define BRCMF_E_ADDTS_IND			27 | ||||||
|  | #define BRCMF_E_DELTS_IND			28 | ||||||
|  | #define BRCMF_E_BCNSENT_IND			29 | ||||||
|  | #define BRCMF_E_BCNRX_MSG			30 | ||||||
|  | #define BRCMF_E_BCNLOST_MSG			31 | ||||||
|  | #define BRCMF_E_ROAM_PREP			32 | ||||||
|  | #define BRCMF_E_PFN_NET_FOUND			33 | ||||||
|  | #define BRCMF_E_PFN_NET_LOST			34 | ||||||
|  | #define BRCMF_E_RESET_COMPLETE			35 | ||||||
|  | #define BRCMF_E_JOIN_START			36 | ||||||
|  | #define BRCMF_E_ROAM_START			37 | ||||||
|  | #define BRCMF_E_ASSOC_START			38 | ||||||
|  | #define BRCMF_E_IBSS_ASSOC			39 | ||||||
|  | #define BRCMF_E_RADIO				40 | ||||||
|  | #define BRCMF_E_PSM_WATCHDOG			41 | ||||||
|  | #define BRCMF_E_PROBREQ_MSG			44 | ||||||
|  | #define BRCMF_E_SCAN_CONFIRM_IND		45 | ||||||
|  | #define BRCMF_E_PSK_SUP				46 | ||||||
|  | #define BRCMF_E_COUNTRY_CODE_CHANGED		47 | ||||||
|  | #define	BRCMF_E_EXCEEDED_MEDIUM_TIME		48 | ||||||
|  | #define BRCMF_E_ICV_ERROR			49 | ||||||
|  | #define BRCMF_E_UNICAST_DECODE_ERROR		50 | ||||||
|  | #define BRCMF_E_MULTICAST_DECODE_ERROR		51 | ||||||
|  | #define BRCMF_E_TRACE				52 | ||||||
|  | #define BRCMF_E_IF				54 | ||||||
|  | #define BRCMF_E_RSSI				56 | ||||||
|  | #define BRCMF_E_PFN_SCAN_COMPLETE		57 | ||||||
|  | #define BRCMF_E_EXTLOG_MSG			58 | ||||||
|  | #define BRCMF_E_ACTION_FRAME			59 | ||||||
|  | #define BRCMF_E_ACTION_FRAME_COMPLETE		60 | ||||||
|  | #define BRCMF_E_PRE_ASSOC_IND			61 | ||||||
|  | #define BRCMF_E_PRE_REASSOC_IND			62 | ||||||
|  | #define BRCMF_E_CHANNEL_ADOPTED			63 | ||||||
|  | #define BRCMF_E_AP_STARTED			64 | ||||||
|  | #define BRCMF_E_DFS_AP_STOP			65 | ||||||
|  | #define BRCMF_E_DFS_AP_RESUME			66 | ||||||
|  | #define BRCMF_E_RESERVED1			67 | ||||||
|  | #define BRCMF_E_RESERVED2			68 | ||||||
|  | #define BRCMF_E_ESCAN_RESULT			69 | ||||||
|  | #define BRCMF_E_ACTION_FRAME_OFF_CHAN_COMPLETE	70 | ||||||
|  | #define BRCMF_E_DCS_REQUEST			73 | ||||||
|  | 
 | ||||||
|  | #define BRCMF_E_FIFO_CREDIT_MAP			74 | ||||||
|  | 
 | ||||||
|  | #define BRCMF_E_LAST				75 | ||||||
|  | 
 | ||||||
|  | #define BRCMF_E_STATUS_SUCCESS			0 | ||||||
|  | #define BRCMF_E_STATUS_FAIL			1 | ||||||
|  | #define BRCMF_E_STATUS_TIMEOUT			2 | ||||||
|  | #define BRCMF_E_STATUS_NO_NETWORKS		3 | ||||||
|  | #define BRCMF_E_STATUS_ABORT			4 | ||||||
|  | #define BRCMF_E_STATUS_NO_ACK			5 | ||||||
|  | #define BRCMF_E_STATUS_UNSOLICITED		6 | ||||||
|  | #define BRCMF_E_STATUS_ATTEMPT			7 | ||||||
|  | #define BRCMF_E_STATUS_PARTIAL			8 | ||||||
|  | #define BRCMF_E_STATUS_NEWSCAN			9 | ||||||
|  | #define BRCMF_E_STATUS_NEWASSOC			10 | ||||||
|  | #define BRCMF_E_STATUS_11HQUIET			11 | ||||||
|  | #define BRCMF_E_STATUS_SUPPRESS			12 | ||||||
|  | #define BRCMF_E_STATUS_NOCHANS			13 | ||||||
|  | #define BRCMF_E_STATUS_CS_ABORT			15 | ||||||
|  | #define BRCMF_E_STATUS_ERROR			16 | ||||||
|  | 
 | ||||||
|  | #define BRCMF_E_REASON_INITIAL_ASSOC		0 | ||||||
|  | #define BRCMF_E_REASON_LOW_RSSI			1 | ||||||
|  | #define BRCMF_E_REASON_DEAUTH			2 | ||||||
|  | #define BRCMF_E_REASON_DISASSOC			3 | ||||||
|  | #define BRCMF_E_REASON_BCNS_LOST		4 | ||||||
|  | #define BRCMF_E_REASON_MINTXRATE		9 | ||||||
|  | #define BRCMF_E_REASON_TXFAIL			10 | ||||||
|  | 
 | ||||||
|  | #define BRCMF_E_REASON_FAST_ROAM_FAILED		5 | ||||||
|  | #define BRCMF_E_REASON_DIRECTED_ROAM		6 | ||||||
|  | #define BRCMF_E_REASON_TSPEC_REJECTED		7 | ||||||
|  | #define BRCMF_E_REASON_BETTER_AP		8 | ||||||
|  | 
 | ||||||
|  | #define BRCMF_E_PRUNE_ENCR_MISMATCH		1 | ||||||
|  | #define BRCMF_E_PRUNE_BCAST_BSSID		2 | ||||||
|  | #define BRCMF_E_PRUNE_MAC_DENY			3 | ||||||
|  | #define BRCMF_E_PRUNE_MAC_NA			4 | ||||||
|  | #define BRCMF_E_PRUNE_REG_PASSV			5 | ||||||
|  | #define BRCMF_E_PRUNE_SPCT_MGMT			6 | ||||||
|  | #define BRCMF_E_PRUNE_RADAR			7 | ||||||
|  | #define BRCMF_E_RSN_MISMATCH			8 | ||||||
|  | #define BRCMF_E_PRUNE_NO_COMMON_RATES		9 | ||||||
|  | #define BRCMF_E_PRUNE_BASIC_RATES		10 | ||||||
|  | #define BRCMF_E_PRUNE_CIPHER_NA			12 | ||||||
|  | #define BRCMF_E_PRUNE_KNOWN_STA			13 | ||||||
|  | #define BRCMF_E_PRUNE_WDS_PEER			15 | ||||||
|  | #define BRCMF_E_PRUNE_QBSS_LOAD			16 | ||||||
|  | #define BRCMF_E_PRUNE_HOME_AP			17 | ||||||
|  | 
 | ||||||
|  | #define BRCMF_E_SUP_OTHER			0 | ||||||
|  | #define BRCMF_E_SUP_DECRYPT_KEY_DATA		1 | ||||||
|  | #define BRCMF_E_SUP_BAD_UCAST_WEP128		2 | ||||||
|  | #define BRCMF_E_SUP_BAD_UCAST_WEP40		3 | ||||||
|  | #define BRCMF_E_SUP_UNSUP_KEY_LEN		4 | ||||||
|  | #define BRCMF_E_SUP_PW_KEY_CIPHER		5 | ||||||
|  | #define BRCMF_E_SUP_MSG3_TOO_MANY_IE		6 | ||||||
|  | #define BRCMF_E_SUP_MSG3_IE_MISMATCH		7 | ||||||
|  | #define BRCMF_E_SUP_NO_INSTALL_FLAG		8 | ||||||
|  | #define BRCMF_E_SUP_MSG3_NO_GTK			9 | ||||||
|  | #define BRCMF_E_SUP_GRP_KEY_CIPHER		10 | ||||||
|  | #define BRCMF_E_SUP_GRP_MSG1_NO_GTK		11 | ||||||
|  | #define BRCMF_E_SUP_GTK_DECRYPT_FAIL		12 | ||||||
|  | #define BRCMF_E_SUP_SEND_FAIL			13 | ||||||
|  | #define BRCMF_E_SUP_DEAUTH			14 | ||||||
|  | 
 | ||||||
|  | #define BRCMF_E_IF_ADD				1 | ||||||
|  | #define BRCMF_E_IF_DEL				2 | ||||||
|  | #define BRCMF_E_IF_CHANGE			3 | ||||||
|  | 
 | ||||||
|  | #define BRCMF_E_IF_ROLE_STA			0 | ||||||
|  | #define BRCMF_E_IF_ROLE_AP			1 | ||||||
|  | #define BRCMF_E_IF_ROLE_WDS			2 | ||||||
|  | 
 | ||||||
|  | #define BRCMF_E_LINK_BCN_LOSS			1 | ||||||
|  | #define BRCMF_E_LINK_DISASSOC			2 | ||||||
|  | #define BRCMF_E_LINK_ASSOC_REC			3 | ||||||
|  | #define BRCMF_E_LINK_BSSCFG_DIS			4 | ||||||
|  | 
 | ||||||
|  | /* The level of bus communication with the dongle */ | ||||||
|  | enum brcmf_bus_state { | ||||||
|  | 	BRCMF_BUS_DOWN,		/* Not ready for frame transfers */ | ||||||
|  | 	BRCMF_BUS_LOAD,		/* Download access only (CPU reset) */ | ||||||
|  | 	BRCMF_BUS_DATA		/* Ready for frame transfers */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* Pattern matching filter. Specifies an offset within received packets to
 | ||||||
|  |  * start matching, the pattern to match, the size of the pattern, and a bitmask | ||||||
|  |  * that indicates which bits within the pattern should be matched. | ||||||
|  |  */ | ||||||
|  | struct brcmf_pkt_filter_pattern { | ||||||
|  | 	/*
 | ||||||
|  | 	 * Offset within received packet to start pattern matching. | ||||||
|  | 	 * Offset '0' is the first byte of the ethernet header. | ||||||
|  | 	 */ | ||||||
|  | 	u32 offset; | ||||||
|  | 	/* Size of the pattern.  Bitmask must be the same size.*/ | ||||||
|  | 	u32 size_bytes; | ||||||
|  | 	/*
 | ||||||
|  | 	 * Variable length mask and pattern data. mask starts at offset 0. | ||||||
|  | 	 * Pattern immediately follows mask. | ||||||
|  | 	 */ | ||||||
|  | 	u8 mask_and_pattern[1]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* IOVAR "pkt_filter_add" parameter. Used to install packet filters. */ | ||||||
|  | struct brcmf_pkt_filter { | ||||||
|  | 	u32 id;		/* Unique filter id, specified by app. */ | ||||||
|  | 	u32 type;		/* Filter type (WL_PKT_FILTER_TYPE_xxx). */ | ||||||
|  | 	u32 negate_match;	/* Negate the result of filter matches */ | ||||||
|  | 	union {			/* Filter definitions */ | ||||||
|  | 		struct brcmf_pkt_filter_pattern pattern; /* Filter pattern */ | ||||||
|  | 	} u; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* IOVAR "pkt_filter_enable" parameter. */ | ||||||
|  | struct brcmf_pkt_filter_enable { | ||||||
|  | 	u32 id;		/* Unique filter id */ | ||||||
|  | 	u32 enable;		/* Enable/disable bool */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* BSS info structure
 | ||||||
|  |  * Applications MUST CHECK ie_offset field and length field to access IEs and | ||||||
|  |  * next bss_info structure in a vector (in struct brcmf_scan_results) | ||||||
|  |  */ | ||||||
|  | struct brcmf_bss_info { | ||||||
|  | 	__le32 version;		/* version field */ | ||||||
|  | 	__le32 length;		/* byte length of data in this record,
 | ||||||
|  | 				 * starting at version and including IEs | ||||||
|  | 				 */ | ||||||
|  | 	u8 BSSID[ETH_ALEN]; | ||||||
|  | 	__le16 beacon_period;	/* units are Kusec */ | ||||||
|  | 	__le16 capability;	/* Capability information */ | ||||||
|  | 	u8 SSID_len; | ||||||
|  | 	u8 SSID[32]; | ||||||
|  | 	struct { | ||||||
|  | 		__le32 count;   /* # rates in this set */ | ||||||
|  | 		u8 rates[16]; /* rates in 500kbps units w/hi bit set if basic */ | ||||||
|  | 	} rateset;		/* supported rates */ | ||||||
|  | 	__le16 chanspec;	/* chanspec for bss */ | ||||||
|  | 	__le16 atim_window;	/* units are Kusec */ | ||||||
|  | 	u8 dtim_period;	/* DTIM period */ | ||||||
|  | 	__le16 RSSI;		/* receive signal strength (in dBm) */ | ||||||
|  | 	s8 phy_noise;		/* noise (in dBm) */ | ||||||
|  | 
 | ||||||
|  | 	u8 n_cap;		/* BSS is 802.11N Capable */ | ||||||
|  | 	/* 802.11N BSS Capabilities (based on HT_CAP_*): */ | ||||||
|  | 	__le32 nbss_cap; | ||||||
|  | 	u8 ctl_ch;		/* 802.11N BSS control channel number */ | ||||||
|  | 	__le32 reserved32[1];	/* Reserved for expansion of BSS properties */ | ||||||
|  | 	u8 flags;		/* flags */ | ||||||
|  | 	u8 reserved[3];	/* Reserved for expansion of BSS properties */ | ||||||
|  | 	u8 basic_mcs[MCSSET_LEN];	/* 802.11N BSS required MCS set */ | ||||||
|  | 
 | ||||||
|  | 	__le16 ie_offset;	/* offset at which IEs start, from beginning */ | ||||||
|  | 	__le32 ie_length;	/* byte length of Information Elements */ | ||||||
|  | 	__le16 SNR;		/* average SNR of during frame reception */ | ||||||
|  | 	/* Add new fields here */ | ||||||
|  | 	/* variable length Information Elements */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct brcm_rateset_le { | ||||||
|  | 	/* # rates in this set */ | ||||||
|  | 	__le32 count; | ||||||
|  | 	/* rates in 500kbps units w/hi bit set if basic */ | ||||||
|  | 	u8 rates[WL_NUMRATES]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct brcmf_ssid { | ||||||
|  | 	u32 SSID_len; | ||||||
|  | 	unsigned char SSID[32]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct brcmf_ssid_le { | ||||||
|  | 	__le32 SSID_len; | ||||||
|  | 	unsigned char SSID[32]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct brcmf_scan_params_le { | ||||||
|  | 	struct brcmf_ssid_le ssid_le;	/* default: {0, ""} */ | ||||||
|  | 	u8 bssid[ETH_ALEN];	/* default: bcast */ | ||||||
|  | 	s8 bss_type;		/* default: any,
 | ||||||
|  | 				 * DOT11_BSSTYPE_ANY/INFRASTRUCTURE/INDEPENDENT | ||||||
|  | 				 */ | ||||||
|  | 	u8 scan_type;	/* flags, 0 use default */ | ||||||
|  | 	__le32 nprobes;	  /* -1 use default, number of probes per channel */ | ||||||
|  | 	__le32 active_time;	/* -1 use default, dwell time per channel for
 | ||||||
|  | 				 * active scanning | ||||||
|  | 				 */ | ||||||
|  | 	__le32 passive_time;	/* -1 use default, dwell time per channel
 | ||||||
|  | 				 * for passive scanning | ||||||
|  | 				 */ | ||||||
|  | 	__le32 home_time;	/* -1 use default, dwell time for the
 | ||||||
|  | 				 * home channel between channel scans | ||||||
|  | 				 */ | ||||||
|  | 	__le32 channel_num;	/* count of channels and ssids that follow
 | ||||||
|  | 				 * | ||||||
|  | 				 * low half is count of channels in | ||||||
|  | 				 * channel_list, 0 means default (use all | ||||||
|  | 				 * available channels) | ||||||
|  | 				 * | ||||||
|  | 				 * high half is entries in struct brcmf_ssid | ||||||
|  | 				 * array that follows channel_list, aligned for | ||||||
|  | 				 * s32 (4 bytes) meaning an odd channel count | ||||||
|  | 				 * implies a 2-byte pad between end of | ||||||
|  | 				 * channel_list and first ssid | ||||||
|  | 				 * | ||||||
|  | 				 * if ssid count is zero, single ssid in the | ||||||
|  | 				 * fixed parameter portion is assumed, otherwise | ||||||
|  | 				 * ssid in the fixed portion is ignored | ||||||
|  | 				 */ | ||||||
|  | 	__le16 channel_list[1];	/* list of chanspecs */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* incremental scan struct */ | ||||||
|  | struct brcmf_iscan_params_le { | ||||||
|  | 	__le32 version; | ||||||
|  | 	__le16 action; | ||||||
|  | 	__le16 scan_duration; | ||||||
|  | 	struct brcmf_scan_params_le params_le; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct brcmf_scan_results { | ||||||
|  | 	u32 buflen; | ||||||
|  | 	u32 version; | ||||||
|  | 	u32 count; | ||||||
|  | 	struct brcmf_bss_info bss_info[1]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct brcmf_scan_results_le { | ||||||
|  | 	__le32 buflen; | ||||||
|  | 	__le32 version; | ||||||
|  | 	__le32 count; | ||||||
|  | 	struct brcmf_bss_info bss_info[1]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* used for association with a specific BSSID and chanspec list */ | ||||||
|  | struct brcmf_assoc_params_le { | ||||||
|  | 	/* 00:00:00:00:00:00: broadcast scan */ | ||||||
|  | 	u8 bssid[ETH_ALEN]; | ||||||
|  | 	/* 0: all available channels, otherwise count of chanspecs in
 | ||||||
|  | 	 * chanspec_list */ | ||||||
|  | 	__le32 chanspec_num; | ||||||
|  | 	/* list of chanspecs */ | ||||||
|  | 	__le16 chanspec_list[1]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* used for join with or without a specific bssid and channel list */ | ||||||
|  | struct brcmf_join_params { | ||||||
|  | 	struct brcmf_ssid_le ssid_le; | ||||||
|  | 	struct brcmf_assoc_params_le params_le; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* size of brcmf_scan_results not including variable length array */ | ||||||
|  | #define BRCMF_SCAN_RESULTS_FIXED_SIZE \ | ||||||
|  | 	(sizeof(struct brcmf_scan_results) - sizeof(struct brcmf_bss_info)) | ||||||
|  | 
 | ||||||
|  | /* incremental scan results struct */ | ||||||
|  | struct brcmf_iscan_results { | ||||||
|  | 	union { | ||||||
|  | 		u32 status; | ||||||
|  | 		__le32 status_le; | ||||||
|  | 	}; | ||||||
|  | 	union { | ||||||
|  | 		struct brcmf_scan_results results; | ||||||
|  | 		struct brcmf_scan_results_le results_le; | ||||||
|  | 	}; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* size of brcmf_iscan_results not including variable length array */ | ||||||
|  | #define BRCMF_ISCAN_RESULTS_FIXED_SIZE \ | ||||||
|  | 	(BRCMF_SCAN_RESULTS_FIXED_SIZE + \ | ||||||
|  | 	 offsetof(struct brcmf_iscan_results, results)) | ||||||
|  | 
 | ||||||
|  | struct brcmf_wsec_key { | ||||||
|  | 	u32 index;		/* key index */ | ||||||
|  | 	u32 len;		/* key length */ | ||||||
|  | 	u8 data[WLAN_MAX_KEY_LEN];	/* key data */ | ||||||
|  | 	u32 pad_1[18]; | ||||||
|  | 	u32 algo;	/* CRYPTO_ALGO_AES_CCM, CRYPTO_ALGO_WEP128, etc */ | ||||||
|  | 	u32 flags;	/* misc flags */ | ||||||
|  | 	u32 pad_2[3]; | ||||||
|  | 	u32 iv_initialized;	/* has IV been initialized already? */ | ||||||
|  | 	u32 pad_3; | ||||||
|  | 	/* Rx IV */ | ||||||
|  | 	struct { | ||||||
|  | 		u32 hi;	/* upper 32 bits of IV */ | ||||||
|  | 		u16 lo;	/* lower 16 bits of IV */ | ||||||
|  | 	} rxiv; | ||||||
|  | 	u32 pad_4[2]; | ||||||
|  | 	u8 ea[ETH_ALEN];	/* per station */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * dongle requires same struct as above but with fields in little endian order | ||||||
|  |  */ | ||||||
|  | struct brcmf_wsec_key_le { | ||||||
|  | 	__le32 index;		/* key index */ | ||||||
|  | 	__le32 len;		/* key length */ | ||||||
|  | 	u8 data[WLAN_MAX_KEY_LEN];	/* key data */ | ||||||
|  | 	__le32 pad_1[18]; | ||||||
|  | 	__le32 algo;	/* CRYPTO_ALGO_AES_CCM, CRYPTO_ALGO_WEP128, etc */ | ||||||
|  | 	__le32 flags;	/* misc flags */ | ||||||
|  | 	__le32 pad_2[3]; | ||||||
|  | 	__le32 iv_initialized;	/* has IV been initialized already? */ | ||||||
|  | 	__le32 pad_3; | ||||||
|  | 	/* Rx IV */ | ||||||
|  | 	struct { | ||||||
|  | 		__le32 hi;	/* upper 32 bits of IV */ | ||||||
|  | 		__le16 lo;	/* lower 16 bits of IV */ | ||||||
|  | 	} rxiv; | ||||||
|  | 	__le32 pad_4[2]; | ||||||
|  | 	u8 ea[ETH_ALEN];	/* per station */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* Used to get specific STA parameters */ | ||||||
|  | struct brcmf_scb_val_le { | ||||||
|  | 	__le32 val; | ||||||
|  | 	u8 ea[ETH_ALEN]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* channel encoding */ | ||||||
|  | struct brcmf_channel_info_le { | ||||||
|  | 	__le32 hw_channel; | ||||||
|  | 	__le32 target_channel; | ||||||
|  | 	__le32 scan_channel; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* Bus independent dongle command */ | ||||||
|  | struct brcmf_dcmd { | ||||||
|  | 	uint cmd;		/* common dongle cmd definition */ | ||||||
|  | 	void *buf;		/* pointer to user buffer */ | ||||||
|  | 	uint len;		/* length of user buffer */ | ||||||
|  | 	u8 set;			/* get or set request (optional) */ | ||||||
|  | 	uint used;		/* bytes read or written (optional) */ | ||||||
|  | 	uint needed;		/* bytes needed (optional) */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* Forward decls for struct brcmf_pub (see below) */ | ||||||
|  | struct brcmf_bus;		/* device bus info */ | ||||||
|  | struct brcmf_proto;	/* device communication protocol info */ | ||||||
|  | struct brcmf_info;	/* device driver info */ | ||||||
|  | struct brcmf_cfg80211_dev; /* cfg80211 device info */ | ||||||
|  | 
 | ||||||
|  | /* Common structure for module and instance linkage */ | ||||||
|  | struct brcmf_pub { | ||||||
|  | 	/* Linkage ponters */ | ||||||
|  | 	struct brcmf_bus *bus; | ||||||
|  | 	struct brcmf_proto *prot; | ||||||
|  | 	struct brcmf_info *info; | ||||||
|  | 	struct brcmf_cfg80211_dev *config; | ||||||
|  | 
 | ||||||
|  | 	/* Internal brcmf items */ | ||||||
|  | 	bool up;		/* Driver up/down (to OS) */ | ||||||
|  | 	bool txoff;		/* Transmit flow-controlled */ | ||||||
|  | 	enum brcmf_bus_state busstate; | ||||||
|  | 	uint hdrlen;		/* Total BRCMF header length (proto + bus) */ | ||||||
|  | 	uint maxctl;		/* Max size rxctl request from proto to bus */ | ||||||
|  | 	uint rxsz;		/* Rx buffer size bus module should use */ | ||||||
|  | 	u8 wme_dp;		/* wme discard priority */ | ||||||
|  | 
 | ||||||
|  | 	/* Dongle media info */ | ||||||
|  | 	bool iswl;		/* Dongle-resident driver is wl */ | ||||||
|  | 	unsigned long drv_version;	/* Version of dongle-resident driver */ | ||||||
|  | 	u8 mac[ETH_ALEN];		/* MAC address obtained from dongle */ | ||||||
|  | 	struct dngl_stats dstats;	/* Stats for dongle-based data */ | ||||||
|  | 
 | ||||||
|  | 	/* Additional stats for the bus level */ | ||||||
|  | 
 | ||||||
|  | 	/* Data packets sent to dongle */ | ||||||
|  | 	unsigned long tx_packets; | ||||||
|  | 	/* Multicast data packets sent to dongle */ | ||||||
|  | 	unsigned long tx_multicast; | ||||||
|  | 	/* Errors in sending data to dongle */ | ||||||
|  | 	unsigned long tx_errors; | ||||||
|  | 	/* Control packets sent to dongle */ | ||||||
|  | 	unsigned long tx_ctlpkts; | ||||||
|  | 	/* Errors sending control frames to dongle */ | ||||||
|  | 	unsigned long tx_ctlerrs; | ||||||
|  | 	/* Packets sent up the network interface */ | ||||||
|  | 	unsigned long rx_packets; | ||||||
|  | 	/* Multicast packets sent up the network interface */ | ||||||
|  | 	unsigned long rx_multicast; | ||||||
|  | 	/* Errors processing rx data packets */ | ||||||
|  | 	unsigned long rx_errors; | ||||||
|  | 	/* Control frames processed from dongle */ | ||||||
|  | 	unsigned long rx_ctlpkts; | ||||||
|  | 
 | ||||||
|  | 	/* Errors in processing rx control frames */ | ||||||
|  | 	unsigned long rx_ctlerrs; | ||||||
|  | 	/* Packets dropped locally (no memory) */ | ||||||
|  | 	unsigned long rx_dropped; | ||||||
|  | 	/* Packets flushed due to unscheduled sendup thread */ | ||||||
|  | 	unsigned long rx_flushed; | ||||||
|  | 	/* Number of times dpc scheduled by watchdog timer */ | ||||||
|  | 	unsigned long wd_dpc_sched; | ||||||
|  | 
 | ||||||
|  | 	/* Number of packets where header read-ahead was used. */ | ||||||
|  | 	unsigned long rx_readahead_cnt; | ||||||
|  | 	/* Number of tx packets we had to realloc for headroom */ | ||||||
|  | 	unsigned long tx_realloc; | ||||||
|  | 	/* Number of flow control pkts recvd */ | ||||||
|  | 	unsigned long fc_packets; | ||||||
|  | 
 | ||||||
|  | 	/* Last error return */ | ||||||
|  | 	int bcmerror; | ||||||
|  | 	uint tickcnt; | ||||||
|  | 
 | ||||||
|  | 	/* Last error from dongle */ | ||||||
|  | 	int dongle_error; | ||||||
|  | 
 | ||||||
|  | 	/* Suspend disable flag  flag */ | ||||||
|  | 	int suspend_disable_flag;	/* "1" to disable all extra powersaving
 | ||||||
|  | 					 during suspend */ | ||||||
|  | 	int in_suspend;		/* flag set to 1 when early suspend called */ | ||||||
|  | 	int dtim_skip;		/* dtim skip , default 0 means wake each dtim */ | ||||||
|  | 
 | ||||||
|  | 	/* Pkt filter defination */ | ||||||
|  | 	char *pktfilter[100]; | ||||||
|  | 	int pktfilter_count; | ||||||
|  | 
 | ||||||
|  | 	u8 country_code[BRCM_CNTRY_BUF_SZ]; | ||||||
|  | 	char eventmask[BRCMF_EVENTING_MASK_LEN]; | ||||||
|  | 
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct brcmf_if_event { | ||||||
|  | 	u8 ifidx; | ||||||
|  | 	u8 action; | ||||||
|  | 	u8 flags; | ||||||
|  | 	u8 bssidx; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct bcmevent_name { | ||||||
|  | 	uint event; | ||||||
|  | 	const char *name; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | extern const struct bcmevent_name bcmevent_names[]; | ||||||
|  | 
 | ||||||
|  | /* Indication from bus module regarding presence/insertion of dongle.
 | ||||||
|  |  * Return struct brcmf_pub pointer, used as handle to OS module in later calls. | ||||||
|  |  * Returned structure should have bus and prot pointers filled in. | ||||||
|  |  * bus_hdrlen specifies required headroom for bus module header. | ||||||
|  |  */ | ||||||
|  | extern struct brcmf_pub *brcmf_attach(struct brcmf_bus *bus, | ||||||
|  | 				      uint bus_hdrlen); | ||||||
|  | extern int brcmf_net_attach(struct brcmf_pub *drvr, int idx); | ||||||
|  | extern int brcmf_netdev_wait_pend8021x(struct net_device *ndev); | ||||||
|  | 
 | ||||||
|  | extern s32 brcmf_exec_dcmd(struct net_device *dev, u32 cmd, void *arg, u32 len); | ||||||
|  | 
 | ||||||
|  | /* Indication from bus module regarding removal/absence of dongle */ | ||||||
|  | extern void brcmf_detach(struct brcmf_pub *drvr); | ||||||
|  | 
 | ||||||
|  | /* Indication from bus module to change flow-control state */ | ||||||
|  | extern void brcmf_txflowcontrol(struct brcmf_pub *drvr, int ifidx, bool on); | ||||||
|  | 
 | ||||||
|  | extern bool brcmf_c_prec_enq(struct brcmf_pub *drvr, struct pktq *q, | ||||||
|  | 			 struct sk_buff *pkt, int prec); | ||||||
|  | 
 | ||||||
|  | /* Receive frame for delivery to OS.  Callee disposes of rxp. */ | ||||||
|  | extern void brcmf_rx_frame(struct brcmf_pub *drvr, int ifidx, | ||||||
|  | 			 struct sk_buff *rxp, int numpkt); | ||||||
|  | 
 | ||||||
|  | /* Return pointer to interface name */ | ||||||
|  | extern char *brcmf_ifname(struct brcmf_pub *drvr, int idx); | ||||||
|  | 
 | ||||||
|  | /* Notify tx completion */ | ||||||
|  | extern void brcmf_txcomplete(struct brcmf_pub *drvr, struct sk_buff *txp, | ||||||
|  | 			     bool success); | ||||||
|  | 
 | ||||||
|  | /* Query dongle */ | ||||||
|  | extern int brcmf_proto_cdc_query_dcmd(struct brcmf_pub *drvr, int ifidx, | ||||||
|  | 				       uint cmd, void *buf, uint len); | ||||||
|  | 
 | ||||||
|  | /* OS independent layer functions */ | ||||||
|  | extern int brcmf_os_proto_block(struct brcmf_pub *drvr); | ||||||
|  | extern int brcmf_os_proto_unblock(struct brcmf_pub *drvr); | ||||||
|  | #ifdef BCMDBG | ||||||
|  | extern int brcmf_write_to_file(struct brcmf_pub *drvr, u8 *buf, int size); | ||||||
|  | #endif				/* BCMDBG */ | ||||||
|  | 
 | ||||||
|  | extern int brcmf_ifname2idx(struct brcmf_info *drvr_priv, char *name); | ||||||
|  | extern int brcmf_c_host_event(struct brcmf_info *drvr_priv, int *idx, | ||||||
|  | 			      void *pktdata, struct brcmf_event_msg *, | ||||||
|  | 			      void **data_ptr); | ||||||
|  | 
 | ||||||
|  | extern void brcmf_c_init(void); | ||||||
|  | 
 | ||||||
|  | extern int brcmf_add_if(struct brcmf_info *drvr_priv, int ifidx, | ||||||
|  | 			struct net_device *ndev, char *name, u8 *mac_addr, | ||||||
|  | 			u32 flags, u8 bssidx); | ||||||
|  | extern void brcmf_del_if(struct brcmf_info *drvr_priv, int ifidx); | ||||||
|  | 
 | ||||||
|  | /* Send packet to dongle via data channel */ | ||||||
|  | extern int brcmf_sendpkt(struct brcmf_pub *drvr, int ifidx,\ | ||||||
|  | 			 struct sk_buff *pkt); | ||||||
|  | 
 | ||||||
|  | extern int brcmf_bus_start(struct brcmf_pub *drvr); | ||||||
|  | 
 | ||||||
|  | extern void brcmf_c_pktfilter_offload_set(struct brcmf_pub *drvr, char *arg); | ||||||
|  | extern void brcmf_c_pktfilter_offload_enable(struct brcmf_pub *drvr, char *arg, | ||||||
|  | 					     int enable, int master_mode); | ||||||
|  | 
 | ||||||
|  | #define	BRCMF_DCMD_SMLEN	256	/* "small" cmd buffer required */ | ||||||
|  | #define BRCMF_DCMD_MEDLEN	1536	/* "med" cmd buffer required */ | ||||||
|  | #define	BRCMF_DCMD_MAXLEN	8192	/* max length cmd buffer required */ | ||||||
|  | 
 | ||||||
|  | /* message levels */ | ||||||
|  | #define BRCMF_ERROR_VAL	0x0001 | ||||||
|  | #define BRCMF_TRACE_VAL	0x0002 | ||||||
|  | #define BRCMF_INFO_VAL	0x0004 | ||||||
|  | #define BRCMF_DATA_VAL	0x0008 | ||||||
|  | #define BRCMF_CTL_VAL	0x0010 | ||||||
|  | #define BRCMF_TIMER_VAL	0x0020 | ||||||
|  | #define BRCMF_HDRS_VAL	0x0040 | ||||||
|  | #define BRCMF_BYTES_VAL	0x0080 | ||||||
|  | #define BRCMF_INTR_VAL	0x0100 | ||||||
|  | #define BRCMF_GLOM_VAL	0x0400 | ||||||
|  | #define BRCMF_EVENT_VAL	0x0800 | ||||||
|  | #define BRCMF_BTA_VAL	0x1000 | ||||||
|  | #define BRCMF_ISCAN_VAL 0x2000 | ||||||
|  | 
 | ||||||
|  | /* Enter idle immediately (no timeout) */ | ||||||
|  | #define BRCMF_IDLE_IMMEDIATE	(-1) | ||||||
|  | #define BRCMF_IDLE_ACTIVE	0	/* Do not request any SD clock change | ||||||
|  | 				 when idle */ | ||||||
|  | #define BRCMF_IDLE_INTERVAL	1 | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCMF_H_ */ | ||||||
							
								
								
									
										57
									
								
								drivers/net/wireless/brcm80211/brcmfmac/dhd_bus.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								drivers/net/wireless/brcm80211/brcmfmac/dhd_bus.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,57 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCMF_BUS_H_ | ||||||
|  | #define _BRCMF_BUS_H_ | ||||||
|  | 
 | ||||||
|  | /* Packet alignment for most efficient SDIO (can change based on platform) */ | ||||||
|  | #define BRCMF_SDALIGN	(1 << 6) | ||||||
|  | 
 | ||||||
|  | /* watchdog polling interval in ms */ | ||||||
|  | #define BRCMF_WD_POLL_MS	10 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Exported from brcmf bus module (brcmf_usb, brcmf_sdio) | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* Indicate (dis)interest in finding dongles. */ | ||||||
|  | extern int brcmf_bus_register(void); | ||||||
|  | extern void brcmf_bus_unregister(void); | ||||||
|  | 
 | ||||||
|  | /* obtain linux device object providing bus function */ | ||||||
|  | extern struct device *brcmf_bus_get_device(struct brcmf_bus *bus); | ||||||
|  | 
 | ||||||
|  | /* Stop bus module: clear pending frames, disable data flow */ | ||||||
|  | extern void brcmf_sdbrcm_bus_stop(struct brcmf_bus *bus); | ||||||
|  | 
 | ||||||
|  | /* Initialize bus module: prepare for communication w/dongle */ | ||||||
|  | extern int brcmf_sdbrcm_bus_init(struct brcmf_pub *drvr); | ||||||
|  | 
 | ||||||
|  | /* Send a data frame to the dongle.  Callee disposes of txp. */ | ||||||
|  | extern int brcmf_sdbrcm_bus_txdata(struct brcmf_bus *bus, struct sk_buff *txp); | ||||||
|  | 
 | ||||||
|  | /* Send/receive a control message to/from the dongle.
 | ||||||
|  |  * Expects caller to enforce a single outstanding transaction. | ||||||
|  |  */ | ||||||
|  | extern int | ||||||
|  | brcmf_sdbrcm_bus_txctl(struct brcmf_bus *bus, unsigned char *msg, uint msglen); | ||||||
|  | 
 | ||||||
|  | extern int | ||||||
|  | brcmf_sdbrcm_bus_rxctl(struct brcmf_bus *bus, unsigned char *msg, uint msglen); | ||||||
|  | 
 | ||||||
|  | extern void brcmf_sdbrcm_wd_timer(struct brcmf_bus *bus, uint wdtick); | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCMF_BUS_H_ */ | ||||||
							
								
								
									
										498
									
								
								drivers/net/wireless/brcm80211/brcmfmac/dhd_cdc.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										498
									
								
								drivers/net/wireless/brcm80211/brcmfmac/dhd_cdc.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,498 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /*******************************************************************************
 | ||||||
|  |  * Communicates with the dongle by using dcmd codes. | ||||||
|  |  * For certain dcmd codes, the dongle interprets string data from the host. | ||||||
|  |  ******************************************************************************/ | ||||||
|  | 
 | ||||||
|  | #include <linux/types.h> | ||||||
|  | #include <linux/netdevice.h> | ||||||
|  | #include <linux/sched.h> | ||||||
|  | #include <defs.h> | ||||||
|  | 
 | ||||||
|  | #include <brcmu_utils.h> | ||||||
|  | #include <brcmu_wifi.h> | ||||||
|  | 
 | ||||||
|  | #include "dhd.h" | ||||||
|  | #include "dhd_proto.h" | ||||||
|  | #include "dhd_bus.h" | ||||||
|  | #include "dhd_dbg.h" | ||||||
|  | 
 | ||||||
|  | struct brcmf_proto_cdc_dcmd { | ||||||
|  | 	__le32 cmd;	/* dongle command value */ | ||||||
|  | 	__le32 len;	/* lower 16: output buflen;
 | ||||||
|  | 			 * upper 16: input buflen (excludes header) */ | ||||||
|  | 	__le32 flags;	/* flag defns given below */ | ||||||
|  | 	__le32 status;	/* status code returned from the device */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* Max valid buffer size that can be sent to the dongle */ | ||||||
|  | #define CDC_MAX_MSG_SIZE	(ETH_FRAME_LEN+ETH_FCS_LEN) | ||||||
|  | 
 | ||||||
|  | /* CDC flag definitions */ | ||||||
|  | #define CDC_DCMD_ERROR		0x01	/* 1=cmd failed */ | ||||||
|  | #define CDC_DCMD_SET		0x02	/* 0=get, 1=set cmd */ | ||||||
|  | #define CDC_DCMD_IF_MASK	0xF000		/* I/F index */ | ||||||
|  | #define CDC_DCMD_IF_SHIFT	12 | ||||||
|  | #define CDC_DCMD_ID_MASK	0xFFFF0000	/* id an cmd pairing */ | ||||||
|  | #define CDC_DCMD_ID_SHIFT	16		/* ID Mask shift bits */ | ||||||
|  | #define CDC_DCMD_ID(flags)	\ | ||||||
|  | 	(((flags) & CDC_DCMD_ID_MASK) >> CDC_DCMD_ID_SHIFT) | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * BDC header - Broadcom specific extension of CDC. | ||||||
|  |  * Used on data packets to convey priority across USB. | ||||||
|  |  */ | ||||||
|  | #define	BDC_HEADER_LEN		4 | ||||||
|  | #define BDC_PROTO_VER		1	/* Protocol version */ | ||||||
|  | #define BDC_FLAG_VER_MASK	0xf0	/* Protocol version mask */ | ||||||
|  | #define BDC_FLAG_VER_SHIFT	4	/* Protocol version shift */ | ||||||
|  | #define BDC_FLAG_SUM_GOOD	0x04	/* Good RX checksums */ | ||||||
|  | #define BDC_FLAG_SUM_NEEDED	0x08	/* Dongle needs to do TX checksums */ | ||||||
|  | #define BDC_PRIORITY_MASK	0x7 | ||||||
|  | #define BDC_FLAG2_IF_MASK	0x0f	/* packet rx interface in APSTA */ | ||||||
|  | #define BDC_FLAG2_IF_SHIFT	0 | ||||||
|  | 
 | ||||||
|  | #define BDC_GET_IF_IDX(hdr) \ | ||||||
|  | 	((int)((((hdr)->flags2) & BDC_FLAG2_IF_MASK) >> BDC_FLAG2_IF_SHIFT)) | ||||||
|  | #define BDC_SET_IF_IDX(hdr, idx) \ | ||||||
|  | 	((hdr)->flags2 = (((hdr)->flags2 & ~BDC_FLAG2_IF_MASK) | \ | ||||||
|  | 	((idx) << BDC_FLAG2_IF_SHIFT))) | ||||||
|  | 
 | ||||||
|  | struct brcmf_proto_bdc_header { | ||||||
|  | 	u8 flags; | ||||||
|  | 	u8 priority;	/* 802.1d Priority, 4:7 flow control info for usb */ | ||||||
|  | 	u8 flags2; | ||||||
|  | 	u8 rssi; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #define RETRIES 2 /* # of retries to retrieve matching dcmd response */ | ||||||
|  | #define BUS_HEADER_LEN	(16+BRCMF_SDALIGN) /* Must be atleast SDPCM_RESERVE | ||||||
|  | 					 * (amount of header tha might be added) | ||||||
|  | 					 * plus any space that might be needed | ||||||
|  | 					 * for alignment padding. | ||||||
|  | 					 */ | ||||||
|  | #define ROUND_UP_MARGIN	2048	/* Biggest SDIO block size possible for | ||||||
|  | 				 * round off at the end of buffer | ||||||
|  | 				 */ | ||||||
|  | 
 | ||||||
|  | struct brcmf_proto { | ||||||
|  | 	u16 reqid; | ||||||
|  | 	u8 pending; | ||||||
|  | 	u32 lastcmd; | ||||||
|  | 	u8 bus_header[BUS_HEADER_LEN]; | ||||||
|  | 	struct brcmf_proto_cdc_dcmd msg; | ||||||
|  | 	unsigned char buf[BRCMF_DCMD_MAXLEN + ROUND_UP_MARGIN]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static int brcmf_proto_cdc_msg(struct brcmf_pub *drvr) | ||||||
|  | { | ||||||
|  | 	struct brcmf_proto *prot = drvr->prot; | ||||||
|  | 	int len = le32_to_cpu(prot->msg.len) + | ||||||
|  | 			sizeof(struct brcmf_proto_cdc_dcmd); | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(TRACE, "Enter\n"); | ||||||
|  | 
 | ||||||
|  | 	/* NOTE : cdc->msg.len holds the desired length of the buffer to be
 | ||||||
|  | 	 *        returned. Only up to CDC_MAX_MSG_SIZE of this buffer area | ||||||
|  | 	 *        is actually sent to the dongle | ||||||
|  | 	 */ | ||||||
|  | 	if (len > CDC_MAX_MSG_SIZE) | ||||||
|  | 		len = CDC_MAX_MSG_SIZE; | ||||||
|  | 
 | ||||||
|  | 	/* Send request */ | ||||||
|  | 	return brcmf_sdbrcm_bus_txctl(drvr->bus, (unsigned char *)&prot->msg, | ||||||
|  | 				      len); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int brcmf_proto_cdc_cmplt(struct brcmf_pub *drvr, u32 id, u32 len) | ||||||
|  | { | ||||||
|  | 	int ret; | ||||||
|  | 	struct brcmf_proto *prot = drvr->prot; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(TRACE, "Enter\n"); | ||||||
|  | 
 | ||||||
|  | 	do { | ||||||
|  | 		ret = brcmf_sdbrcm_bus_rxctl(drvr->bus, | ||||||
|  | 				(unsigned char *)&prot->msg, | ||||||
|  | 				len + sizeof(struct brcmf_proto_cdc_dcmd)); | ||||||
|  | 		if (ret < 0) | ||||||
|  | 			break; | ||||||
|  | 	} while (CDC_DCMD_ID(le32_to_cpu(prot->msg.flags)) != id); | ||||||
|  | 
 | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | brcmf_proto_cdc_query_dcmd(struct brcmf_pub *drvr, int ifidx, uint cmd, | ||||||
|  | 			       void *buf, uint len) | ||||||
|  | { | ||||||
|  | 	struct brcmf_proto *prot = drvr->prot; | ||||||
|  | 	struct brcmf_proto_cdc_dcmd *msg = &prot->msg; | ||||||
|  | 	void *info; | ||||||
|  | 	int ret = 0, retries = 0; | ||||||
|  | 	u32 id, flags; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(TRACE, "Enter\n"); | ||||||
|  | 	brcmf_dbg(CTL, "cmd %d len %d\n", cmd, len); | ||||||
|  | 
 | ||||||
|  | 	/* Respond "bcmerror" and "bcmerrorstr" with local cache */ | ||||||
|  | 	if (cmd == BRCMF_C_GET_VAR && buf) { | ||||||
|  | 		if (!strcmp((char *)buf, "bcmerrorstr")) { | ||||||
|  | 			strncpy((char *)buf, "bcm_error", | ||||||
|  | 				BCME_STRLEN); | ||||||
|  | 			goto done; | ||||||
|  | 		} else if (!strcmp((char *)buf, "bcmerror")) { | ||||||
|  | 			*(int *)buf = drvr->dongle_error; | ||||||
|  | 			goto done; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	memset(msg, 0, sizeof(struct brcmf_proto_cdc_dcmd)); | ||||||
|  | 
 | ||||||
|  | 	msg->cmd = cpu_to_le32(cmd); | ||||||
|  | 	msg->len = cpu_to_le32(len); | ||||||
|  | 	flags = (++prot->reqid << CDC_DCMD_ID_SHIFT); | ||||||
|  | 	flags = (flags & ~CDC_DCMD_IF_MASK) | | ||||||
|  | 		(ifidx << CDC_DCMD_IF_SHIFT); | ||||||
|  | 	msg->flags = cpu_to_le32(flags); | ||||||
|  | 
 | ||||||
|  | 	if (buf) | ||||||
|  | 		memcpy(prot->buf, buf, len); | ||||||
|  | 
 | ||||||
|  | 	ret = brcmf_proto_cdc_msg(drvr); | ||||||
|  | 	if (ret < 0) { | ||||||
|  | 		brcmf_dbg(ERROR, "brcmf_proto_cdc_msg failed w/status %d\n", | ||||||
|  | 			  ret); | ||||||
|  | 		goto done; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | retry: | ||||||
|  | 	/* wait for interrupt and get first fragment */ | ||||||
|  | 	ret = brcmf_proto_cdc_cmplt(drvr, prot->reqid, len); | ||||||
|  | 	if (ret < 0) | ||||||
|  | 		goto done; | ||||||
|  | 
 | ||||||
|  | 	flags = le32_to_cpu(msg->flags); | ||||||
|  | 	id = (flags & CDC_DCMD_ID_MASK) >> CDC_DCMD_ID_SHIFT; | ||||||
|  | 
 | ||||||
|  | 	if ((id < prot->reqid) && (++retries < RETRIES)) | ||||||
|  | 		goto retry; | ||||||
|  | 	if (id != prot->reqid) { | ||||||
|  | 		brcmf_dbg(ERROR, "%s: unexpected request id %d (expected %d)\n", | ||||||
|  | 			  brcmf_ifname(drvr, ifidx), id, prot->reqid); | ||||||
|  | 		ret = -EINVAL; | ||||||
|  | 		goto done; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Check info buffer */ | ||||||
|  | 	info = (void *)&msg[1]; | ||||||
|  | 
 | ||||||
|  | 	/* Copy info buffer */ | ||||||
|  | 	if (buf) { | ||||||
|  | 		if (ret < (int)len) | ||||||
|  | 			len = ret; | ||||||
|  | 		memcpy(buf, info, len); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Check the ERROR flag */ | ||||||
|  | 	if (flags & CDC_DCMD_ERROR) { | ||||||
|  | 		ret = le32_to_cpu(msg->status); | ||||||
|  | 		/* Cache error from dongle */ | ||||||
|  | 		drvr->dongle_error = ret; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | done: | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int brcmf_proto_cdc_set_dcmd(struct brcmf_pub *drvr, int ifidx, uint cmd, | ||||||
|  | 				 void *buf, uint len) | ||||||
|  | { | ||||||
|  | 	struct brcmf_proto *prot = drvr->prot; | ||||||
|  | 	struct brcmf_proto_cdc_dcmd *msg = &prot->msg; | ||||||
|  | 	int ret = 0; | ||||||
|  | 	u32 flags, id; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(TRACE, "Enter\n"); | ||||||
|  | 	brcmf_dbg(CTL, "cmd %d len %d\n", cmd, len); | ||||||
|  | 
 | ||||||
|  | 	memset(msg, 0, sizeof(struct brcmf_proto_cdc_dcmd)); | ||||||
|  | 
 | ||||||
|  | 	msg->cmd = cpu_to_le32(cmd); | ||||||
|  | 	msg->len = cpu_to_le32(len); | ||||||
|  | 	flags = (++prot->reqid << CDC_DCMD_ID_SHIFT) | CDC_DCMD_SET; | ||||||
|  | 	flags = (flags & ~CDC_DCMD_IF_MASK) | | ||||||
|  | 		(ifidx << CDC_DCMD_IF_SHIFT); | ||||||
|  | 	msg->flags = cpu_to_le32(flags); | ||||||
|  | 
 | ||||||
|  | 	if (buf) | ||||||
|  | 		memcpy(prot->buf, buf, len); | ||||||
|  | 
 | ||||||
|  | 	ret = brcmf_proto_cdc_msg(drvr); | ||||||
|  | 	if (ret < 0) | ||||||
|  | 		goto done; | ||||||
|  | 
 | ||||||
|  | 	ret = brcmf_proto_cdc_cmplt(drvr, prot->reqid, len); | ||||||
|  | 	if (ret < 0) | ||||||
|  | 		goto done; | ||||||
|  | 
 | ||||||
|  | 	flags = le32_to_cpu(msg->flags); | ||||||
|  | 	id = (flags & CDC_DCMD_ID_MASK) >> CDC_DCMD_ID_SHIFT; | ||||||
|  | 
 | ||||||
|  | 	if (id != prot->reqid) { | ||||||
|  | 		brcmf_dbg(ERROR, "%s: unexpected request id %d (expected %d)\n", | ||||||
|  | 			  brcmf_ifname(drvr, ifidx), id, prot->reqid); | ||||||
|  | 		ret = -EINVAL; | ||||||
|  | 		goto done; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Check the ERROR flag */ | ||||||
|  | 	if (flags & CDC_DCMD_ERROR) { | ||||||
|  | 		ret = le32_to_cpu(msg->status); | ||||||
|  | 		/* Cache error from dongle */ | ||||||
|  | 		drvr->dongle_error = ret; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | done: | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | brcmf_proto_dcmd(struct brcmf_pub *drvr, int ifidx, struct brcmf_dcmd *dcmd, | ||||||
|  | 		  int len) | ||||||
|  | { | ||||||
|  | 	struct brcmf_proto *prot = drvr->prot; | ||||||
|  | 	int ret = -1; | ||||||
|  | 
 | ||||||
|  | 	if (drvr->busstate == BRCMF_BUS_DOWN) { | ||||||
|  | 		brcmf_dbg(ERROR, "bus is down. we have nothing to do.\n"); | ||||||
|  | 		return ret; | ||||||
|  | 	} | ||||||
|  | 	brcmf_os_proto_block(drvr); | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(TRACE, "Enter\n"); | ||||||
|  | 
 | ||||||
|  | 	if (len > BRCMF_DCMD_MAXLEN) | ||||||
|  | 		goto done; | ||||||
|  | 
 | ||||||
|  | 	if (prot->pending == true) { | ||||||
|  | 		brcmf_dbg(TRACE, "CDC packet is pending!!!! cmd=0x%x (%lu) lastcmd=0x%x (%lu)\n", | ||||||
|  | 			  dcmd->cmd, (unsigned long)dcmd->cmd, prot->lastcmd, | ||||||
|  | 			  (unsigned long)prot->lastcmd); | ||||||
|  | 		if (dcmd->cmd == BRCMF_C_SET_VAR || | ||||||
|  | 		    dcmd->cmd == BRCMF_C_GET_VAR) | ||||||
|  | 			brcmf_dbg(TRACE, "iovar cmd=%s\n", (char *)dcmd->buf); | ||||||
|  | 
 | ||||||
|  | 		goto done; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	prot->pending = true; | ||||||
|  | 	prot->lastcmd = dcmd->cmd; | ||||||
|  | 	if (dcmd->set) | ||||||
|  | 		ret = brcmf_proto_cdc_set_dcmd(drvr, ifidx, dcmd->cmd, | ||||||
|  | 						   dcmd->buf, len); | ||||||
|  | 	else { | ||||||
|  | 		ret = brcmf_proto_cdc_query_dcmd(drvr, ifidx, dcmd->cmd, | ||||||
|  | 						     dcmd->buf, len); | ||||||
|  | 		if (ret > 0) | ||||||
|  | 			dcmd->used = ret - | ||||||
|  | 					sizeof(struct brcmf_proto_cdc_dcmd); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (ret >= 0) | ||||||
|  | 		ret = 0; | ||||||
|  | 	else { | ||||||
|  | 		struct brcmf_proto_cdc_dcmd *msg = &prot->msg; | ||||||
|  | 		/* len == needed when set/query fails from dongle */ | ||||||
|  | 		dcmd->needed = le32_to_cpu(msg->len); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Intercept the wme_dp dongle cmd here */ | ||||||
|  | 	if (!ret && dcmd->cmd == BRCMF_C_SET_VAR && | ||||||
|  | 	    !strcmp(dcmd->buf, "wme_dp")) { | ||||||
|  | 		int slen; | ||||||
|  | 		__le32 val = 0; | ||||||
|  | 
 | ||||||
|  | 		slen = strlen("wme_dp") + 1; | ||||||
|  | 		if (len >= (int)(slen + sizeof(int))) | ||||||
|  | 			memcpy(&val, (char *)dcmd->buf + slen, sizeof(int)); | ||||||
|  | 		drvr->wme_dp = (u8) le32_to_cpu(val); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	prot->pending = false; | ||||||
|  | 
 | ||||||
|  | done: | ||||||
|  | 	brcmf_os_proto_unblock(drvr); | ||||||
|  | 
 | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static bool pkt_sum_needed(struct sk_buff *skb) | ||||||
|  | { | ||||||
|  | 	return skb->ip_summed == CHECKSUM_PARTIAL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void pkt_set_sum_good(struct sk_buff *skb, bool x) | ||||||
|  | { | ||||||
|  | 	skb->ip_summed = (x ? CHECKSUM_UNNECESSARY : CHECKSUM_NONE); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void brcmf_proto_hdrpush(struct brcmf_pub *drvr, int ifidx, | ||||||
|  | 			 struct sk_buff *pktbuf) | ||||||
|  | { | ||||||
|  | 	struct brcmf_proto_bdc_header *h; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(TRACE, "Enter\n"); | ||||||
|  | 
 | ||||||
|  | 	/* Push BDC header used to convey priority for buses that don't */ | ||||||
|  | 
 | ||||||
|  | 	skb_push(pktbuf, BDC_HEADER_LEN); | ||||||
|  | 
 | ||||||
|  | 	h = (struct brcmf_proto_bdc_header *)(pktbuf->data); | ||||||
|  | 
 | ||||||
|  | 	h->flags = (BDC_PROTO_VER << BDC_FLAG_VER_SHIFT); | ||||||
|  | 	if (pkt_sum_needed(pktbuf)) | ||||||
|  | 		h->flags |= BDC_FLAG_SUM_NEEDED; | ||||||
|  | 
 | ||||||
|  | 	h->priority = (pktbuf->priority & BDC_PRIORITY_MASK); | ||||||
|  | 	h->flags2 = 0; | ||||||
|  | 	h->rssi = 0; | ||||||
|  | 	BDC_SET_IF_IDX(h, ifidx); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int brcmf_proto_hdrpull(struct brcmf_pub *drvr, int *ifidx, | ||||||
|  | 			struct sk_buff *pktbuf) | ||||||
|  | { | ||||||
|  | 	struct brcmf_proto_bdc_header *h; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(TRACE, "Enter\n"); | ||||||
|  | 
 | ||||||
|  | 	/* Pop BDC header used to convey priority for buses that don't */ | ||||||
|  | 
 | ||||||
|  | 	if (pktbuf->len < BDC_HEADER_LEN) { | ||||||
|  | 		brcmf_dbg(ERROR, "rx data too short (%d < %d)\n", | ||||||
|  | 			  pktbuf->len, BDC_HEADER_LEN); | ||||||
|  | 		return -EBADE; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	h = (struct brcmf_proto_bdc_header *)(pktbuf->data); | ||||||
|  | 
 | ||||||
|  | 	*ifidx = BDC_GET_IF_IDX(h); | ||||||
|  | 	if (*ifidx >= BRCMF_MAX_IFS) { | ||||||
|  | 		brcmf_dbg(ERROR, "rx data ifnum out of range (%d)\n", *ifidx); | ||||||
|  | 		return -EBADE; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (((h->flags & BDC_FLAG_VER_MASK) >> BDC_FLAG_VER_SHIFT) != | ||||||
|  | 	    BDC_PROTO_VER) { | ||||||
|  | 		brcmf_dbg(ERROR, "%s: non-BDC packet received, flags 0x%x\n", | ||||||
|  | 			  brcmf_ifname(drvr, *ifidx), h->flags); | ||||||
|  | 		return -EBADE; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (h->flags & BDC_FLAG_SUM_GOOD) { | ||||||
|  | 		brcmf_dbg(INFO, "%s: BDC packet received with good rx-csum, flags 0x%x\n", | ||||||
|  | 			  brcmf_ifname(drvr, *ifidx), h->flags); | ||||||
|  | 		pkt_set_sum_good(pktbuf, true); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	pktbuf->priority = h->priority & BDC_PRIORITY_MASK; | ||||||
|  | 
 | ||||||
|  | 	skb_pull(pktbuf, BDC_HEADER_LEN); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int brcmf_proto_attach(struct brcmf_pub *drvr) | ||||||
|  | { | ||||||
|  | 	struct brcmf_proto *cdc; | ||||||
|  | 
 | ||||||
|  | 	cdc = kzalloc(sizeof(struct brcmf_proto), GFP_ATOMIC); | ||||||
|  | 	if (!cdc) | ||||||
|  | 		goto fail; | ||||||
|  | 
 | ||||||
|  | 	/* ensure that the msg buf directly follows the cdc msg struct */ | ||||||
|  | 	if ((unsigned long)(&cdc->msg + 1) != (unsigned long)cdc->buf) { | ||||||
|  | 		brcmf_dbg(ERROR, "struct brcmf_proto is not correctly defined\n"); | ||||||
|  | 		goto fail; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	drvr->prot = cdc; | ||||||
|  | 	drvr->hdrlen += BDC_HEADER_LEN; | ||||||
|  | 	drvr->maxctl = BRCMF_DCMD_MAXLEN + | ||||||
|  | 			sizeof(struct brcmf_proto_cdc_dcmd) + ROUND_UP_MARGIN; | ||||||
|  | 	return 0; | ||||||
|  | 
 | ||||||
|  | fail: | ||||||
|  | 	kfree(cdc); | ||||||
|  | 	return -ENOMEM; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* ~NOTE~ What if another thread is waiting on the semaphore?  Holding it? */ | ||||||
|  | void brcmf_proto_detach(struct brcmf_pub *drvr) | ||||||
|  | { | ||||||
|  | 	kfree(drvr->prot); | ||||||
|  | 	drvr->prot = NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void brcmf_proto_dstats(struct brcmf_pub *drvr) | ||||||
|  | { | ||||||
|  | 	/* No stats from dongle added yet, copy bus stats */ | ||||||
|  | 	drvr->dstats.tx_packets = drvr->tx_packets; | ||||||
|  | 	drvr->dstats.tx_errors = drvr->tx_errors; | ||||||
|  | 	drvr->dstats.rx_packets = drvr->rx_packets; | ||||||
|  | 	drvr->dstats.rx_errors = drvr->rx_errors; | ||||||
|  | 	drvr->dstats.rx_dropped = drvr->rx_dropped; | ||||||
|  | 	drvr->dstats.multicast = drvr->rx_multicast; | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int brcmf_proto_init(struct brcmf_pub *drvr) | ||||||
|  | { | ||||||
|  | 	int ret = 0; | ||||||
|  | 	char buf[128]; | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(TRACE, "Enter\n"); | ||||||
|  | 
 | ||||||
|  | 	brcmf_os_proto_block(drvr); | ||||||
|  | 
 | ||||||
|  | 	/* Get the device MAC address */ | ||||||
|  | 	strcpy(buf, "cur_etheraddr"); | ||||||
|  | 	ret = brcmf_proto_cdc_query_dcmd(drvr, 0, BRCMF_C_GET_VAR, | ||||||
|  | 					  buf, sizeof(buf)); | ||||||
|  | 	if (ret < 0) { | ||||||
|  | 		brcmf_os_proto_unblock(drvr); | ||||||
|  | 		return ret; | ||||||
|  | 	} | ||||||
|  | 	memcpy(drvr->mac, buf, ETH_ALEN); | ||||||
|  | 
 | ||||||
|  | 	brcmf_os_proto_unblock(drvr); | ||||||
|  | 
 | ||||||
|  | 	ret = brcmf_c_preinit_dcmds(drvr); | ||||||
|  | 
 | ||||||
|  | 	/* Always assumes wl for now */ | ||||||
|  | 	drvr->iswl = true; | ||||||
|  | 
 | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void brcmf_proto_stop(struct brcmf_pub *drvr) | ||||||
|  | { | ||||||
|  | 	/* Nothing to do for CDC */ | ||||||
|  | } | ||||||
							
								
								
									
										872
									
								
								drivers/net/wireless/brcm80211/brcmfmac/dhd_common.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										872
									
								
								drivers/net/wireless/brcm80211/brcmfmac/dhd_common.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,872 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | #include <linux/kernel.h> | ||||||
|  | #include <linux/string.h> | ||||||
|  | #include <linux/sched.h> | ||||||
|  | #include <linux/netdevice.h> | ||||||
|  | #include <asm/unaligned.h> | ||||||
|  | #include <defs.h> | ||||||
|  | #include <brcmu_wifi.h> | ||||||
|  | #include <brcmu_utils.h> | ||||||
|  | #include "dhd.h" | ||||||
|  | #include "dhd_bus.h" | ||||||
|  | #include "dhd_proto.h" | ||||||
|  | #include "dhd_dbg.h" | ||||||
|  | 
 | ||||||
|  | #define BRCM_OUI			"\x00\x10\x18" | ||||||
|  | #define DOT11_OUI_LEN			3 | ||||||
|  | #define BCMILCP_BCM_SUBTYPE_EVENT	1 | ||||||
|  | #define PKTFILTER_BUF_SIZE		2048 | ||||||
|  | #define BRCMF_ARPOL_MODE		0xb	/* agent|snoop|peer_autoreply */ | ||||||
|  | 
 | ||||||
|  | int brcmf_msg_level; | ||||||
|  | 
 | ||||||
|  | #define MSGTRACE_VERSION	1 | ||||||
|  | 
 | ||||||
|  | #define BRCMF_PKT_FILTER_FIXED_LEN	offsetof(struct brcmf_pkt_filter, u) | ||||||
|  | #define BRCMF_PKT_FILTER_PATTERN_FIXED_LEN	\ | ||||||
|  | 	offsetof(struct brcmf_pkt_filter_pattern, mask_and_pattern) | ||||||
|  | 
 | ||||||
|  | #ifdef BCMDBG | ||||||
|  | static const char brcmf_version[] = | ||||||
|  | 	"Dongle Host Driver, version " BRCMF_VERSION_STR "\nCompiled on " | ||||||
|  | 	__DATE__ " at " __TIME__; | ||||||
|  | #else | ||||||
|  | static const char brcmf_version[] = | ||||||
|  | 	"Dongle Host Driver, version " BRCMF_VERSION_STR; | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | /* Message trace header */ | ||||||
|  | struct msgtrace_hdr { | ||||||
|  | 	u8 version; | ||||||
|  | 	u8 spare; | ||||||
|  | 	__be16 len;		/* Len of the trace */ | ||||||
|  | 	__be32 seqnum;		/* Sequence number of message. Useful
 | ||||||
|  | 				 * if the messsage has been lost | ||||||
|  | 				 * because of DMA error or a bus reset | ||||||
|  | 				 * (ex: SDIO Func2) | ||||||
|  | 				 */ | ||||||
|  | 	__be32 discarded_bytes;	/* Number of discarded bytes because of
 | ||||||
|  | 				 trace overflow  */ | ||||||
|  | 	__be32 discarded_printf;	/* Number of discarded printf
 | ||||||
|  | 				 because of trace overflow */ | ||||||
|  | } __packed; | ||||||
|  | 
 | ||||||
|  | void brcmf_c_init(void) | ||||||
|  | { | ||||||
|  | 	/* Init global variables at run-time, not as part of the declaration.
 | ||||||
|  | 	 * This is required to support init/de-init of the driver. | ||||||
|  | 	 * Initialization | ||||||
|  | 	 * of globals as part of the declaration results in non-deterministic | ||||||
|  | 	 * behaviour since the value of the globals may be different on the | ||||||
|  | 	 * first time that the driver is initialized vs subsequent | ||||||
|  | 	 * initializations. | ||||||
|  | 	 */ | ||||||
|  | 	brcmf_msg_level = BRCMF_ERROR_VAL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool brcmf_c_prec_enq(struct brcmf_pub *drvr, struct pktq *q, | ||||||
|  | 		      struct sk_buff *pkt, int prec) | ||||||
|  | { | ||||||
|  | 	struct sk_buff *p; | ||||||
|  | 	int eprec = -1;		/* precedence to evict from */ | ||||||
|  | 	bool discard_oldest; | ||||||
|  | 
 | ||||||
|  | 	/* Fast case, precedence queue is not full and we are also not
 | ||||||
|  | 	 * exceeding total queue length | ||||||
|  | 	 */ | ||||||
|  | 	if (!pktq_pfull(q, prec) && !pktq_full(q)) { | ||||||
|  | 		brcmu_pktq_penq(q, prec, pkt); | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Determine precedence from which to evict packet, if any */ | ||||||
|  | 	if (pktq_pfull(q, prec)) | ||||||
|  | 		eprec = prec; | ||||||
|  | 	else if (pktq_full(q)) { | ||||||
|  | 		p = brcmu_pktq_peek_tail(q, &eprec); | ||||||
|  | 		if (eprec > prec) | ||||||
|  | 			return false; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Evict if needed */ | ||||||
|  | 	if (eprec >= 0) { | ||||||
|  | 		/* Detect queueing to unconfigured precedence */ | ||||||
|  | 		discard_oldest = ac_bitmap_tst(drvr->wme_dp, eprec); | ||||||
|  | 		if (eprec == prec && !discard_oldest) | ||||||
|  | 			return false;	/* refuse newer (incoming) packet */ | ||||||
|  | 		/* Evict packet according to discard policy */ | ||||||
|  | 		p = discard_oldest ? brcmu_pktq_pdeq(q, eprec) : | ||||||
|  | 			brcmu_pktq_pdeq_tail(q, eprec); | ||||||
|  | 		if (p == NULL) | ||||||
|  | 			brcmf_dbg(ERROR, "brcmu_pktq_penq() failed, oldest %d\n", | ||||||
|  | 				  discard_oldest); | ||||||
|  | 
 | ||||||
|  | 		brcmu_pkt_buf_free_skb(p); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Enqueue */ | ||||||
|  | 	p = brcmu_pktq_penq(q, prec, pkt); | ||||||
|  | 	if (p == NULL) | ||||||
|  | 		brcmf_dbg(ERROR, "brcmu_pktq_penq() failed\n"); | ||||||
|  | 
 | ||||||
|  | 	return p != NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #ifdef BCMDBG | ||||||
|  | static void | ||||||
|  | brcmf_c_show_host_event(struct brcmf_event_msg *event, void *event_data) | ||||||
|  | { | ||||||
|  | 	uint i, status, reason; | ||||||
|  | 	bool group = false, flush_txq = false, link = false; | ||||||
|  | 	char *auth_str, *event_name; | ||||||
|  | 	unsigned char *buf; | ||||||
|  | 	char err_msg[256], eabuf[ETHER_ADDR_STR_LEN]; | ||||||
|  | 	static struct { | ||||||
|  | 		uint event; | ||||||
|  | 		char *event_name; | ||||||
|  | 	} event_names[] = { | ||||||
|  | 		{ | ||||||
|  | 		BRCMF_E_SET_SSID, "SET_SSID"}, { | ||||||
|  | 		BRCMF_E_JOIN, "JOIN"}, { | ||||||
|  | 		BRCMF_E_START, "START"}, { | ||||||
|  | 		BRCMF_E_AUTH, "AUTH"}, { | ||||||
|  | 		BRCMF_E_AUTH_IND, "AUTH_IND"}, { | ||||||
|  | 		BRCMF_E_DEAUTH, "DEAUTH"}, { | ||||||
|  | 		BRCMF_E_DEAUTH_IND, "DEAUTH_IND"}, { | ||||||
|  | 		BRCMF_E_ASSOC, "ASSOC"}, { | ||||||
|  | 		BRCMF_E_ASSOC_IND, "ASSOC_IND"}, { | ||||||
|  | 		BRCMF_E_REASSOC, "REASSOC"}, { | ||||||
|  | 		BRCMF_E_REASSOC_IND, "REASSOC_IND"}, { | ||||||
|  | 		BRCMF_E_DISASSOC, "DISASSOC"}, { | ||||||
|  | 		BRCMF_E_DISASSOC_IND, "DISASSOC_IND"}, { | ||||||
|  | 		BRCMF_E_QUIET_START, "START_QUIET"}, { | ||||||
|  | 		BRCMF_E_QUIET_END, "END_QUIET"}, { | ||||||
|  | 		BRCMF_E_BEACON_RX, "BEACON_RX"}, { | ||||||
|  | 		BRCMF_E_LINK, "LINK"}, { | ||||||
|  | 		BRCMF_E_MIC_ERROR, "MIC_ERROR"}, { | ||||||
|  | 		BRCMF_E_NDIS_LINK, "NDIS_LINK"}, { | ||||||
|  | 		BRCMF_E_ROAM, "ROAM"}, { | ||||||
|  | 		BRCMF_E_TXFAIL, "TXFAIL"}, { | ||||||
|  | 		BRCMF_E_PMKID_CACHE, "PMKID_CACHE"}, { | ||||||
|  | 		BRCMF_E_RETROGRADE_TSF, "RETROGRADE_TSF"}, { | ||||||
|  | 		BRCMF_E_PRUNE, "PRUNE"}, { | ||||||
|  | 		BRCMF_E_AUTOAUTH, "AUTOAUTH"}, { | ||||||
|  | 		BRCMF_E_EAPOL_MSG, "EAPOL_MSG"}, { | ||||||
|  | 		BRCMF_E_SCAN_COMPLETE, "SCAN_COMPLETE"}, { | ||||||
|  | 		BRCMF_E_ADDTS_IND, "ADDTS_IND"}, { | ||||||
|  | 		BRCMF_E_DELTS_IND, "DELTS_IND"}, { | ||||||
|  | 		BRCMF_E_BCNSENT_IND, "BCNSENT_IND"}, { | ||||||
|  | 		BRCMF_E_BCNRX_MSG, "BCNRX_MSG"}, { | ||||||
|  | 		BRCMF_E_BCNLOST_MSG, "BCNLOST_MSG"}, { | ||||||
|  | 		BRCMF_E_ROAM_PREP, "ROAM_PREP"}, { | ||||||
|  | 		BRCMF_E_PFN_NET_FOUND, "PNO_NET_FOUND"}, { | ||||||
|  | 		BRCMF_E_PFN_NET_LOST, "PNO_NET_LOST"}, { | ||||||
|  | 		BRCMF_E_RESET_COMPLETE, "RESET_COMPLETE"}, { | ||||||
|  | 		BRCMF_E_JOIN_START, "JOIN_START"}, { | ||||||
|  | 		BRCMF_E_ROAM_START, "ROAM_START"}, { | ||||||
|  | 		BRCMF_E_ASSOC_START, "ASSOC_START"}, { | ||||||
|  | 		BRCMF_E_IBSS_ASSOC, "IBSS_ASSOC"}, { | ||||||
|  | 		BRCMF_E_RADIO, "RADIO"}, { | ||||||
|  | 		BRCMF_E_PSM_WATCHDOG, "PSM_WATCHDOG"}, { | ||||||
|  | 		BRCMF_E_PROBREQ_MSG, "PROBREQ_MSG"}, { | ||||||
|  | 		BRCMF_E_SCAN_CONFIRM_IND, "SCAN_CONFIRM_IND"}, { | ||||||
|  | 		BRCMF_E_PSK_SUP, "PSK_SUP"}, { | ||||||
|  | 		BRCMF_E_COUNTRY_CODE_CHANGED, "COUNTRY_CODE_CHANGED"}, { | ||||||
|  | 		BRCMF_E_EXCEEDED_MEDIUM_TIME, "EXCEEDED_MEDIUM_TIME"}, { | ||||||
|  | 		BRCMF_E_ICV_ERROR, "ICV_ERROR"}, { | ||||||
|  | 		BRCMF_E_UNICAST_DECODE_ERROR, "UNICAST_DECODE_ERROR"}, { | ||||||
|  | 		BRCMF_E_MULTICAST_DECODE_ERROR, "MULTICAST_DECODE_ERROR"}, { | ||||||
|  | 		BRCMF_E_TRACE, "TRACE"}, { | ||||||
|  | 		BRCMF_E_ACTION_FRAME, "ACTION FRAME"}, { | ||||||
|  | 		BRCMF_E_ACTION_FRAME_COMPLETE, "ACTION FRAME TX COMPLETE"}, { | ||||||
|  | 		BRCMF_E_IF, "IF"}, { | ||||||
|  | 		BRCMF_E_RSSI, "RSSI"}, { | ||||||
|  | 		BRCMF_E_PFN_SCAN_COMPLETE, "SCAN_COMPLETE"} | ||||||
|  | 	}; | ||||||
|  | 	uint event_type, flags, auth_type, datalen; | ||||||
|  | 	static u32 seqnum_prev; | ||||||
|  | 	struct msgtrace_hdr hdr; | ||||||
|  | 	u32 nblost; | ||||||
|  | 	char *s, *p; | ||||||
|  | 
 | ||||||
|  | 	event_type = be32_to_cpu(event->event_type); | ||||||
|  | 	flags = be16_to_cpu(event->flags); | ||||||
|  | 	status = be32_to_cpu(event->status); | ||||||
|  | 	reason = be32_to_cpu(event->reason); | ||||||
|  | 	auth_type = be32_to_cpu(event->auth_type); | ||||||
|  | 	datalen = be32_to_cpu(event->datalen); | ||||||
|  | 	/* debug dump of event messages */ | ||||||
|  | 	sprintf(eabuf, "%pM", event->addr); | ||||||
|  | 
 | ||||||
|  | 	event_name = "UNKNOWN"; | ||||||
|  | 	for (i = 0; i < ARRAY_SIZE(event_names); i++) { | ||||||
|  | 		if (event_names[i].event == event_type) | ||||||
|  | 			event_name = event_names[i].event_name; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	brcmf_dbg(EVENT, "EVENT: %s, event ID = %d\n", event_name, event_type); | ||||||
|  | 	brcmf_dbg(EVENT, "flags 0x%04x, status %d, reason %d, auth_type %d MAC %s\n", | ||||||
|  | 		  flags, status, reason, auth_type, eabuf); | ||||||
|  | 
 | ||||||
|  | 	if (flags & BRCMF_EVENT_MSG_LINK) | ||||||
|  | 		link = true; | ||||||
|  | 	if (flags & BRCMF_EVENT_MSG_GROUP) | ||||||
|  | 		group = true; | ||||||
|  | 	if (flags & BRCMF_EVENT_MSG_FLUSHTXQ) | ||||||
|  | 		flush_txq = true; | ||||||
|  | 
 | ||||||
|  | 	switch (event_type) { | ||||||
|  | 	case BRCMF_E_START: | ||||||
|  | 	case BRCMF_E_DEAUTH: | ||||||
|  | 	case BRCMF_E_DISASSOC: | ||||||
|  | 		brcmf_dbg(EVENT, "MACEVENT: %s, MAC %s\n", event_name, eabuf); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case BRCMF_E_ASSOC_IND: | ||||||
|  | 	case BRCMF_E_REASSOC_IND: | ||||||
|  | 		brcmf_dbg(EVENT, "MACEVENT: %s, MAC %s\n", event_name, eabuf); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case BRCMF_E_ASSOC: | ||||||
|  | 	case BRCMF_E_REASSOC: | ||||||
|  | 		if (status == BRCMF_E_STATUS_SUCCESS) | ||||||
|  | 			brcmf_dbg(EVENT, "MACEVENT: %s, MAC %s, SUCCESS\n", | ||||||
|  | 				  event_name, eabuf); | ||||||
|  | 		else if (status == BRCMF_E_STATUS_TIMEOUT) | ||||||
|  | 			brcmf_dbg(EVENT, "MACEVENT: %s, MAC %s, TIMEOUT\n", | ||||||
|  | 				  event_name, eabuf); | ||||||
|  | 		else if (status == BRCMF_E_STATUS_FAIL) | ||||||
|  | 			brcmf_dbg(EVENT, "MACEVENT: %s, MAC %s, FAILURE, reason %d\n", | ||||||
|  | 				  event_name, eabuf, (int)reason); | ||||||
|  | 		else | ||||||
|  | 			brcmf_dbg(EVENT, "MACEVENT: %s, MAC %s, unexpected status %d\n", | ||||||
|  | 				  event_name, eabuf, (int)status); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case BRCMF_E_DEAUTH_IND: | ||||||
|  | 	case BRCMF_E_DISASSOC_IND: | ||||||
|  | 		brcmf_dbg(EVENT, "MACEVENT: %s, MAC %s, reason %d\n", | ||||||
|  | 			  event_name, eabuf, (int)reason); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case BRCMF_E_AUTH: | ||||||
|  | 	case BRCMF_E_AUTH_IND: | ||||||
|  | 		if (auth_type == WLAN_AUTH_OPEN) | ||||||
|  | 			auth_str = "Open System"; | ||||||
|  | 		else if (auth_type == WLAN_AUTH_SHARED_KEY) | ||||||
|  | 			auth_str = "Shared Key"; | ||||||
|  | 		else { | ||||||
|  | 			sprintf(err_msg, "AUTH unknown: %d", (int)auth_type); | ||||||
|  | 			auth_str = err_msg; | ||||||
|  | 		} | ||||||
|  | 		if (event_type == BRCMF_E_AUTH_IND) | ||||||
|  | 			brcmf_dbg(EVENT, "MACEVENT: %s, MAC %s, %s\n", | ||||||
|  | 				  event_name, eabuf, auth_str); | ||||||
|  | 		else if (status == BRCMF_E_STATUS_SUCCESS) | ||||||
|  | 			brcmf_dbg(EVENT, "MACEVENT: %s, MAC %s, %s, SUCCESS\n", | ||||||
|  | 				  event_name, eabuf, auth_str); | ||||||
|  | 		else if (status == BRCMF_E_STATUS_TIMEOUT) | ||||||
|  | 			brcmf_dbg(EVENT, "MACEVENT: %s, MAC %s, %s, TIMEOUT\n", | ||||||
|  | 				  event_name, eabuf, auth_str); | ||||||
|  | 		else if (status == BRCMF_E_STATUS_FAIL) { | ||||||
|  | 			brcmf_dbg(EVENT, "MACEVENT: %s, MAC %s, %s, FAILURE, reason %d\n", | ||||||
|  | 				  event_name, eabuf, auth_str, (int)reason); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case BRCMF_E_JOIN: | ||||||
|  | 	case BRCMF_E_ROAM: | ||||||
|  | 	case BRCMF_E_SET_SSID: | ||||||
|  | 		if (status == BRCMF_E_STATUS_SUCCESS) | ||||||
|  | 			brcmf_dbg(EVENT, "MACEVENT: %s, MAC %s\n", | ||||||
|  | 				  event_name, eabuf); | ||||||
|  | 		else if (status == BRCMF_E_STATUS_FAIL) | ||||||
|  | 			brcmf_dbg(EVENT, "MACEVENT: %s, failed\n", event_name); | ||||||
|  | 		else if (status == BRCMF_E_STATUS_NO_NETWORKS) | ||||||
|  | 			brcmf_dbg(EVENT, "MACEVENT: %s, no networks found\n", | ||||||
|  | 				  event_name); | ||||||
|  | 		else | ||||||
|  | 			brcmf_dbg(EVENT, "MACEVENT: %s, unexpected status %d\n", | ||||||
|  | 				  event_name, (int)status); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case BRCMF_E_BEACON_RX: | ||||||
|  | 		if (status == BRCMF_E_STATUS_SUCCESS) | ||||||
|  | 			brcmf_dbg(EVENT, "MACEVENT: %s, SUCCESS\n", event_name); | ||||||
|  | 		else if (status == BRCMF_E_STATUS_FAIL) | ||||||
|  | 			brcmf_dbg(EVENT, "MACEVENT: %s, FAIL\n", event_name); | ||||||
|  | 		else | ||||||
|  | 			brcmf_dbg(EVENT, "MACEVENT: %s, status %d\n", | ||||||
|  | 				  event_name, status); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case BRCMF_E_LINK: | ||||||
|  | 		brcmf_dbg(EVENT, "MACEVENT: %s %s\n", | ||||||
|  | 			  event_name, link ? "UP" : "DOWN"); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case BRCMF_E_MIC_ERROR: | ||||||
|  | 		brcmf_dbg(EVENT, "MACEVENT: %s, MAC %s, Group %d, Flush %d\n", | ||||||
|  | 			  event_name, eabuf, group, flush_txq); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case BRCMF_E_ICV_ERROR: | ||||||
|  | 	case BRCMF_E_UNICAST_DECODE_ERROR: | ||||||
|  | 	case BRCMF_E_MULTICAST_DECODE_ERROR: | ||||||
|  | 		brcmf_dbg(EVENT, "MACEVENT: %s, MAC %s\n", event_name, eabuf); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case BRCMF_E_TXFAIL: | ||||||
|  | 		brcmf_dbg(EVENT, "MACEVENT: %s, RA %s\n", event_name, eabuf); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case BRCMF_E_SCAN_COMPLETE: | ||||||
|  | 	case BRCMF_E_PMKID_CACHE: | ||||||
|  | 		brcmf_dbg(EVENT, "MACEVENT: %s\n", event_name); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case BRCMF_E_PFN_NET_FOUND: | ||||||
|  | 	case BRCMF_E_PFN_NET_LOST: | ||||||
|  | 	case BRCMF_E_PFN_SCAN_COMPLETE: | ||||||
|  | 		brcmf_dbg(EVENT, "PNOEVENT: %s\n", event_name); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case BRCMF_E_PSK_SUP: | ||||||
|  | 	case BRCMF_E_PRUNE: | ||||||
|  | 		brcmf_dbg(EVENT, "MACEVENT: %s, status %d, reason %d\n", | ||||||
|  | 			  event_name, (int)status, (int)reason); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case BRCMF_E_TRACE: | ||||||
|  | 		buf = (unsigned char *) event_data; | ||||||
|  | 		memcpy(&hdr, buf, sizeof(struct msgtrace_hdr)); | ||||||
|  | 
 | ||||||
|  | 		if (hdr.version != MSGTRACE_VERSION) { | ||||||
|  | 			brcmf_dbg(ERROR, | ||||||
|  | 				  "MACEVENT: %s [unsupported version --> brcmf" | ||||||
|  | 				  " version:%d dongle version:%d]\n", | ||||||
|  | 				  event_name, MSGTRACE_VERSION, hdr.version); | ||||||
|  | 			/* Reset datalen to avoid display below */ | ||||||
|  | 			datalen = 0; | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		/* There are 2 bytes available at the end of data */ | ||||||
|  | 		*(buf + sizeof(struct msgtrace_hdr) | ||||||
|  | 			 + be16_to_cpu(hdr.len)) = '\0'; | ||||||
|  | 
 | ||||||
|  | 		if (be32_to_cpu(hdr.discarded_bytes) | ||||||
|  | 		    || be32_to_cpu(hdr.discarded_printf)) | ||||||
|  | 			brcmf_dbg(ERROR, | ||||||
|  | 				  "WLC_E_TRACE: [Discarded traces in dongle -->" | ||||||
|  | 				  " discarded_bytes %d discarded_printf %d]\n", | ||||||
|  | 				  be32_to_cpu(hdr.discarded_bytes), | ||||||
|  | 				  be32_to_cpu(hdr.discarded_printf)); | ||||||
|  | 
 | ||||||
|  | 		nblost = be32_to_cpu(hdr.seqnum) - seqnum_prev - 1; | ||||||
|  | 		if (nblost > 0) | ||||||
|  | 			brcmf_dbg(ERROR, "WLC_E_TRACE: [Event lost --> seqnum " | ||||||
|  | 				  " %d nblost %d\n", be32_to_cpu(hdr.seqnum), | ||||||
|  | 				  nblost); | ||||||
|  | 		seqnum_prev = be32_to_cpu(hdr.seqnum); | ||||||
|  | 
 | ||||||
|  | 		/* Display the trace buffer. Advance from \n to \n to
 | ||||||
|  | 		 * avoid display big | ||||||
|  | 		 * printf (issue with Linux printk ) | ||||||
|  | 		 */ | ||||||
|  | 		p = (char *)&buf[sizeof(struct msgtrace_hdr)]; | ||||||
|  | 		while ((s = strstr(p, "\n")) != NULL) { | ||||||
|  | 			*s = '\0'; | ||||||
|  | 			printk(KERN_DEBUG"%s\n", p); | ||||||
|  | 			p = s + 1; | ||||||
|  | 		} | ||||||
|  | 		printk(KERN_DEBUG "%s\n", p); | ||||||
|  | 
 | ||||||
|  | 		/* Reset datalen to avoid display below */ | ||||||
|  | 		datalen = 0; | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case BRCMF_E_RSSI: | ||||||
|  | 		brcmf_dbg(EVENT, "MACEVENT: %s %d\n", | ||||||
|  | 			  event_name, be32_to_cpu(*((__be32 *)event_data))); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	default: | ||||||
|  | 		brcmf_dbg(EVENT, | ||||||
|  | 			  "MACEVENT: %s %d, MAC %s, status %d, reason %d, " | ||||||
|  | 			  "auth %d\n", event_name, event_type, eabuf, | ||||||
|  | 			  (int)status, (int)reason, (int)auth_type); | ||||||
|  | 		break; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* show any appended data */ | ||||||
|  | 	if (datalen) { | ||||||
|  | 		buf = (unsigned char *) event_data; | ||||||
|  | 		brcmf_dbg(EVENT, " data (%d) : ", datalen); | ||||||
|  | 		for (i = 0; i < datalen; i++) | ||||||
|  | 			brcmf_dbg(EVENT, " 0x%02x ", *buf++); | ||||||
|  | 		brcmf_dbg(EVENT, "\n"); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | #endif				/* BCMDBG */ | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | brcmf_c_host_event(struct brcmf_info *drvr_priv, int *ifidx, void *pktdata, | ||||||
|  | 		   struct brcmf_event_msg *event, void **data_ptr) | ||||||
|  | { | ||||||
|  | 	/* check whether packet is a BRCM event pkt */ | ||||||
|  | 	struct brcmf_event *pvt_data = (struct brcmf_event *) pktdata; | ||||||
|  | 	struct brcmf_if_event *ifevent; | ||||||
|  | 	char *event_data; | ||||||
|  | 	u32 type, status; | ||||||
|  | 	u16 flags; | ||||||
|  | 	int evlen; | ||||||
|  | 
 | ||||||
|  | 	if (memcmp(BRCM_OUI, &pvt_data->hdr.oui[0], DOT11_OUI_LEN)) { | ||||||
|  | 		brcmf_dbg(ERROR, "mismatched OUI, bailing\n"); | ||||||
|  | 		return -EBADE; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* BRCM event pkt may be unaligned - use xxx_ua to load user_subtype. */ | ||||||
|  | 	if (get_unaligned_be16(&pvt_data->hdr.usr_subtype) != | ||||||
|  | 	    BCMILCP_BCM_SUBTYPE_EVENT) { | ||||||
|  | 		brcmf_dbg(ERROR, "mismatched subtype, bailing\n"); | ||||||
|  | 		return -EBADE; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	*data_ptr = &pvt_data[1]; | ||||||
|  | 	event_data = *data_ptr; | ||||||
|  | 
 | ||||||
|  | 	/* memcpy since BRCM event pkt may be unaligned. */ | ||||||
|  | 	memcpy(event, &pvt_data->msg, sizeof(struct brcmf_event_msg)); | ||||||
|  | 
 | ||||||
|  | 	type = get_unaligned_be32(&event->event_type); | ||||||
|  | 	flags = get_unaligned_be16(&event->flags); | ||||||
|  | 	status = get_unaligned_be32(&event->status); | ||||||
|  | 	evlen = get_unaligned_be32(&event->datalen) + | ||||||
|  | 		sizeof(struct brcmf_event); | ||||||
|  | 
 | ||||||
|  | 	switch (type) { | ||||||
|  | 	case BRCMF_E_IF: | ||||||
|  | 		ifevent = (struct brcmf_if_event *) event_data; | ||||||
|  | 		brcmf_dbg(TRACE, "if event\n"); | ||||||
|  | 
 | ||||||
|  | 		if (ifevent->ifidx > 0 && ifevent->ifidx < BRCMF_MAX_IFS) { | ||||||
|  | 			if (ifevent->action == BRCMF_E_IF_ADD) | ||||||
|  | 				brcmf_add_if(drvr_priv, ifevent->ifidx, NULL, | ||||||
|  | 					     event->ifname, | ||||||
|  | 					     pvt_data->eth.h_dest, | ||||||
|  | 					     ifevent->flags, ifevent->bssidx); | ||||||
|  | 			else | ||||||
|  | 				brcmf_del_if(drvr_priv, ifevent->ifidx); | ||||||
|  | 		} else { | ||||||
|  | 			brcmf_dbg(ERROR, "Invalid ifidx %d for %s\n", | ||||||
|  | 				  ifevent->ifidx, event->ifname); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		/* send up the if event: btamp user needs it */ | ||||||
|  | 		*ifidx = brcmf_ifname2idx(drvr_priv, event->ifname); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 		/* These are what external supplicant/authenticator wants */ | ||||||
|  | 	case BRCMF_E_LINK: | ||||||
|  | 	case BRCMF_E_ASSOC_IND: | ||||||
|  | 	case BRCMF_E_REASSOC_IND: | ||||||
|  | 	case BRCMF_E_DISASSOC_IND: | ||||||
|  | 	case BRCMF_E_MIC_ERROR: | ||||||
|  | 	default: | ||||||
|  | 		/* Fall through: this should get _everything_  */ | ||||||
|  | 
 | ||||||
|  | 		*ifidx = brcmf_ifname2idx(drvr_priv, event->ifname); | ||||||
|  | 		brcmf_dbg(TRACE, "MAC event %d, flags %x, status %x\n", | ||||||
|  | 			  type, flags, status); | ||||||
|  | 
 | ||||||
|  | 		/* put it back to BRCMF_E_NDIS_LINK */ | ||||||
|  | 		if (type == BRCMF_E_NDIS_LINK) { | ||||||
|  | 			u32 temp1; | ||||||
|  | 			__be32 temp2; | ||||||
|  | 
 | ||||||
|  | 			temp1 = get_unaligned_be32(&event->event_type); | ||||||
|  | 			brcmf_dbg(TRACE, "Converted to WLC_E_LINK type %d\n", | ||||||
|  | 				  temp1); | ||||||
|  | 
 | ||||||
|  | 			temp2 = cpu_to_be32(BRCMF_E_NDIS_LINK); | ||||||
|  | 			memcpy((void *)(&pvt_data->msg.event_type), &temp2, | ||||||
|  | 			       sizeof(pvt_data->msg.event_type)); | ||||||
|  | 		} | ||||||
|  | 		break; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | #ifdef BCMDBG | ||||||
|  | 	brcmf_c_show_host_event(event, event_data); | ||||||
|  | #endif				/* BCMDBG */ | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Convert user's input in hex pattern to byte-size mask */ | ||||||
|  | static int brcmf_c_pattern_atoh(char *src, char *dst) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	if (strncmp(src, "0x", 2) != 0 && strncmp(src, "0X", 2) != 0) { | ||||||
|  | 		brcmf_dbg(ERROR, "Mask invalid format. Needs to start with 0x\n"); | ||||||
|  | 		return -EINVAL; | ||||||
|  | 	} | ||||||
|  | 	src = src + 2;		/* Skip past 0x */ | ||||||
|  | 	if (strlen(src) % 2 != 0) { | ||||||
|  | 		brcmf_dbg(ERROR, "Mask invalid format. Length must be even.\n"); | ||||||
|  | 		return -EINVAL; | ||||||
|  | 	} | ||||||
|  | 	for (i = 0; *src != '\0'; i++) { | ||||||
|  | 		unsigned long res; | ||||||
|  | 		char num[3]; | ||||||
|  | 		strncpy(num, src, 2); | ||||||
|  | 		num[2] = '\0'; | ||||||
|  | 		if (kstrtoul(num, 16, &res)) | ||||||
|  | 			return -EINVAL; | ||||||
|  | 		dst[i] = (u8)res; | ||||||
|  | 		src += 2; | ||||||
|  | 	} | ||||||
|  | 	return i; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | brcmf_c_pktfilter_offload_enable(struct brcmf_pub *drvr, char *arg, int enable, | ||||||
|  | 			     int master_mode) | ||||||
|  | { | ||||||
|  | 	unsigned long res; | ||||||
|  | 	char *argv[8]; | ||||||
|  | 	int i = 0; | ||||||
|  | 	const char *str; | ||||||
|  | 	int buf_len; | ||||||
|  | 	int str_len; | ||||||
|  | 	char *arg_save = NULL, *arg_org = NULL; | ||||||
|  | 	int rc; | ||||||
|  | 	char buf[128]; | ||||||
|  | 	struct brcmf_pkt_filter_enable enable_parm; | ||||||
|  | 	struct brcmf_pkt_filter_enable *pkt_filterp; | ||||||
|  | 
 | ||||||
|  | 	arg_save = kmalloc(strlen(arg) + 1, GFP_ATOMIC); | ||||||
|  | 	if (!arg_save) | ||||||
|  | 		goto fail; | ||||||
|  | 
 | ||||||
|  | 	arg_org = arg_save; | ||||||
|  | 	memcpy(arg_save, arg, strlen(arg) + 1); | ||||||
|  | 
 | ||||||
|  | 	argv[i] = strsep(&arg_save, " "); | ||||||
|  | 
 | ||||||
|  | 	i = 0; | ||||||
|  | 	if (NULL == argv[i]) { | ||||||
|  | 		brcmf_dbg(ERROR, "No args provided\n"); | ||||||
|  | 		goto fail; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	str = "pkt_filter_enable"; | ||||||
|  | 	str_len = strlen(str); | ||||||
|  | 	strncpy(buf, str, str_len); | ||||||
|  | 	buf[str_len] = '\0'; | ||||||
|  | 	buf_len = str_len + 1; | ||||||
|  | 
 | ||||||
|  | 	pkt_filterp = (struct brcmf_pkt_filter_enable *) (buf + str_len + 1); | ||||||
|  | 
 | ||||||
|  | 	/* Parse packet filter id. */ | ||||||
|  | 	enable_parm.id = 0; | ||||||
|  | 	if (!kstrtoul(argv[i], 0, &res)) | ||||||
|  | 		enable_parm.id = (u32)res; | ||||||
|  | 
 | ||||||
|  | 	/* Parse enable/disable value. */ | ||||||
|  | 	enable_parm.enable = enable; | ||||||
|  | 
 | ||||||
|  | 	buf_len += sizeof(enable_parm); | ||||||
|  | 	memcpy((char *)pkt_filterp, &enable_parm, sizeof(enable_parm)); | ||||||
|  | 
 | ||||||
|  | 	/* Enable/disable the specified filter. */ | ||||||
|  | 	rc = brcmf_proto_cdc_set_dcmd(drvr, 0, BRCMF_C_SET_VAR, buf, buf_len); | ||||||
|  | 	rc = rc >= 0 ? 0 : rc; | ||||||
|  | 	if (rc) | ||||||
|  | 		brcmf_dbg(TRACE, "failed to add pktfilter %s, retcode = %d\n", | ||||||
|  | 			  arg, rc); | ||||||
|  | 	else | ||||||
|  | 		brcmf_dbg(TRACE, "successfully added pktfilter %s\n", arg); | ||||||
|  | 
 | ||||||
|  | 	/* Contorl the master mode */ | ||||||
|  | 	brcmu_mkiovar("pkt_filter_mode", (char *)&master_mode, 4, buf, | ||||||
|  | 		    sizeof(buf)); | ||||||
|  | 	rc = brcmf_proto_cdc_set_dcmd(drvr, 0, BRCMF_C_SET_VAR, buf, | ||||||
|  | 				       sizeof(buf)); | ||||||
|  | 	rc = rc >= 0 ? 0 : rc; | ||||||
|  | 	if (rc) | ||||||
|  | 		brcmf_dbg(TRACE, "failed to add pktfilter %s, retcode = %d\n", | ||||||
|  | 			  arg, rc); | ||||||
|  | 
 | ||||||
|  | fail: | ||||||
|  | 	kfree(arg_org); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void brcmf_c_pktfilter_offload_set(struct brcmf_pub *drvr, char *arg) | ||||||
|  | { | ||||||
|  | 	const char *str; | ||||||
|  | 	struct brcmf_pkt_filter pkt_filter; | ||||||
|  | 	struct brcmf_pkt_filter *pkt_filterp; | ||||||
|  | 	unsigned long res; | ||||||
|  | 	int buf_len; | ||||||
|  | 	int str_len; | ||||||
|  | 	int rc; | ||||||
|  | 	u32 mask_size; | ||||||
|  | 	u32 pattern_size; | ||||||
|  | 	char *argv[8], *buf = NULL; | ||||||
|  | 	int i = 0; | ||||||
|  | 	char *arg_save = NULL, *arg_org = NULL; | ||||||
|  | 
 | ||||||
|  | 	arg_save = kstrdup(arg, GFP_ATOMIC); | ||||||
|  | 	if (!arg_save) | ||||||
|  | 		goto fail; | ||||||
|  | 
 | ||||||
|  | 	arg_org = arg_save; | ||||||
|  | 
 | ||||||
|  | 	buf = kmalloc(PKTFILTER_BUF_SIZE, GFP_ATOMIC); | ||||||
|  | 	if (!buf) | ||||||
|  | 		goto fail; | ||||||
|  | 
 | ||||||
|  | 	argv[i] = strsep(&arg_save, " "); | ||||||
|  | 	while (argv[i++]) | ||||||
|  | 		argv[i] = strsep(&arg_save, " "); | ||||||
|  | 
 | ||||||
|  | 	i = 0; | ||||||
|  | 	if (NULL == argv[i]) { | ||||||
|  | 		brcmf_dbg(ERROR, "No args provided\n"); | ||||||
|  | 		goto fail; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	str = "pkt_filter_add"; | ||||||
|  | 	strcpy(buf, str); | ||||||
|  | 	str_len = strlen(str); | ||||||
|  | 	buf_len = str_len + 1; | ||||||
|  | 
 | ||||||
|  | 	pkt_filterp = (struct brcmf_pkt_filter *) (buf + str_len + 1); | ||||||
|  | 
 | ||||||
|  | 	/* Parse packet filter id. */ | ||||||
|  | 	pkt_filter.id = 0; | ||||||
|  | 	if (!kstrtoul(argv[i], 0, &res)) | ||||||
|  | 		pkt_filter.id = (u32)res; | ||||||
|  | 
 | ||||||
|  | 	if (NULL == argv[++i]) { | ||||||
|  | 		brcmf_dbg(ERROR, "Polarity not provided\n"); | ||||||
|  | 		goto fail; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Parse filter polarity. */ | ||||||
|  | 	pkt_filter.negate_match = 0; | ||||||
|  | 	if (!kstrtoul(argv[i], 0, &res)) | ||||||
|  | 		pkt_filter.negate_match = (u32)res; | ||||||
|  | 
 | ||||||
|  | 	if (NULL == argv[++i]) { | ||||||
|  | 		brcmf_dbg(ERROR, "Filter type not provided\n"); | ||||||
|  | 		goto fail; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Parse filter type. */ | ||||||
|  | 	pkt_filter.type = 0; | ||||||
|  | 	if (!kstrtoul(argv[i], 0, &res)) | ||||||
|  | 		pkt_filter.type = (u32)res; | ||||||
|  | 
 | ||||||
|  | 	if (NULL == argv[++i]) { | ||||||
|  | 		brcmf_dbg(ERROR, "Offset not provided\n"); | ||||||
|  | 		goto fail; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Parse pattern filter offset. */ | ||||||
|  | 	pkt_filter.u.pattern.offset = 0; | ||||||
|  | 	if (!kstrtoul(argv[i], 0, &res)) | ||||||
|  | 		pkt_filter.u.pattern.offset = (u32)res; | ||||||
|  | 
 | ||||||
|  | 	if (NULL == argv[++i]) { | ||||||
|  | 		brcmf_dbg(ERROR, "Bitmask not provided\n"); | ||||||
|  | 		goto fail; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Parse pattern filter mask. */ | ||||||
|  | 	mask_size = | ||||||
|  | 	    brcmf_c_pattern_atoh | ||||||
|  | 		   (argv[i], (char *)pkt_filterp->u.pattern.mask_and_pattern); | ||||||
|  | 
 | ||||||
|  | 	if (NULL == argv[++i]) { | ||||||
|  | 		brcmf_dbg(ERROR, "Pattern not provided\n"); | ||||||
|  | 		goto fail; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Parse pattern filter pattern. */ | ||||||
|  | 	pattern_size = | ||||||
|  | 	    brcmf_c_pattern_atoh(argv[i], | ||||||
|  | 				   (char *)&pkt_filterp->u.pattern. | ||||||
|  | 				   mask_and_pattern[mask_size]); | ||||||
|  | 
 | ||||||
|  | 	if (mask_size != pattern_size) { | ||||||
|  | 		brcmf_dbg(ERROR, "Mask and pattern not the same size\n"); | ||||||
|  | 		goto fail; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	pkt_filter.u.pattern.size_bytes = mask_size; | ||||||
|  | 	buf_len += BRCMF_PKT_FILTER_FIXED_LEN; | ||||||
|  | 	buf_len += (BRCMF_PKT_FILTER_PATTERN_FIXED_LEN + 2 * mask_size); | ||||||
|  | 
 | ||||||
|  | 	/* Keep-alive attributes are set in local
 | ||||||
|  | 	 * variable (keep_alive_pkt), and | ||||||
|  | 	 ** then memcpy'ed into buffer (keep_alive_pktp) since there is no | ||||||
|  | 	 ** guarantee that the buffer is properly aligned. | ||||||
|  | 	 */ | ||||||
|  | 	memcpy((char *)pkt_filterp, | ||||||
|  | 	       &pkt_filter, | ||||||
|  | 	       BRCMF_PKT_FILTER_FIXED_LEN + BRCMF_PKT_FILTER_PATTERN_FIXED_LEN); | ||||||
|  | 
 | ||||||
|  | 	rc = brcmf_proto_cdc_set_dcmd(drvr, 0, BRCMF_C_SET_VAR, buf, buf_len); | ||||||
|  | 	rc = rc >= 0 ? 0 : rc; | ||||||
|  | 
 | ||||||
|  | 	if (rc) | ||||||
|  | 		brcmf_dbg(TRACE, "failed to add pktfilter %s, retcode = %d\n", | ||||||
|  | 			  arg, rc); | ||||||
|  | 	else | ||||||
|  | 		brcmf_dbg(TRACE, "successfully added pktfilter %s\n", arg); | ||||||
|  | 
 | ||||||
|  | fail: | ||||||
|  | 	kfree(arg_org); | ||||||
|  | 
 | ||||||
|  | 	kfree(buf); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void brcmf_c_arp_offload_set(struct brcmf_pub *drvr, int arp_mode) | ||||||
|  | { | ||||||
|  | 	char iovbuf[32]; | ||||||
|  | 	int retcode; | ||||||
|  | 
 | ||||||
|  | 	brcmu_mkiovar("arp_ol", (char *)&arp_mode, 4, iovbuf, sizeof(iovbuf)); | ||||||
|  | 	retcode = brcmf_proto_cdc_set_dcmd(drvr, 0, BRCMF_C_SET_VAR, | ||||||
|  | 				   iovbuf, sizeof(iovbuf)); | ||||||
|  | 	retcode = retcode >= 0 ? 0 : retcode; | ||||||
|  | 	if (retcode) | ||||||
|  | 		brcmf_dbg(TRACE, "failed to set ARP offload mode to 0x%x, retcode = %d\n", | ||||||
|  | 			  arp_mode, retcode); | ||||||
|  | 	else | ||||||
|  | 		brcmf_dbg(TRACE, "successfully set ARP offload mode to 0x%x\n", | ||||||
|  | 			  arp_mode); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void brcmf_c_arp_offload_enable(struct brcmf_pub *drvr, int arp_enable) | ||||||
|  | { | ||||||
|  | 	char iovbuf[32]; | ||||||
|  | 	int retcode; | ||||||
|  | 
 | ||||||
|  | 	brcmu_mkiovar("arpoe", (char *)&arp_enable, 4, iovbuf, sizeof(iovbuf)); | ||||||
|  | 	retcode = brcmf_proto_cdc_set_dcmd(drvr, 0, BRCMF_C_SET_VAR, | ||||||
|  | 				   iovbuf, sizeof(iovbuf)); | ||||||
|  | 	retcode = retcode >= 0 ? 0 : retcode; | ||||||
|  | 	if (retcode) | ||||||
|  | 		brcmf_dbg(TRACE, "failed to enable ARP offload to %d, retcode = %d\n", | ||||||
|  | 			  arp_enable, retcode); | ||||||
|  | 	else | ||||||
|  | 		brcmf_dbg(TRACE, "successfully enabled ARP offload to %d\n", | ||||||
|  | 			  arp_enable); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int brcmf_c_preinit_dcmds(struct brcmf_pub *drvr) | ||||||
|  | { | ||||||
|  | 	char iovbuf[BRCMF_EVENTING_MASK_LEN + 12];	/*  Room for
 | ||||||
|  | 				 "event_msgs" + '\0' + bitvec  */ | ||||||
|  | 	uint up = 0; | ||||||
|  | 	char buf[128], *ptr; | ||||||
|  | 	u32 dongle_align = BRCMF_SDALIGN; | ||||||
|  | 	u32 glom = 0; | ||||||
|  | 	u32 roaming = 1; | ||||||
|  | 	uint bcn_timeout = 3; | ||||||
|  | 	int scan_assoc_time = 40; | ||||||
|  | 	int scan_unassoc_time = 40; | ||||||
|  | 	int i; | ||||||
|  | 
 | ||||||
|  | 	brcmf_os_proto_block(drvr); | ||||||
|  | 
 | ||||||
|  | 	/* Set Country code */ | ||||||
|  | 	if (drvr->country_code[0] != 0) { | ||||||
|  | 		if (brcmf_proto_cdc_set_dcmd(drvr, 0, BRCMF_C_SET_COUNTRY, | ||||||
|  | 					      drvr->country_code, | ||||||
|  | 					      sizeof(drvr->country_code)) < 0) | ||||||
|  | 			brcmf_dbg(ERROR, "country code setting failed\n"); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* query for 'ver' to get version info from firmware */ | ||||||
|  | 	memset(buf, 0, sizeof(buf)); | ||||||
|  | 	ptr = buf; | ||||||
|  | 	brcmu_mkiovar("ver", NULL, 0, buf, sizeof(buf)); | ||||||
|  | 	brcmf_proto_cdc_query_dcmd(drvr, 0, BRCMF_C_GET_VAR, buf, sizeof(buf)); | ||||||
|  | 	strsep(&ptr, "\n"); | ||||||
|  | 	/* Print fw version info */ | ||||||
|  | 	brcmf_dbg(ERROR, "Firmware version = %s\n", buf); | ||||||
|  | 
 | ||||||
|  | 	/* Match Host and Dongle rx alignment */ | ||||||
|  | 	brcmu_mkiovar("bus:txglomalign", (char *)&dongle_align, 4, iovbuf, | ||||||
|  | 		    sizeof(iovbuf)); | ||||||
|  | 	brcmf_proto_cdc_set_dcmd(drvr, 0, BRCMF_C_SET_VAR, iovbuf, | ||||||
|  | 				  sizeof(iovbuf)); | ||||||
|  | 
 | ||||||
|  | 	/* disable glom option per default */ | ||||||
|  | 	brcmu_mkiovar("bus:txglom", (char *)&glom, 4, iovbuf, sizeof(iovbuf)); | ||||||
|  | 	brcmf_proto_cdc_set_dcmd(drvr, 0, BRCMF_C_SET_VAR, iovbuf, | ||||||
|  | 				  sizeof(iovbuf)); | ||||||
|  | 
 | ||||||
|  | 	/* Setup timeout if Beacons are lost and roam is off to report
 | ||||||
|  | 		 link down */ | ||||||
|  | 	brcmu_mkiovar("bcn_timeout", (char *)&bcn_timeout, 4, iovbuf, | ||||||
|  | 		    sizeof(iovbuf)); | ||||||
|  | 	brcmf_proto_cdc_set_dcmd(drvr, 0, BRCMF_C_SET_VAR, iovbuf, | ||||||
|  | 				  sizeof(iovbuf)); | ||||||
|  | 
 | ||||||
|  | 	/* Enable/Disable build-in roaming to allowed ext supplicant to take
 | ||||||
|  | 		 of romaing */ | ||||||
|  | 	brcmu_mkiovar("roam_off", (char *)&roaming, 4, | ||||||
|  | 		      iovbuf, sizeof(iovbuf)); | ||||||
|  | 	brcmf_proto_cdc_set_dcmd(drvr, 0, BRCMF_C_SET_VAR, iovbuf, | ||||||
|  | 				  sizeof(iovbuf)); | ||||||
|  | 
 | ||||||
|  | 	/* Force STA UP */ | ||||||
|  | 	brcmf_proto_cdc_set_dcmd(drvr, 0, BRCMF_C_UP, (char *)&up, sizeof(up)); | ||||||
|  | 
 | ||||||
|  | 	/* Setup event_msgs */ | ||||||
|  | 	brcmu_mkiovar("event_msgs", drvr->eventmask, BRCMF_EVENTING_MASK_LEN, | ||||||
|  | 		      iovbuf, sizeof(iovbuf)); | ||||||
|  | 	brcmf_proto_cdc_set_dcmd(drvr, 0, BRCMF_C_SET_VAR, iovbuf, | ||||||
|  | 				  sizeof(iovbuf)); | ||||||
|  | 
 | ||||||
|  | 	brcmf_proto_cdc_set_dcmd(drvr, 0, BRCMF_C_SET_SCAN_CHANNEL_TIME, | ||||||
|  | 			 (char *)&scan_assoc_time, sizeof(scan_assoc_time)); | ||||||
|  | 	brcmf_proto_cdc_set_dcmd(drvr, 0, BRCMF_C_SET_SCAN_UNASSOC_TIME, | ||||||
|  | 			 (char *)&scan_unassoc_time, sizeof(scan_unassoc_time)); | ||||||
|  | 
 | ||||||
|  | 	/* Set and enable ARP offload feature */ | ||||||
|  | 	brcmf_c_arp_offload_set(drvr, BRCMF_ARPOL_MODE); | ||||||
|  | 	brcmf_c_arp_offload_enable(drvr, true); | ||||||
|  | 
 | ||||||
|  | 	/* Set up pkt filter */ | ||||||
|  | 	for (i = 0; i < drvr->pktfilter_count; i++) { | ||||||
|  | 		brcmf_c_pktfilter_offload_set(drvr, drvr->pktfilter[i]); | ||||||
|  | 		brcmf_c_pktfilter_offload_enable(drvr, drvr->pktfilter[i], | ||||||
|  | 						 0, true); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	brcmf_os_proto_unblock(drvr); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
							
								
								
									
										58
									
								
								drivers/net/wireless/brcm80211/brcmfmac/dhd_dbg.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								drivers/net/wireless/brcm80211/brcmfmac/dhd_dbg.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,58 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCMF_DBG_H_ | ||||||
|  | #define _BRCMF_DBG_H_ | ||||||
|  | 
 | ||||||
|  | #if defined(BCMDBG) | ||||||
|  | 
 | ||||||
|  | #define brcmf_dbg(level, fmt, ...)					\ | ||||||
|  | do {									\ | ||||||
|  | 	if (BRCMF_ERROR_VAL == BRCMF_##level##_VAL) {			\ | ||||||
|  | 		if (brcmf_msg_level & BRCMF_##level##_VAL) {		\ | ||||||
|  | 			if (net_ratelimit())				\ | ||||||
|  | 				printk(KERN_DEBUG "%s: " fmt,		\ | ||||||
|  | 				       __func__, ##__VA_ARGS__);	\ | ||||||
|  | 		}							\ | ||||||
|  | 	} else {							\ | ||||||
|  | 		if (brcmf_msg_level & BRCMF_##level##_VAL) {		\ | ||||||
|  | 			printk(KERN_DEBUG "%s: " fmt,			\ | ||||||
|  | 			       __func__, ##__VA_ARGS__);		\ | ||||||
|  | 		}							\ | ||||||
|  | 	}								\ | ||||||
|  | } while (0) | ||||||
|  | 
 | ||||||
|  | #define BRCMF_DATA_ON()		(brcmf_msg_level & BRCMF_DATA_VAL) | ||||||
|  | #define BRCMF_CTL_ON()		(brcmf_msg_level & BRCMF_CTL_VAL) | ||||||
|  | #define BRCMF_HDRS_ON()		(brcmf_msg_level & BRCMF_HDRS_VAL) | ||||||
|  | #define BRCMF_BYTES_ON()	(brcmf_msg_level & BRCMF_BYTES_VAL) | ||||||
|  | #define BRCMF_GLOM_ON()		(brcmf_msg_level & BRCMF_GLOM_VAL) | ||||||
|  | 
 | ||||||
|  | #else	/* (defined BCMDBG) || (defined BCMDBG) */ | ||||||
|  | 
 | ||||||
|  | #define brcmf_dbg(level, fmt, ...) no_printk(fmt, ##__VA_ARGS__) | ||||||
|  | 
 | ||||||
|  | #define BRCMF_DATA_ON()		0 | ||||||
|  | #define BRCMF_CTL_ON()		0 | ||||||
|  | #define BRCMF_HDRS_ON()		0 | ||||||
|  | #define BRCMF_BYTES_ON()	0 | ||||||
|  | #define BRCMF_GLOM_ON()		0 | ||||||
|  | 
 | ||||||
|  | #endif				/* defined(BCMDBG) */ | ||||||
|  | 
 | ||||||
|  | extern int brcmf_msg_level; | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCMF_DBG_H_ */ | ||||||
							
								
								
									
										1354
									
								
								drivers/net/wireless/brcm80211/brcmfmac/dhd_linux.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1354
									
								
								drivers/net/wireless/brcm80211/brcmfmac/dhd_linux.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										60
									
								
								drivers/net/wireless/brcm80211/brcmfmac/dhd_proto.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								drivers/net/wireless/brcm80211/brcmfmac/dhd_proto.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,60 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCMF_PROTO_H_ | ||||||
|  | #define _BRCMF_PROTO_H_ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Exported from the brcmf protocol module (brcmf_cdc) | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* Linkage, sets prot link and updates hdrlen in pub */ | ||||||
|  | extern int brcmf_proto_attach(struct brcmf_pub *drvr); | ||||||
|  | 
 | ||||||
|  | /* Unlink, frees allocated protocol memory (including brcmf_proto) */ | ||||||
|  | extern void brcmf_proto_detach(struct brcmf_pub *drvr); | ||||||
|  | 
 | ||||||
|  | /* Initialize protocol: sync w/dongle state.
 | ||||||
|  |  * Sets dongle media info (iswl, drv_version, mac address). | ||||||
|  |  */ | ||||||
|  | extern int brcmf_proto_init(struct brcmf_pub *drvr); | ||||||
|  | 
 | ||||||
|  | /* Stop protocol: sync w/dongle state. */ | ||||||
|  | extern void brcmf_proto_stop(struct brcmf_pub *drvr); | ||||||
|  | 
 | ||||||
|  | /* Add any protocol-specific data header.
 | ||||||
|  |  * Caller must reserve prot_hdrlen prepend space. | ||||||
|  |  */ | ||||||
|  | extern void brcmf_proto_hdrpush(struct brcmf_pub *, int ifidx, | ||||||
|  | 				struct sk_buff *txp); | ||||||
|  | 
 | ||||||
|  | /* Remove any protocol-specific data header. */ | ||||||
|  | extern int brcmf_proto_hdrpull(struct brcmf_pub *, int *ifidx, | ||||||
|  | 			       struct sk_buff *rxp); | ||||||
|  | 
 | ||||||
|  | /* Use protocol to issue command to dongle */ | ||||||
|  | extern int brcmf_proto_dcmd(struct brcmf_pub *drvr, int ifidx, | ||||||
|  | 				struct brcmf_dcmd *dcmd, int len); | ||||||
|  | 
 | ||||||
|  | /* Update local copy of dongle statistics */ | ||||||
|  | extern void brcmf_proto_dstats(struct brcmf_pub *drvr); | ||||||
|  | 
 | ||||||
|  | extern int brcmf_c_preinit_dcmds(struct brcmf_pub *drvr); | ||||||
|  | 
 | ||||||
|  | extern int brcmf_proto_cdc_set_dcmd(struct brcmf_pub *drvr, int ifidx, | ||||||
|  | 				     uint cmd, void *buf, uint len); | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCMF_PROTO_H_ */ | ||||||
							
								
								
									
										4581
									
								
								drivers/net/wireless/brcm80211/brcmfmac/dhd_sdio.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4581
									
								
								drivers/net/wireless/brcm80211/brcmfmac/dhd_sdio.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										252
									
								
								drivers/net/wireless/brcm80211/brcmfmac/sdio_host.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										252
									
								
								drivers/net/wireless/brcm80211/brcmfmac/sdio_host.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,252 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef	_BRCM_SDH_H_ | ||||||
|  | #define	_BRCM_SDH_H_ | ||||||
|  | 
 | ||||||
|  | #include <linux/skbuff.h> | ||||||
|  | 
 | ||||||
|  | #define SDIO_FUNC_0		0 | ||||||
|  | #define SDIO_FUNC_1		1 | ||||||
|  | #define SDIO_FUNC_2		2 | ||||||
|  | 
 | ||||||
|  | #define SDIOD_FBR_SIZE		0x100 | ||||||
|  | 
 | ||||||
|  | /* io_en */ | ||||||
|  | #define SDIO_FUNC_ENABLE_1	0x02 | ||||||
|  | #define SDIO_FUNC_ENABLE_2	0x04 | ||||||
|  | 
 | ||||||
|  | /* io_rdys */ | ||||||
|  | #define SDIO_FUNC_READY_1	0x02 | ||||||
|  | #define SDIO_FUNC_READY_2	0x04 | ||||||
|  | 
 | ||||||
|  | /* intr_status */ | ||||||
|  | #define INTR_STATUS_FUNC1	0x2 | ||||||
|  | #define INTR_STATUS_FUNC2	0x4 | ||||||
|  | 
 | ||||||
|  | /* Maximum number of I/O funcs */ | ||||||
|  | #define SDIOD_MAX_IOFUNCS	7 | ||||||
|  | 
 | ||||||
|  | /* as of sdiod rev 0, supports 3 functions */ | ||||||
|  | #define SBSDIO_NUM_FUNCTION		3 | ||||||
|  | 
 | ||||||
|  | /* function 1 miscellaneous registers */ | ||||||
|  | 
 | ||||||
|  | /* sprom command and status */ | ||||||
|  | #define SBSDIO_SPROM_CS			0x10000 | ||||||
|  | /* sprom info register */ | ||||||
|  | #define SBSDIO_SPROM_INFO		0x10001 | ||||||
|  | /* sprom indirect access data byte 0 */ | ||||||
|  | #define SBSDIO_SPROM_DATA_LOW		0x10002 | ||||||
|  | /* sprom indirect access data byte 1 */ | ||||||
|  | #define SBSDIO_SPROM_DATA_HIGH		0x10003 | ||||||
|  | /* sprom indirect access addr byte 0 */ | ||||||
|  | #define SBSDIO_SPROM_ADDR_LOW		0x10004 | ||||||
|  | /* sprom indirect access addr byte 0 */ | ||||||
|  | #define SBSDIO_SPROM_ADDR_HIGH		0x10005 | ||||||
|  | /* xtal_pu (gpio) output */ | ||||||
|  | #define SBSDIO_CHIP_CTRL_DATA		0x10006 | ||||||
|  | /* xtal_pu (gpio) enable */ | ||||||
|  | #define SBSDIO_CHIP_CTRL_EN		0x10007 | ||||||
|  | /* rev < 7, watermark for sdio device */ | ||||||
|  | #define SBSDIO_WATERMARK		0x10008 | ||||||
|  | /* control busy signal generation */ | ||||||
|  | #define SBSDIO_DEVICE_CTL		0x10009 | ||||||
|  | 
 | ||||||
|  | /* SB Address Window Low (b15) */ | ||||||
|  | #define SBSDIO_FUNC1_SBADDRLOW		0x1000A | ||||||
|  | /* SB Address Window Mid (b23:b16) */ | ||||||
|  | #define SBSDIO_FUNC1_SBADDRMID		0x1000B | ||||||
|  | /* SB Address Window High (b31:b24)    */ | ||||||
|  | #define SBSDIO_FUNC1_SBADDRHIGH		0x1000C | ||||||
|  | /* Frame Control (frame term/abort) */ | ||||||
|  | #define SBSDIO_FUNC1_FRAMECTRL		0x1000D | ||||||
|  | /* ChipClockCSR (ALP/HT ctl/status) */ | ||||||
|  | #define SBSDIO_FUNC1_CHIPCLKCSR		0x1000E | ||||||
|  | /* SdioPullUp (on cmd, d0-d2) */ | ||||||
|  | #define SBSDIO_FUNC1_SDIOPULLUP		0x1000F | ||||||
|  | /* Write Frame Byte Count Low */ | ||||||
|  | #define SBSDIO_FUNC1_WFRAMEBCLO		0x10019 | ||||||
|  | /* Write Frame Byte Count High */ | ||||||
|  | #define SBSDIO_FUNC1_WFRAMEBCHI		0x1001A | ||||||
|  | /* Read Frame Byte Count Low */ | ||||||
|  | #define SBSDIO_FUNC1_RFRAMEBCLO		0x1001B | ||||||
|  | /* Read Frame Byte Count High */ | ||||||
|  | #define SBSDIO_FUNC1_RFRAMEBCHI		0x1001C | ||||||
|  | 
 | ||||||
|  | #define SBSDIO_FUNC1_MISC_REG_START	0x10000	/* f1 misc register start */ | ||||||
|  | #define SBSDIO_FUNC1_MISC_REG_LIMIT	0x1001C	/* f1 misc register end */ | ||||||
|  | 
 | ||||||
|  | /* function 1 OCP space */ | ||||||
|  | 
 | ||||||
|  | /* sb offset addr is <= 15 bits, 32k */ | ||||||
|  | #define SBSDIO_SB_OFT_ADDR_MASK		0x07FFF | ||||||
|  | #define SBSDIO_SB_OFT_ADDR_LIMIT	0x08000 | ||||||
|  | /* with b15, maps to 32-bit SB access */ | ||||||
|  | #define SBSDIO_SB_ACCESS_2_4B_FLAG	0x08000 | ||||||
|  | 
 | ||||||
|  | /* valid bits in SBSDIO_FUNC1_SBADDRxxx regs */ | ||||||
|  | 
 | ||||||
|  | #define SBSDIO_SBADDRLOW_MASK		0x80	/* Valid bits in SBADDRLOW */ | ||||||
|  | #define SBSDIO_SBADDRMID_MASK		0xff	/* Valid bits in SBADDRMID */ | ||||||
|  | #define SBSDIO_SBADDRHIGH_MASK		0xffU	/* Valid bits in SBADDRHIGH */ | ||||||
|  | /* Address bits from SBADDR regs */ | ||||||
|  | #define SBSDIO_SBWINDOW_MASK		0xffff8000 | ||||||
|  | 
 | ||||||
|  | #define SDIOH_READ              0	/* Read request */ | ||||||
|  | #define SDIOH_WRITE             1	/* Write request */ | ||||||
|  | 
 | ||||||
|  | #define SDIOH_DATA_FIX          0	/* Fixed addressing */ | ||||||
|  | #define SDIOH_DATA_INC          1	/* Incremental addressing */ | ||||||
|  | 
 | ||||||
|  | /* internal return code */ | ||||||
|  | #define SUCCESS	0 | ||||||
|  | #define ERROR	1 | ||||||
|  | 
 | ||||||
|  | struct brcmf_sdreg { | ||||||
|  | 	int func; | ||||||
|  | 	int offset; | ||||||
|  | 	int value; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct brcmf_sdio_dev { | ||||||
|  | 	struct sdio_func *func[SDIO_MAX_FUNCS]; | ||||||
|  | 	u8 num_funcs;			/* Supported funcs on client */ | ||||||
|  | 	u32 func_cis_ptr[SDIOD_MAX_IOFUNCS]; | ||||||
|  | 	u32 sbwad;			/* Save backplane window address */ | ||||||
|  | 	bool regfail;			/* status of last reg_r/w call */ | ||||||
|  | 	void *bus; | ||||||
|  | 	atomic_t suspend;		/* suspend flag */ | ||||||
|  | 	wait_queue_head_t request_byte_wait; | ||||||
|  | 	wait_queue_head_t request_word_wait; | ||||||
|  | 	wait_queue_head_t request_packet_wait; | ||||||
|  | 	wait_queue_head_t request_buffer_wait; | ||||||
|  | 
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* Register/deregister device interrupt handler. */ | ||||||
|  | extern int | ||||||
|  | brcmf_sdcard_intr_reg(struct brcmf_sdio_dev *sdiodev); | ||||||
|  | 
 | ||||||
|  | extern int brcmf_sdcard_intr_dereg(struct brcmf_sdio_dev *sdiodev); | ||||||
|  | 
 | ||||||
|  | /* Access SDIO address space (e.g. CCCR) using CMD52 (single-byte interface).
 | ||||||
|  |  *   fn:   function number | ||||||
|  |  *   addr: unmodified SDIO-space address | ||||||
|  |  *   data: data byte to write | ||||||
|  |  *   err:  pointer to error code (or NULL) | ||||||
|  |  */ | ||||||
|  | extern u8 brcmf_sdcard_cfg_read(struct brcmf_sdio_dev *sdiodev, uint func, | ||||||
|  | 				u32 addr, int *err); | ||||||
|  | extern void brcmf_sdcard_cfg_write(struct brcmf_sdio_dev *sdiodev, uint func, | ||||||
|  | 				   u32 addr, u8 data, int *err); | ||||||
|  | 
 | ||||||
|  | /* Synchronous access to device (client) core registers via CMD53 to F1.
 | ||||||
|  |  *   addr: backplane address (i.e. >= regsva from attach) | ||||||
|  |  *   size: register width in bytes (2 or 4) | ||||||
|  |  *   data: data for register write | ||||||
|  |  */ | ||||||
|  | extern u32 | ||||||
|  | brcmf_sdcard_reg_read(struct brcmf_sdio_dev *sdiodev, u32 addr, uint size); | ||||||
|  | 
 | ||||||
|  | extern u32 | ||||||
|  | brcmf_sdcard_reg_write(struct brcmf_sdio_dev *sdiodev, u32 addr, uint size, | ||||||
|  | 		       u32 data); | ||||||
|  | 
 | ||||||
|  | /* Indicate if last reg read/write failed */ | ||||||
|  | extern bool brcmf_sdcard_regfail(struct brcmf_sdio_dev *sdiodev); | ||||||
|  | 
 | ||||||
|  | /* Buffer transfer to/from device (client) core via cmd53.
 | ||||||
|  |  *   fn:       function number | ||||||
|  |  *   addr:     backplane address (i.e. >= regsva from attach) | ||||||
|  |  *   flags:    backplane width, address increment, sync/async | ||||||
|  |  *   buf:      pointer to memory data buffer | ||||||
|  |  *   nbytes:   number of bytes to transfer to/from buf | ||||||
|  |  *   pkt:      pointer to packet associated with buf (if any) | ||||||
|  |  *   complete: callback function for command completion (async only) | ||||||
|  |  *   handle:   handle for completion callback (first arg in callback) | ||||||
|  |  * Returns 0 or error code. | ||||||
|  |  * NOTE: Async operation is not currently supported. | ||||||
|  |  */ | ||||||
|  | extern int | ||||||
|  | brcmf_sdcard_send_buf(struct brcmf_sdio_dev *sdiodev, u32 addr, uint fn, | ||||||
|  | 		      uint flags, u8 *buf, uint nbytes, struct sk_buff *pkt); | ||||||
|  | extern int | ||||||
|  | brcmf_sdcard_recv_buf(struct brcmf_sdio_dev *sdiodev, u32 addr, uint fn, | ||||||
|  | 		      uint flags, u8 *buf, uint nbytes, struct sk_buff *pkt); | ||||||
|  | 
 | ||||||
|  | /* Flags bits */ | ||||||
|  | 
 | ||||||
|  | /* Four-byte target (backplane) width (vs. two-byte) */ | ||||||
|  | #define SDIO_REQ_4BYTE	0x1 | ||||||
|  | /* Fixed address (FIFO) (vs. incrementing address) */ | ||||||
|  | #define SDIO_REQ_FIXED	0x2 | ||||||
|  | /* Async request (vs. sync request) */ | ||||||
|  | #define SDIO_REQ_ASYNC	0x4 | ||||||
|  | 
 | ||||||
|  | /* Read/write to memory block (F1, no FIFO) via CMD53 (sync only).
 | ||||||
|  |  *   rw:       read or write (0/1) | ||||||
|  |  *   addr:     direct SDIO address | ||||||
|  |  *   buf:      pointer to memory data buffer | ||||||
|  |  *   nbytes:   number of bytes to transfer to/from buf | ||||||
|  |  * Returns 0 or error code. | ||||||
|  |  */ | ||||||
|  | extern int brcmf_sdcard_rwdata(struct brcmf_sdio_dev *sdiodev, uint rw, | ||||||
|  | 			       u32 addr, u8 *buf, uint nbytes); | ||||||
|  | 
 | ||||||
|  | /* Issue an abort to the specified function */ | ||||||
|  | extern int brcmf_sdcard_abort(struct brcmf_sdio_dev *sdiodev, uint fn); | ||||||
|  | 
 | ||||||
|  | /* platform specific/high level functions */ | ||||||
|  | extern int brcmf_sdio_probe(struct brcmf_sdio_dev *sdiodev); | ||||||
|  | extern int brcmf_sdio_remove(struct brcmf_sdio_dev *sdiodev); | ||||||
|  | 
 | ||||||
|  | extern int brcmf_sdcard_set_sbaddr_window(struct brcmf_sdio_dev *sdiodev, | ||||||
|  | 					  u32 address); | ||||||
|  | 
 | ||||||
|  | /* attach, return handler on success, NULL if failed.
 | ||||||
|  |  *  The handler shall be provided by all subsequent calls. No local cache | ||||||
|  |  *  cfghdl points to the starting address of pci device mapped memory | ||||||
|  |  */ | ||||||
|  | extern int brcmf_sdioh_attach(struct brcmf_sdio_dev *sdiodev); | ||||||
|  | extern void brcmf_sdioh_detach(struct brcmf_sdio_dev *sdiodev); | ||||||
|  | 
 | ||||||
|  | /* read or write one byte using cmd52 */ | ||||||
|  | extern int brcmf_sdioh_request_byte(struct brcmf_sdio_dev *sdiodev, uint rw, | ||||||
|  | 				    uint fnc, uint addr, u8 *byte); | ||||||
|  | 
 | ||||||
|  | /* read or write 2/4 bytes using cmd53 */ | ||||||
|  | extern int | ||||||
|  | brcmf_sdioh_request_word(struct brcmf_sdio_dev *sdiodev, | ||||||
|  | 			 uint rw, uint fnc, uint addr, | ||||||
|  | 			 u32 *word, uint nbyte); | ||||||
|  | 
 | ||||||
|  | /* read or write any buffer using cmd53 */ | ||||||
|  | extern int | ||||||
|  | brcmf_sdioh_request_buffer(struct brcmf_sdio_dev *sdiodev, | ||||||
|  | 			   uint fix_inc, uint rw, uint fnc_num, | ||||||
|  | 			   u32 addr, uint regwidth, | ||||||
|  | 			   u32 buflen, u8 *buffer, struct sk_buff *pkt); | ||||||
|  | 
 | ||||||
|  | /* Watchdog timer interface for pm ops */ | ||||||
|  | extern void brcmf_sdio_wdtmr_enable(struct brcmf_sdio_dev *sdiodev, | ||||||
|  | 				    bool enable); | ||||||
|  | 
 | ||||||
|  | extern void *brcmf_sdbrcm_probe(u16 bus_no, u16 slot, u16 func, uint bustype, | ||||||
|  | 				u32 regsva, struct brcmf_sdio_dev *sdiodev); | ||||||
|  | extern void brcmf_sdbrcm_disconnect(void *ptr); | ||||||
|  | extern void brcmf_sdbrcm_isr(void *arg); | ||||||
|  | #endif				/* _BRCM_SDH_H_ */ | ||||||
							
								
								
									
										3730
									
								
								drivers/net/wireless/brcm80211/brcmfmac/wl_cfg80211.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3730
									
								
								drivers/net/wireless/brcm80211/brcmfmac/wl_cfg80211.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										375
									
								
								drivers/net/wireless/brcm80211/brcmfmac/wl_cfg80211.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										375
									
								
								drivers/net/wireless/brcm80211/brcmfmac/wl_cfg80211.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,375 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _wl_cfg80211_h_ | ||||||
|  | #define _wl_cfg80211_h_ | ||||||
|  | 
 | ||||||
|  | struct brcmf_cfg80211_conf; | ||||||
|  | struct brcmf_cfg80211_iface; | ||||||
|  | struct brcmf_cfg80211_priv; | ||||||
|  | struct brcmf_cfg80211_security; | ||||||
|  | struct brcmf_cfg80211_ibss; | ||||||
|  | 
 | ||||||
|  | #define WL_DBG_NONE		0 | ||||||
|  | #define WL_DBG_CONN		(1 << 5) | ||||||
|  | #define WL_DBG_SCAN		(1 << 4) | ||||||
|  | #define WL_DBG_TRACE		(1 << 3) | ||||||
|  | #define WL_DBG_INFO		(1 << 1) | ||||||
|  | #define WL_DBG_ERR		(1 << 0) | ||||||
|  | #define WL_DBG_MASK		((WL_DBG_INFO | WL_DBG_ERR | WL_DBG_TRACE) | \ | ||||||
|  | 				(WL_DBG_SCAN) | (WL_DBG_CONN)) | ||||||
|  | 
 | ||||||
|  | #define	WL_ERR(fmt, args...)					\ | ||||||
|  | do {								\ | ||||||
|  | 	if (brcmf_dbg_level & WL_DBG_ERR) {			\ | ||||||
|  | 		if (net_ratelimit()) {				\ | ||||||
|  | 			printk(KERN_ERR "ERROR @%s : " fmt,	\ | ||||||
|  | 				__func__, ##args);		\ | ||||||
|  | 		}						\ | ||||||
|  | 	}							\ | ||||||
|  | } while (0) | ||||||
|  | 
 | ||||||
|  | #if (defined BCMDBG) | ||||||
|  | #define	WL_INFO(fmt, args...)					\ | ||||||
|  | do {								\ | ||||||
|  | 	if (brcmf_dbg_level & WL_DBG_INFO) {			\ | ||||||
|  | 		if (net_ratelimit()) {				\ | ||||||
|  | 			printk(KERN_ERR "INFO @%s : " fmt,	\ | ||||||
|  | 				__func__, ##args);		\ | ||||||
|  | 		}						\ | ||||||
|  | 	}							\ | ||||||
|  | } while (0) | ||||||
|  | 
 | ||||||
|  | #define	WL_TRACE(fmt, args...)					\ | ||||||
|  | do {								\ | ||||||
|  | 	if (brcmf_dbg_level & WL_DBG_TRACE) {			\ | ||||||
|  | 		if (net_ratelimit()) {				\ | ||||||
|  | 			printk(KERN_ERR "TRACE @%s : " fmt,	\ | ||||||
|  | 				__func__, ##args);		\ | ||||||
|  | 		}						\ | ||||||
|  | 	}							\ | ||||||
|  | } while (0) | ||||||
|  | 
 | ||||||
|  | #define	WL_SCAN(fmt, args...)					\ | ||||||
|  | do {								\ | ||||||
|  | 	if (brcmf_dbg_level & WL_DBG_SCAN) {			\ | ||||||
|  | 		if (net_ratelimit()) {				\ | ||||||
|  | 			printk(KERN_ERR "SCAN @%s : " fmt,	\ | ||||||
|  | 				__func__, ##args);		\ | ||||||
|  | 		}						\ | ||||||
|  | 	}							\ | ||||||
|  | } while (0) | ||||||
|  | 
 | ||||||
|  | #define	WL_CONN(fmt, args...)					\ | ||||||
|  | do {								\ | ||||||
|  | 	if (brcmf_dbg_level & WL_DBG_CONN) {			\ | ||||||
|  | 		if (net_ratelimit()) {				\ | ||||||
|  | 			printk(KERN_ERR "CONN @%s : " fmt,	\ | ||||||
|  | 				__func__, ##args);		\ | ||||||
|  | 		}						\ | ||||||
|  | 	}							\ | ||||||
|  | } while (0) | ||||||
|  | 
 | ||||||
|  | #else /* (defined BCMDBG) */ | ||||||
|  | #define	WL_INFO(fmt, args...) | ||||||
|  | #define	WL_TRACE(fmt, args...) | ||||||
|  | #define	WL_SCAN(fmt, args...) | ||||||
|  | #define	WL_CONN(fmt, args...) | ||||||
|  | #endif /* (defined BCMDBG) */ | ||||||
|  | 
 | ||||||
|  | #define WL_NUM_SCAN_MAX		1 | ||||||
|  | #define WL_NUM_PMKIDS_MAX	MAXPMKID	/* will be used | ||||||
|  | 						 * for 2.6.33 kernel | ||||||
|  | 						 * or later | ||||||
|  | 						 */ | ||||||
|  | #define WL_SCAN_BUF_MAX			(1024 * 8) | ||||||
|  | #define WL_TLV_INFO_MAX			1024 | ||||||
|  | #define WL_BSS_INFO_MAX			2048 | ||||||
|  | #define WL_ASSOC_INFO_MAX	512	/* | ||||||
|  | 				 * needs to grab assoc info from dongle to | ||||||
|  | 				 * report it to cfg80211 through "connect" | ||||||
|  | 				 * event | ||||||
|  | 				 */ | ||||||
|  | #define WL_DCMD_LEN_MAX	1024 | ||||||
|  | #define WL_EXTRA_BUF_MAX	2048 | ||||||
|  | #define WL_ISCAN_BUF_MAX	2048	/* | ||||||
|  | 				 * the buf length can be BRCMF_DCMD_MAXLEN | ||||||
|  | 				 * to reduce iteration | ||||||
|  | 				 */ | ||||||
|  | #define WL_ISCAN_TIMER_INTERVAL_MS	3000 | ||||||
|  | #define WL_SCAN_ERSULTS_LAST	(BRCMF_SCAN_RESULTS_NO_MEM+1) | ||||||
|  | #define WL_AP_MAX	256	/* virtually unlimitted as long | ||||||
|  | 				 * as kernel memory allows | ||||||
|  | 				 */ | ||||||
|  | 
 | ||||||
|  | #define WL_ROAM_TRIGGER_LEVEL		-75 | ||||||
|  | #define WL_ROAM_DELTA			20 | ||||||
|  | #define WL_BEACON_TIMEOUT		3 | ||||||
|  | 
 | ||||||
|  | #define WL_SCAN_CHANNEL_TIME		40 | ||||||
|  | #define WL_SCAN_UNASSOC_TIME		40 | ||||||
|  | #define WL_SCAN_PASSIVE_TIME		120 | ||||||
|  | 
 | ||||||
|  | /* dongle status */ | ||||||
|  | enum wl_status { | ||||||
|  | 	WL_STATUS_READY, | ||||||
|  | 	WL_STATUS_SCANNING, | ||||||
|  | 	WL_STATUS_SCAN_ABORTING, | ||||||
|  | 	WL_STATUS_CONNECTING, | ||||||
|  | 	WL_STATUS_CONNECTED | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* wi-fi mode */ | ||||||
|  | enum wl_mode { | ||||||
|  | 	WL_MODE_BSS, | ||||||
|  | 	WL_MODE_IBSS, | ||||||
|  | 	WL_MODE_AP | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* dongle profile list */ | ||||||
|  | enum wl_prof_list { | ||||||
|  | 	WL_PROF_MODE, | ||||||
|  | 	WL_PROF_SSID, | ||||||
|  | 	WL_PROF_SEC, | ||||||
|  | 	WL_PROF_IBSS, | ||||||
|  | 	WL_PROF_BAND, | ||||||
|  | 	WL_PROF_BSSID, | ||||||
|  | 	WL_PROF_ACT, | ||||||
|  | 	WL_PROF_BEACONINT, | ||||||
|  | 	WL_PROF_DTIMPERIOD | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* dongle iscan state */ | ||||||
|  | enum wl_iscan_state { | ||||||
|  | 	WL_ISCAN_STATE_IDLE, | ||||||
|  | 	WL_ISCAN_STATE_SCANING | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* dongle configuration */ | ||||||
|  | struct brcmf_cfg80211_conf { | ||||||
|  | 	u32 mode;		/* adhoc , infrastructure or ap */ | ||||||
|  | 	u32 frag_threshold; | ||||||
|  | 	u32 rts_threshold; | ||||||
|  | 	u32 retry_short; | ||||||
|  | 	u32 retry_long; | ||||||
|  | 	s32 tx_power; | ||||||
|  | 	struct ieee80211_channel channel; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* cfg80211 main event loop */ | ||||||
|  | struct brcmf_cfg80211_event_loop { | ||||||
|  | 	s32(*handler[BRCMF_E_LAST]) (struct brcmf_cfg80211_priv *cfg_priv, | ||||||
|  | 				     struct net_device *ndev, | ||||||
|  | 				     const struct brcmf_event_msg *e, | ||||||
|  | 				     void *data); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* representing interface of cfg80211 plane */ | ||||||
|  | struct brcmf_cfg80211_iface { | ||||||
|  | 	struct brcmf_cfg80211_priv *cfg_priv; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct brcmf_cfg80211_dev { | ||||||
|  | 	void *driver_data;	/* to store cfg80211 object information */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* basic structure of scan request */ | ||||||
|  | struct brcmf_cfg80211_scan_req { | ||||||
|  | 	struct brcmf_ssid_le ssid_le; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* basic structure of information element */ | ||||||
|  | struct brcmf_cfg80211_ie { | ||||||
|  | 	u16 offset; | ||||||
|  | 	u8 buf[WL_TLV_INFO_MAX]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* event queue for cfg80211 main event */ | ||||||
|  | struct brcmf_cfg80211_event_q { | ||||||
|  | 	struct list_head evt_q_list; | ||||||
|  | 	u32 etype; | ||||||
|  | 	struct brcmf_event_msg emsg; | ||||||
|  | 	s8 edata[1]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* security information with currently associated ap */ | ||||||
|  | struct brcmf_cfg80211_security { | ||||||
|  | 	u32 wpa_versions; | ||||||
|  | 	u32 auth_type; | ||||||
|  | 	u32 cipher_pairwise; | ||||||
|  | 	u32 cipher_group; | ||||||
|  | 	u32 wpa_auth; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* ibss information for currently joined ibss network */ | ||||||
|  | struct brcmf_cfg80211_ibss { | ||||||
|  | 	u8 beacon_interval;	/* in millisecond */ | ||||||
|  | 	u8 atim;		/* in millisecond */ | ||||||
|  | 	s8 join_only; | ||||||
|  | 	u8 band; | ||||||
|  | 	u8 channel; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* dongle profile */ | ||||||
|  | struct brcmf_cfg80211_profile { | ||||||
|  | 	u32 mode; | ||||||
|  | 	struct brcmf_ssid ssid; | ||||||
|  | 	u8 bssid[ETH_ALEN]; | ||||||
|  | 	u16 beacon_interval; | ||||||
|  | 	u8 dtim_period; | ||||||
|  | 	struct brcmf_cfg80211_security sec; | ||||||
|  | 	struct brcmf_cfg80211_ibss ibss; | ||||||
|  | 	s32 band; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* dongle iscan event loop */ | ||||||
|  | struct brcmf_cfg80211_iscan_eloop { | ||||||
|  | 	s32 (*handler[WL_SCAN_ERSULTS_LAST]) | ||||||
|  | 		(struct brcmf_cfg80211_priv *cfg_priv); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* dongle iscan controller */ | ||||||
|  | struct brcmf_cfg80211_iscan_ctrl { | ||||||
|  | 	struct net_device *ndev; | ||||||
|  | 	struct timer_list timer; | ||||||
|  | 	u32 timer_ms; | ||||||
|  | 	u32 timer_on; | ||||||
|  | 	s32 state; | ||||||
|  | 	struct work_struct work; | ||||||
|  | 	struct brcmf_cfg80211_iscan_eloop el; | ||||||
|  | 	void *data; | ||||||
|  | 	s8 dcmd_buf[BRCMF_DCMD_SMLEN]; | ||||||
|  | 	s8 scan_buf[WL_ISCAN_BUF_MAX]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* association inform */ | ||||||
|  | struct brcmf_cfg80211_connect_info { | ||||||
|  | 	u8 *req_ie; | ||||||
|  | 	s32 req_ie_len; | ||||||
|  | 	u8 *resp_ie; | ||||||
|  | 	s32 resp_ie_len; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* assoc ie length */ | ||||||
|  | struct brcmf_cfg80211_assoc_ielen { | ||||||
|  | 	u32 req_len; | ||||||
|  | 	u32 resp_len; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* wpa2 pmk list */ | ||||||
|  | struct brcmf_cfg80211_pmk_list { | ||||||
|  | 	struct pmkid_list pmkids; | ||||||
|  | 	struct pmkid foo[MAXPMKID - 1]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* dongle private data of cfg80211 interface */ | ||||||
|  | struct brcmf_cfg80211_priv { | ||||||
|  | 	struct wireless_dev *wdev;	/* representing wl cfg80211 device */ | ||||||
|  | 	struct brcmf_cfg80211_conf *conf;	/* dongle configuration */ | ||||||
|  | 	struct cfg80211_scan_request *scan_request;	/* scan request
 | ||||||
|  | 							 object */ | ||||||
|  | 	struct brcmf_cfg80211_event_loop el;	/* main event loop */ | ||||||
|  | 	struct list_head evt_q_list;	/* used for event queue */ | ||||||
|  | 	spinlock_t	 evt_q_lock;	/* for event queue synchronization */ | ||||||
|  | 	struct mutex usr_sync;	/* maily for dongle up/down synchronization */ | ||||||
|  | 	struct brcmf_scan_results *bss_list;	/* bss_list holding scanned
 | ||||||
|  | 						 ap information */ | ||||||
|  | 	struct brcmf_scan_results *scan_results; | ||||||
|  | 	struct brcmf_cfg80211_scan_req *scan_req_int;	/* scan request object
 | ||||||
|  | 						 for internal purpose */ | ||||||
|  | 	struct wl_cfg80211_bss_info *bss_info;	/* bss information for
 | ||||||
|  | 						 cfg80211 layer */ | ||||||
|  | 	struct brcmf_cfg80211_ie ie;	/* information element object for
 | ||||||
|  | 					 internal purpose */ | ||||||
|  | 	struct brcmf_cfg80211_profile *profile;	/* holding dongle profile */ | ||||||
|  | 	struct brcmf_cfg80211_iscan_ctrl *iscan;	/* iscan controller */ | ||||||
|  | 	struct brcmf_cfg80211_connect_info conn_info; /* association info */ | ||||||
|  | 	struct brcmf_cfg80211_pmk_list *pmk_list;	/* wpa2 pmk list */ | ||||||
|  | 	struct work_struct event_work;	/* event handler work struct */ | ||||||
|  | 	unsigned long status;		/* current dongle status */ | ||||||
|  | 	void *pub; | ||||||
|  | 	u32 channel;		/* current channel */ | ||||||
|  | 	bool iscan_on;		/* iscan on/off switch */ | ||||||
|  | 	bool iscan_kickstart;	/* indicate iscan already started */ | ||||||
|  | 	bool active_scan;	/* current scan mode */ | ||||||
|  | 	bool ibss_starter;	/* indicates this sta is ibss starter */ | ||||||
|  | 	bool link_up;		/* link/connection up flag */ | ||||||
|  | 	bool pwr_save;		/* indicate whether dongle to support
 | ||||||
|  | 					 power save mode */ | ||||||
|  | 	bool dongle_up;		/* indicate whether dongle up or not */ | ||||||
|  | 	bool roam_on;		/* on/off switch for dongle self-roaming */ | ||||||
|  | 	bool scan_tried;	/* indicates if first scan attempted */ | ||||||
|  | 	u8 *dcmd_buf;		/* dcmd buffer */ | ||||||
|  | 	u8 *extra_buf;		/* maily to grab assoc information */ | ||||||
|  | 	struct dentry *debugfsdir; | ||||||
|  | 	u8 ci[0] __aligned(NETDEV_ALIGN); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static inline struct wiphy *cfg_to_wiphy(struct brcmf_cfg80211_priv *w) | ||||||
|  | { | ||||||
|  | 	return w->wdev->wiphy; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline struct brcmf_cfg80211_priv *wiphy_to_cfg(struct wiphy *w) | ||||||
|  | { | ||||||
|  | 	return (struct brcmf_cfg80211_priv *)(wiphy_priv(w)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline struct brcmf_cfg80211_priv *wdev_to_cfg(struct wireless_dev *wd) | ||||||
|  | { | ||||||
|  | 	return (struct brcmf_cfg80211_priv *)(wdev_priv(wd)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline struct net_device *cfg_to_ndev(struct brcmf_cfg80211_priv *cfg) | ||||||
|  | { | ||||||
|  | 	return cfg->wdev->netdev; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline struct brcmf_cfg80211_priv *ndev_to_cfg(struct net_device *ndev) | ||||||
|  | { | ||||||
|  | 	return wdev_to_cfg(ndev->ieee80211_ptr); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #define iscan_to_cfg(i) ((struct brcmf_cfg80211_priv *)(i->data)) | ||||||
|  | #define cfg_to_iscan(w) (w->iscan) | ||||||
|  | 
 | ||||||
|  | static inline struct | ||||||
|  | brcmf_cfg80211_connect_info *cfg_to_conn(struct brcmf_cfg80211_priv *cfg) | ||||||
|  | { | ||||||
|  | 	return &cfg->conn_info; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline struct brcmf_bss_info *next_bss(struct brcmf_scan_results *list, | ||||||
|  | 					   struct brcmf_bss_info *bss) | ||||||
|  | { | ||||||
|  | 	return bss = bss ? | ||||||
|  | 		(struct brcmf_bss_info *)((unsigned long)bss + | ||||||
|  | 				       le32_to_cpu(bss->length)) : | ||||||
|  | 		list->bss_info; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | extern struct brcmf_cfg80211_dev *brcmf_cfg80211_attach(struct net_device *ndev, | ||||||
|  | 							struct device *busdev, | ||||||
|  | 							void *data); | ||||||
|  | extern void brcmf_cfg80211_detach(struct brcmf_cfg80211_dev *cfg); | ||||||
|  | 
 | ||||||
|  | /* event handler from dongle */ | ||||||
|  | extern void brcmf_cfg80211_event(struct net_device *ndev, | ||||||
|  | 				 const struct brcmf_event_msg *e, void *data); | ||||||
|  | extern s32 brcmf_cfg80211_up(struct brcmf_cfg80211_dev *cfg_dev); | ||||||
|  | extern s32 brcmf_cfg80211_down(struct brcmf_cfg80211_dev *cfg_dev); | ||||||
|  | 
 | ||||||
|  | #endif				/* _wl_cfg80211_h_ */ | ||||||
							
								
								
									
										51
									
								
								drivers/net/wireless/brcm80211/brcmsmac/Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								drivers/net/wireless/brcm80211/brcmsmac/Makefile
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,51 @@ | |||||||
|  | #
 | ||||||
|  | # Makefile fragment for Broadcom 802.11n Networking Device Driver
 | ||||||
|  | #
 | ||||||
|  | # Copyright (c) 2010 Broadcom Corporation
 | ||||||
|  | #
 | ||||||
|  | # 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.
 | ||||||
|  | 
 | ||||||
|  | ccflags-y := \
 | ||||||
|  | 	-D__CHECK_ENDIAN__ \
 | ||||||
|  | 	-Idrivers/net/wireless/brcm80211/brcmsmac \
 | ||||||
|  | 	-Idrivers/net/wireless/brcm80211/brcmsmac/phy \
 | ||||||
|  | 	-Idrivers/net/wireless/brcm80211/include | ||||||
|  | 
 | ||||||
|  | BRCMSMAC_OFILES := \
 | ||||||
|  | 	mac80211_if.o \
 | ||||||
|  | 	ucode_loader.o \
 | ||||||
|  | 	ampdu.o \
 | ||||||
|  | 	antsel.o \
 | ||||||
|  | 	channel.o \
 | ||||||
|  | 	main.o \
 | ||||||
|  | 	phy_shim.o \
 | ||||||
|  | 	pmu.o \
 | ||||||
|  | 	rate.o \
 | ||||||
|  | 	stf.o \
 | ||||||
|  | 	aiutils.o \
 | ||||||
|  | 	phy/phy_cmn.o \
 | ||||||
|  | 	phy/phy_lcn.o \
 | ||||||
|  | 	phy/phy_n.o \
 | ||||||
|  | 	phy/phytbl_lcn.o \
 | ||||||
|  | 	phy/phytbl_n.o \
 | ||||||
|  | 	phy/phy_qmath.o \
 | ||||||
|  | 	otp.o \
 | ||||||
|  | 	srom.o \
 | ||||||
|  | 	dma.o \
 | ||||||
|  | 	nicpci.o \
 | ||||||
|  | 	brcms_trace_events.o | ||||||
|  | 
 | ||||||
|  | MODULEPFX := brcmsmac | ||||||
|  | 
 | ||||||
|  | obj-$(CONFIG_BRCMSMAC)	+= $(MODULEPFX).o | ||||||
|  | $(MODULEPFX)-objs	= $(BRCMSMAC_OFILES) | ||||||
							
								
								
									
										2079
									
								
								drivers/net/wireless/brcm80211/brcmsmac/aiutils.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2079
									
								
								drivers/net/wireless/brcm80211/brcmsmac/aiutils.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										378
									
								
								drivers/net/wireless/brcm80211/brcmsmac/aiutils.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										378
									
								
								drivers/net/wireless/brcm80211/brcmsmac/aiutils.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,378 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2011 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef	_BRCM_AIUTILS_H_ | ||||||
|  | #define	_BRCM_AIUTILS_H_ | ||||||
|  | 
 | ||||||
|  | #include "types.h" | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * SOC Interconnect Address Map. | ||||||
|  |  * All regions may not exist on all chips. | ||||||
|  |  */ | ||||||
|  | /* each core gets 4Kbytes for registers */ | ||||||
|  | #define SI_CORE_SIZE		0x1000 | ||||||
|  | /*
 | ||||||
|  |  * Max cores (this is arbitrary, for software | ||||||
|  |  * convenience and could be changed if we | ||||||
|  |  * make any larger chips | ||||||
|  |  */ | ||||||
|  | #define	SI_MAXCORES		16 | ||||||
|  | 
 | ||||||
|  | /* Client Mode sb2pcitranslation2 size in bytes */ | ||||||
|  | #define SI_PCI_DMA_SZ		0x40000000 | ||||||
|  | 
 | ||||||
|  | /* PCIE Client Mode sb2pcitranslation2 (2 ZettaBytes), high 32 bits */ | ||||||
|  | #define SI_PCIE_DMA_H32		0x80000000 | ||||||
|  | 
 | ||||||
|  | /* core codes */ | ||||||
|  | #define	NODEV_CORE_ID		0x700	/* Invalid coreid */ | ||||||
|  | #define	CC_CORE_ID		0x800	/* chipcommon core */ | ||||||
|  | #define	ILINE20_CORE_ID		0x801	/* iline20 core */ | ||||||
|  | #define	SRAM_CORE_ID		0x802	/* sram core */ | ||||||
|  | #define	SDRAM_CORE_ID		0x803	/* sdram core */ | ||||||
|  | #define	PCI_CORE_ID		0x804	/* pci core */ | ||||||
|  | #define	MIPS_CORE_ID		0x805	/* mips core */ | ||||||
|  | #define	ENET_CORE_ID		0x806	/* enet mac core */ | ||||||
|  | #define	CODEC_CORE_ID		0x807	/* v90 codec core */ | ||||||
|  | #define	USB_CORE_ID		0x808	/* usb 1.1 host/device core */ | ||||||
|  | #define	ADSL_CORE_ID		0x809	/* ADSL core */ | ||||||
|  | #define	ILINE100_CORE_ID	0x80a	/* iline100 core */ | ||||||
|  | #define	IPSEC_CORE_ID		0x80b	/* ipsec core */ | ||||||
|  | #define	UTOPIA_CORE_ID		0x80c	/* utopia core */ | ||||||
|  | #define	PCMCIA_CORE_ID		0x80d	/* pcmcia core */ | ||||||
|  | #define	SOCRAM_CORE_ID		0x80e	/* internal memory core */ | ||||||
|  | #define	MEMC_CORE_ID		0x80f	/* memc sdram core */ | ||||||
|  | #define	OFDM_CORE_ID		0x810	/* OFDM phy core */ | ||||||
|  | #define	EXTIF_CORE_ID		0x811	/* external interface core */ | ||||||
|  | #define	D11_CORE_ID		0x812	/* 802.11 MAC core */ | ||||||
|  | #define	APHY_CORE_ID		0x813	/* 802.11a phy core */ | ||||||
|  | #define	BPHY_CORE_ID		0x814	/* 802.11b phy core */ | ||||||
|  | #define	GPHY_CORE_ID		0x815	/* 802.11g phy core */ | ||||||
|  | #define	MIPS33_CORE_ID		0x816	/* mips3302 core */ | ||||||
|  | #define	USB11H_CORE_ID		0x817	/* usb 1.1 host core */ | ||||||
|  | #define	USB11D_CORE_ID		0x818	/* usb 1.1 device core */ | ||||||
|  | #define	USB20H_CORE_ID		0x819	/* usb 2.0 host core */ | ||||||
|  | #define	USB20D_CORE_ID		0x81a	/* usb 2.0 device core */ | ||||||
|  | #define	SDIOH_CORE_ID		0x81b	/* sdio host core */ | ||||||
|  | #define	ROBO_CORE_ID		0x81c	/* roboswitch core */ | ||||||
|  | #define	ATA100_CORE_ID		0x81d	/* parallel ATA core */ | ||||||
|  | #define	SATAXOR_CORE_ID		0x81e	/* serial ATA & XOR DMA core */ | ||||||
|  | #define	GIGETH_CORE_ID		0x81f	/* gigabit ethernet core */ | ||||||
|  | #define	PCIE_CORE_ID		0x820	/* pci express core */ | ||||||
|  | #define	NPHY_CORE_ID		0x821	/* 802.11n 2x2 phy core */ | ||||||
|  | #define	SRAMC_CORE_ID		0x822	/* SRAM controller core */ | ||||||
|  | #define	MINIMAC_CORE_ID		0x823	/* MINI MAC/phy core */ | ||||||
|  | #define	ARM11_CORE_ID		0x824	/* ARM 1176 core */ | ||||||
|  | #define	ARM7S_CORE_ID		0x825	/* ARM7tdmi-s core */ | ||||||
|  | #define	LPPHY_CORE_ID		0x826	/* 802.11a/b/g phy core */ | ||||||
|  | #define	PMU_CORE_ID		0x827	/* PMU core */ | ||||||
|  | #define	SSNPHY_CORE_ID		0x828	/* 802.11n single-stream phy core */ | ||||||
|  | #define	SDIOD_CORE_ID		0x829	/* SDIO device core */ | ||||||
|  | #define	ARMCM3_CORE_ID		0x82a	/* ARM Cortex M3 core */ | ||||||
|  | #define	HTPHY_CORE_ID		0x82b	/* 802.11n 4x4 phy core */ | ||||||
|  | #define	MIPS74K_CORE_ID		0x82c	/* mips 74k core */ | ||||||
|  | #define	GMAC_CORE_ID		0x82d	/* Gigabit MAC core */ | ||||||
|  | #define	DMEMC_CORE_ID		0x82e	/* DDR1/2 memory controller core */ | ||||||
|  | #define	PCIERC_CORE_ID		0x82f	/* PCIE Root Complex core */ | ||||||
|  | #define	OCP_CORE_ID		0x830	/* OCP2OCP bridge core */ | ||||||
|  | #define	SC_CORE_ID		0x831	/* shared common core */ | ||||||
|  | #define	AHB_CORE_ID		0x832	/* OCP2AHB bridge core */ | ||||||
|  | #define	SPIH_CORE_ID		0x833	/* SPI host core */ | ||||||
|  | #define	I2S_CORE_ID		0x834	/* I2S core */ | ||||||
|  | #define	DMEMS_CORE_ID		0x835	/* SDR/DDR1 memory controller core */ | ||||||
|  | #define	DEF_SHIM_COMP		0x837	/* SHIM component in ubus/6362 */ | ||||||
|  | #define OOB_ROUTER_CORE_ID	0x367	/* OOB router core ID */ | ||||||
|  | #define	DEF_AI_COMP		0xfff	/* Default component, in ai chips it | ||||||
|  | 					 * maps all unused address ranges | ||||||
|  | 					 */ | ||||||
|  | 
 | ||||||
|  | /* chipcommon being the first core: */ | ||||||
|  | #define	SI_CC_IDX		0 | ||||||
|  | 
 | ||||||
|  | /* SOC Interconnect types (aka chip types) */ | ||||||
|  | #define	SOCI_AI			1 | ||||||
|  | 
 | ||||||
|  | /* Common core control flags */ | ||||||
|  | #define	SICF_BIST_EN		0x8000 | ||||||
|  | #define	SICF_PME_EN		0x4000 | ||||||
|  | #define	SICF_CORE_BITS		0x3ffc | ||||||
|  | #define	SICF_FGC		0x0002 | ||||||
|  | #define	SICF_CLOCK_EN		0x0001 | ||||||
|  | 
 | ||||||
|  | /* Common core status flags */ | ||||||
|  | #define	SISF_BIST_DONE		0x8000 | ||||||
|  | #define	SISF_BIST_ERROR		0x4000 | ||||||
|  | #define	SISF_GATED_CLK		0x2000 | ||||||
|  | #define	SISF_DMA64		0x1000 | ||||||
|  | #define	SISF_CORE_BITS		0x0fff | ||||||
|  | 
 | ||||||
|  | /* A register that is common to all cores to
 | ||||||
|  |  * communicate w/PMU regarding clock control. | ||||||
|  |  */ | ||||||
|  | #define SI_CLK_CTL_ST		0x1e0	/* clock control and status */ | ||||||
|  | 
 | ||||||
|  | /* clk_ctl_st register */ | ||||||
|  | #define	CCS_FORCEALP		0x00000001	/* force ALP request */ | ||||||
|  | #define	CCS_FORCEHT		0x00000002	/* force HT request */ | ||||||
|  | #define	CCS_FORCEILP		0x00000004	/* force ILP request */ | ||||||
|  | #define	CCS_ALPAREQ		0x00000008	/* ALP Avail Request */ | ||||||
|  | #define	CCS_HTAREQ		0x00000010	/* HT Avail Request */ | ||||||
|  | #define	CCS_FORCEHWREQOFF	0x00000020	/* Force HW Clock Request Off */ | ||||||
|  | #define CCS_ERSRC_REQ_MASK	0x00000700	/* external resource requests */ | ||||||
|  | #define CCS_ERSRC_REQ_SHIFT	8 | ||||||
|  | #define	CCS_ALPAVAIL		0x00010000	/* ALP is available */ | ||||||
|  | #define	CCS_HTAVAIL		0x00020000	/* HT is available */ | ||||||
|  | #define CCS_BP_ON_APL		0x00040000	/* RO: running on ALP clock */ | ||||||
|  | #define CCS_BP_ON_HT		0x00080000	/* RO: running on HT clock */ | ||||||
|  | #define CCS_ERSRC_STS_MASK	0x07000000	/* external resource status */ | ||||||
|  | #define CCS_ERSRC_STS_SHIFT	24 | ||||||
|  | 
 | ||||||
|  | /* HT avail in chipc and pcmcia on 4328a0 */ | ||||||
|  | #define	CCS0_HTAVAIL		0x00010000 | ||||||
|  | /* ALP avail in chipc and pcmcia on 4328a0 */ | ||||||
|  | #define	CCS0_ALPAVAIL		0x00020000 | ||||||
|  | 
 | ||||||
|  | /* Not really related to SOC Interconnect, but a couple of software
 | ||||||
|  |  * conventions for the use the flash space: | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* Minumum amount of flash we support */ | ||||||
|  | #define FLASH_MIN		0x00020000	/* Minimum flash size */ | ||||||
|  | 
 | ||||||
|  | #define	CC_SROM_OTP		0x800	/* SROM/OTP address space */ | ||||||
|  | 
 | ||||||
|  | /* gpiotimerval */ | ||||||
|  | #define GPIO_ONTIME_SHIFT	16 | ||||||
|  | 
 | ||||||
|  | /* Fields in clkdiv */ | ||||||
|  | #define	CLKD_OTP		0x000f0000 | ||||||
|  | #define	CLKD_OTP_SHIFT		16 | ||||||
|  | 
 | ||||||
|  | /* Package IDs */ | ||||||
|  | #define	BCM4717_PKG_ID		9	/* 4717 package id */ | ||||||
|  | #define	BCM4718_PKG_ID		10	/* 4718 package id */ | ||||||
|  | #define BCM43224_FAB_SMIC	0xa	/* the chip is manufactured by SMIC */ | ||||||
|  | 
 | ||||||
|  | /* these are router chips */ | ||||||
|  | #define	BCM4716_CHIP_ID		0x4716	/* 4716 chipcommon chipid */ | ||||||
|  | #define	BCM47162_CHIP_ID	47162	/* 47162 chipcommon chipid */ | ||||||
|  | #define	BCM4748_CHIP_ID		0x4748	/* 4716 chipcommon chipid (OTP, RBBU) */ | ||||||
|  | 
 | ||||||
|  | /* dynamic clock control defines */ | ||||||
|  | #define	LPOMINFREQ		25000	/* low power oscillator min */ | ||||||
|  | #define	LPOMAXFREQ		43000	/* low power oscillator max */ | ||||||
|  | #define	XTALMINFREQ		19800000	/* 20 MHz - 1% */ | ||||||
|  | #define	XTALMAXFREQ		20200000	/* 20 MHz + 1% */ | ||||||
|  | #define	PCIMINFREQ		25000000	/* 25 MHz */ | ||||||
|  | #define	PCIMAXFREQ		34000000	/* 33 MHz + fudge */ | ||||||
|  | 
 | ||||||
|  | #define	ILP_DIV_5MHZ		0	/* ILP = 5 MHz */ | ||||||
|  | #define	ILP_DIV_1MHZ		4	/* ILP = 1 MHz */ | ||||||
|  | 
 | ||||||
|  | /* clkctl xtal what flags */ | ||||||
|  | #define	XTAL			0x1	/* primary crystal oscillator (2050) */ | ||||||
|  | #define	PLL			0x2	/* main chip pll */ | ||||||
|  | 
 | ||||||
|  | /* clkctl clk mode */ | ||||||
|  | #define	CLK_FAST		0	/* force fast (pll) clock */ | ||||||
|  | #define	CLK_DYNAMIC		2	/* enable dynamic clock control */ | ||||||
|  | 
 | ||||||
|  | /* GPIO usage priorities */ | ||||||
|  | #define GPIO_DRV_PRIORITY	0	/* Driver */ | ||||||
|  | #define GPIO_APP_PRIORITY	1	/* Application */ | ||||||
|  | #define GPIO_HI_PRIORITY	2	/* Highest priority. Ignore GPIO | ||||||
|  | 					 * reservation | ||||||
|  | 					 */ | ||||||
|  | 
 | ||||||
|  | /* GPIO pull up/down */ | ||||||
|  | #define GPIO_PULLUP		0 | ||||||
|  | #define GPIO_PULLDN		1 | ||||||
|  | 
 | ||||||
|  | /* GPIO event regtype */ | ||||||
|  | #define GPIO_REGEVT		0	/* GPIO register event */ | ||||||
|  | #define GPIO_REGEVT_INTMSK	1	/* GPIO register event int mask */ | ||||||
|  | #define GPIO_REGEVT_INTPOL	2	/* GPIO register event int polarity */ | ||||||
|  | 
 | ||||||
|  | /* device path */ | ||||||
|  | #define SI_DEVPATH_BUFSZ	16	/* min buffer size in bytes */ | ||||||
|  | 
 | ||||||
|  | /* SI routine enumeration: to be used by update function with multiple hooks */ | ||||||
|  | #define	SI_DOATTACH	1 | ||||||
|  | #define SI_PCIDOWN	2 | ||||||
|  | #define SI_PCIUP	3 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Data structure to export all chip specific common variables | ||||||
|  |  *   public (read-only) portion of aiutils handle returned by si_attach() | ||||||
|  |  */ | ||||||
|  | struct si_pub { | ||||||
|  | 	uint buscoretype;	/* PCI_CORE_ID, PCIE_CORE_ID, PCMCIA_CORE_ID */ | ||||||
|  | 	uint buscorerev;	/* buscore rev */ | ||||||
|  | 	uint buscoreidx;	/* buscore index */ | ||||||
|  | 	int ccrev;		/* chip common core rev */ | ||||||
|  | 	u32 cccaps;		/* chip common capabilities */ | ||||||
|  | 	u32 cccaps_ext;	/* chip common capabilities extension */ | ||||||
|  | 	int pmurev;		/* pmu core rev */ | ||||||
|  | 	u32 pmucaps;		/* pmu capabilities */ | ||||||
|  | 	uint boardtype;		/* board type */ | ||||||
|  | 	uint boardvendor;	/* board vendor */ | ||||||
|  | 	uint boardflags;	/* board flags */ | ||||||
|  | 	uint boardflags2;	/* board flags2 */ | ||||||
|  | 	uint chip;		/* chip number */ | ||||||
|  | 	uint chiprev;		/* chip revision */ | ||||||
|  | 	uint chippkg;		/* chip package option */ | ||||||
|  | 	u32 chipst;		/* chip status */ | ||||||
|  | 	bool issim;		/* chip is in simulation or emulation */ | ||||||
|  | 	uint socirev;		/* SOC interconnect rev */ | ||||||
|  | 	bool pci_pr32414; | ||||||
|  | 
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct pci_dev; | ||||||
|  | 
 | ||||||
|  | struct gpioh_item { | ||||||
|  | 	void *arg; | ||||||
|  | 	bool level; | ||||||
|  | 	void (*handler) (u32 stat, void *arg); | ||||||
|  | 	u32 event; | ||||||
|  | 	struct gpioh_item *next; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* misc si info needed by some of the routines */ | ||||||
|  | struct si_info { | ||||||
|  | 	struct si_pub pub;	/* back plane public state (must be first) */ | ||||||
|  | 	struct pci_dev *pbus;	/* handle to pci bus */ | ||||||
|  | 	uint dev_coreid;	/* the core provides driver functions */ | ||||||
|  | 	void *intr_arg;		/* interrupt callback function arg */ | ||||||
|  | 	u32 (*intrsoff_fn) (void *intr_arg); /* turns chip interrupts off */ | ||||||
|  | 	/* restore chip interrupts */ | ||||||
|  | 	void (*intrsrestore_fn) (void *intr_arg, u32 arg); | ||||||
|  | 	/* check if interrupts are enabled */ | ||||||
|  | 	bool (*intrsenabled_fn) (void *intr_arg); | ||||||
|  | 
 | ||||||
|  | 	struct pcicore_info *pch; /* PCI/E core handle */ | ||||||
|  | 
 | ||||||
|  | 	struct list_head var_list; /* list of srom variables */ | ||||||
|  | 
 | ||||||
|  | 	void __iomem *curmap;			/* current regs va */ | ||||||
|  | 	void __iomem *regs[SI_MAXCORES];	/* other regs va */ | ||||||
|  | 
 | ||||||
|  | 	uint curidx;		/* current core index */ | ||||||
|  | 	uint numcores;		/* # discovered cores */ | ||||||
|  | 	uint coreid[SI_MAXCORES]; /* id of each core */ | ||||||
|  | 	u32 coresba[SI_MAXCORES]; /* backplane address of each core */ | ||||||
|  | 	void *regs2[SI_MAXCORES]; /* 2nd virtual address per core (usbh20) */ | ||||||
|  | 	u32 coresba2[SI_MAXCORES]; /* 2nd phys address per core (usbh20) */ | ||||||
|  | 	u32 coresba_size[SI_MAXCORES]; /* backplane address space size */ | ||||||
|  | 	u32 coresba2_size[SI_MAXCORES];	/* second address space size */ | ||||||
|  | 
 | ||||||
|  | 	void *curwrap;		/* current wrapper va */ | ||||||
|  | 	void *wrappers[SI_MAXCORES];	/* other cores wrapper va */ | ||||||
|  | 	u32 wrapba[SI_MAXCORES];	/* address of controlling wrapper */ | ||||||
|  | 
 | ||||||
|  | 	u32 cia[SI_MAXCORES];	/* erom cia entry for each core */ | ||||||
|  | 	u32 cib[SI_MAXCORES];	/* erom cia entry for each core */ | ||||||
|  | 	u32 oob_router;	/* oob router registers for axi */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Many of the routines below take an 'sih' handle as their first arg. | ||||||
|  |  * Allocate this by calling si_attach().  Free it by calling si_detach(). | ||||||
|  |  * At any one time, the sih is logically focused on one particular si core | ||||||
|  |  * (the "current core"). | ||||||
|  |  * Use si_setcore() or si_setcoreidx() to change the association to another core | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /* AMBA Interconnect exported externs */ | ||||||
|  | extern uint ai_flag(struct si_pub *sih); | ||||||
|  | extern void ai_setint(struct si_pub *sih, int siflag); | ||||||
|  | extern uint ai_coreidx(struct si_pub *sih); | ||||||
|  | extern uint ai_corevendor(struct si_pub *sih); | ||||||
|  | extern uint ai_corerev(struct si_pub *sih); | ||||||
|  | extern bool ai_iscoreup(struct si_pub *sih); | ||||||
|  | extern u32 ai_core_cflags(struct si_pub *sih, u32 mask, u32 val); | ||||||
|  | extern void ai_core_cflags_wo(struct si_pub *sih, u32 mask, u32 val); | ||||||
|  | extern u32 ai_core_sflags(struct si_pub *sih, u32 mask, u32 val); | ||||||
|  | extern uint ai_corereg(struct si_pub *sih, uint coreidx, uint regoff, uint mask, | ||||||
|  | 		       uint val); | ||||||
|  | extern void ai_core_reset(struct si_pub *sih, u32 bits, u32 resetbits); | ||||||
|  | extern void ai_core_disable(struct si_pub *sih, u32 bits); | ||||||
|  | extern int ai_numaddrspaces(struct si_pub *sih); | ||||||
|  | extern u32 ai_addrspace(struct si_pub *sih, uint asidx); | ||||||
|  | extern u32 ai_addrspacesize(struct si_pub *sih, uint asidx); | ||||||
|  | extern void ai_write_wrap_reg(struct si_pub *sih, u32 offset, u32 val); | ||||||
|  | 
 | ||||||
|  | /* === exported functions === */ | ||||||
|  | extern struct si_pub *ai_attach(void __iomem *regs, struct pci_dev *sdh); | ||||||
|  | extern void ai_detach(struct si_pub *sih); | ||||||
|  | extern uint ai_coreid(struct si_pub *sih); | ||||||
|  | extern uint ai_corerev(struct si_pub *sih); | ||||||
|  | extern uint ai_corereg(struct si_pub *sih, uint coreidx, uint regoff, uint mask, | ||||||
|  | 		uint val); | ||||||
|  | extern void ai_write_wrapperreg(struct si_pub *sih, u32 offset, u32 val); | ||||||
|  | extern u32 ai_core_cflags(struct si_pub *sih, u32 mask, u32 val); | ||||||
|  | extern u32 ai_core_sflags(struct si_pub *sih, u32 mask, u32 val); | ||||||
|  | extern bool ai_iscoreup(struct si_pub *sih); | ||||||
|  | extern uint ai_findcoreidx(struct si_pub *sih, uint coreid, uint coreunit); | ||||||
|  | extern void __iomem *ai_setcoreidx(struct si_pub *sih, uint coreidx); | ||||||
|  | extern void __iomem *ai_setcore(struct si_pub *sih, uint coreid, uint coreunit); | ||||||
|  | extern void __iomem *ai_switch_core(struct si_pub *sih, uint coreid, | ||||||
|  | 				    uint *origidx, uint *intr_val); | ||||||
|  | extern void ai_restore_core(struct si_pub *sih, uint coreid, uint intr_val); | ||||||
|  | extern void ai_core_reset(struct si_pub *sih, u32 bits, u32 resetbits); | ||||||
|  | extern void ai_core_disable(struct si_pub *sih, u32 bits); | ||||||
|  | extern u32 ai_alp_clock(struct si_pub *sih); | ||||||
|  | extern u32 ai_ilp_clock(struct si_pub *sih); | ||||||
|  | extern void ai_pci_setup(struct si_pub *sih, uint coremask); | ||||||
|  | extern void ai_setint(struct si_pub *sih, int siflag); | ||||||
|  | extern bool ai_backplane64(struct si_pub *sih); | ||||||
|  | extern void ai_register_intr_callback(struct si_pub *sih, void *intrsoff_fn, | ||||||
|  | 				      void *intrsrestore_fn, | ||||||
|  | 				      void *intrsenabled_fn, void *intr_arg); | ||||||
|  | extern void ai_deregister_intr_callback(struct si_pub *sih); | ||||||
|  | extern void ai_clkctl_init(struct si_pub *sih); | ||||||
|  | extern u16 ai_clkctl_fast_pwrup_delay(struct si_pub *sih); | ||||||
|  | extern bool ai_clkctl_cc(struct si_pub *sih, uint mode); | ||||||
|  | extern int ai_clkctl_xtal(struct si_pub *sih, uint what, bool on); | ||||||
|  | extern bool ai_deviceremoved(struct si_pub *sih); | ||||||
|  | extern u32 ai_gpiocontrol(struct si_pub *sih, u32 mask, u32 val, | ||||||
|  | 			     u8 priority); | ||||||
|  | 
 | ||||||
|  | /* OTP status */ | ||||||
|  | extern bool ai_is_otp_disabled(struct si_pub *sih); | ||||||
|  | 
 | ||||||
|  | /* SPROM availability */ | ||||||
|  | extern bool ai_is_sprom_available(struct si_pub *sih); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Build device path. Path size must be >= SI_DEVPATH_BUFSZ. | ||||||
|  |  * The returned path is NULL terminated and has trailing '/'. | ||||||
|  |  * Return 0 on success, nonzero otherwise. | ||||||
|  |  */ | ||||||
|  | extern int ai_devpath(struct si_pub *sih, char *path, int size); | ||||||
|  | 
 | ||||||
|  | extern void ai_pci_sleep(struct si_pub *sih); | ||||||
|  | extern void ai_pci_down(struct si_pub *sih); | ||||||
|  | extern void ai_pci_up(struct si_pub *sih); | ||||||
|  | extern int ai_pci_fixcfg(struct si_pub *sih); | ||||||
|  | 
 | ||||||
|  | extern void ai_chipcontrl_epa4331(struct si_pub *sih, bool on); | ||||||
|  | /* Enable Ex-PA for 4313 */ | ||||||
|  | extern void ai_epa_4313war(struct si_pub *sih); | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCM_AIUTILS_H_ */ | ||||||
							
								
								
									
										1241
									
								
								drivers/net/wireless/brcm80211/brcmsmac/ampdu.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1241
									
								
								drivers/net/wireless/brcm80211/brcmsmac/ampdu.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										30
									
								
								drivers/net/wireless/brcm80211/brcmsmac/ampdu.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								drivers/net/wireless/brcm80211/brcmsmac/ampdu.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,30 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCM_AMPDU_H_ | ||||||
|  | #define _BRCM_AMPDU_H_ | ||||||
|  | 
 | ||||||
|  | extern struct ampdu_info *brcms_c_ampdu_attach(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_ampdu_detach(struct ampdu_info *ampdu); | ||||||
|  | extern int brcms_c_sendampdu(struct ampdu_info *ampdu, | ||||||
|  | 			     struct brcms_txq_info *qi, | ||||||
|  | 			     struct sk_buff **aggp, int prec); | ||||||
|  | extern void brcms_c_ampdu_dotxstatus(struct ampdu_info *ampdu, struct scb *scb, | ||||||
|  | 				 struct sk_buff *p, struct tx_status *txs); | ||||||
|  | extern void brcms_c_ampdu_macaddr_upd(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_ampdu_shm_upd(struct ampdu_info *ampdu); | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCM_AMPDU_H_ */ | ||||||
							
								
								
									
										307
									
								
								drivers/net/wireless/brcm80211/brcmsmac/antsel.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										307
									
								
								drivers/net/wireless/brcm80211/brcmsmac/antsel.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,307 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <linux/slab.h> | ||||||
|  | #include <net/mac80211.h> | ||||||
|  | 
 | ||||||
|  | #include "types.h" | ||||||
|  | #include "main.h" | ||||||
|  | #include "phy_shim.h" | ||||||
|  | #include "antsel.h" | ||||||
|  | 
 | ||||||
|  | #define ANT_SELCFG_AUTO		0x80	/* bit indicates antenna sel AUTO */ | ||||||
|  | #define ANT_SELCFG_MASK		0x33	/* antenna configuration mask */ | ||||||
|  | #define ANT_SELCFG_TX_UNICAST	0	/* unicast tx antenna configuration */ | ||||||
|  | #define ANT_SELCFG_RX_UNICAST	1	/* unicast rx antenna configuration */ | ||||||
|  | #define ANT_SELCFG_TX_DEF	2	/* default tx antenna configuration */ | ||||||
|  | #define ANT_SELCFG_RX_DEF	3	/* default rx antenna configuration */ | ||||||
|  | 
 | ||||||
|  | /* useful macros */ | ||||||
|  | #define BRCMS_ANTSEL_11N_0(ant)	((((ant) & ANT_SELCFG_MASK) >> 4) & 0xf) | ||||||
|  | #define BRCMS_ANTSEL_11N_1(ant)	(((ant) & ANT_SELCFG_MASK) & 0xf) | ||||||
|  | #define BRCMS_ANTIDX_11N(ant)	(((BRCMS_ANTSEL_11N_0(ant)) << 2) +\ | ||||||
|  | 				(BRCMS_ANTSEL_11N_1(ant))) | ||||||
|  | #define BRCMS_ANT_ISAUTO_11N(ant) (((ant) & ANT_SELCFG_AUTO) == ANT_SELCFG_AUTO) | ||||||
|  | #define BRCMS_ANTSEL_11N(ant)	((ant) & ANT_SELCFG_MASK) | ||||||
|  | 
 | ||||||
|  | /* antenna switch */ | ||||||
|  | /* defines for no boardlevel antenna diversity */ | ||||||
|  | #define ANT_SELCFG_DEF_2x2	0x01	/* default antenna configuration */ | ||||||
|  | 
 | ||||||
|  | /* 2x3 antdiv defines and tables for GPIO communication */ | ||||||
|  | #define ANT_SELCFG_NUM_2x3	3 | ||||||
|  | #define ANT_SELCFG_DEF_2x3	0x01	/* default antenna configuration */ | ||||||
|  | 
 | ||||||
|  | /* 2x4 antdiv rev4 defines and tables for GPIO communication */ | ||||||
|  | #define ANT_SELCFG_NUM_2x4	4 | ||||||
|  | #define ANT_SELCFG_DEF_2x4	0x02	/* default antenna configuration */ | ||||||
|  | 
 | ||||||
|  | static const u16 mimo_2x4_div_antselpat_tbl[] = { | ||||||
|  | 	0, 0, 0x9, 0xa,		/* ant0: 0 ant1: 2,3 */ | ||||||
|  | 	0, 0, 0x5, 0x6,		/* ant0: 1 ant1: 2,3 */ | ||||||
|  | 	0, 0, 0, 0,		/* n.a.              */ | ||||||
|  | 	0, 0, 0, 0		/* n.a.              */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static const u8 mimo_2x4_div_antselid_tbl[16] = { | ||||||
|  | 	0, 0, 0, 0, 0, 2, 3, 0, | ||||||
|  | 	0, 0, 1, 0, 0, 0, 0, 0	/* pat to antselid */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static const u16 mimo_2x3_div_antselpat_tbl[] = { | ||||||
|  | 	16, 0, 1, 16,		/* ant0: 0 ant1: 1,2 */ | ||||||
|  | 	16, 16, 16, 16,		/* n.a.              */ | ||||||
|  | 	16, 2, 16, 16,		/* ant0: 2 ant1: 1   */ | ||||||
|  | 	16, 16, 16, 16		/* n.a.              */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static const u8 mimo_2x3_div_antselid_tbl[16] = { | ||||||
|  | 	0, 1, 2, 0, 0, 0, 0, 0, | ||||||
|  | 	0, 0, 0, 0, 0, 0, 0, 0	/* pat to antselid */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* boardlevel antenna selection: init antenna selection structure */ | ||||||
|  | static void | ||||||
|  | brcms_c_antsel_init_cfg(struct antsel_info *asi, struct brcms_antselcfg *antsel, | ||||||
|  | 		    bool auto_sel) | ||||||
|  | { | ||||||
|  | 	if (asi->antsel_type == ANTSEL_2x3) { | ||||||
|  | 		u8 antcfg_def = ANT_SELCFG_DEF_2x3 | | ||||||
|  | 		    ((asi->antsel_avail && auto_sel) ? ANT_SELCFG_AUTO : 0); | ||||||
|  | 		antsel->ant_config[ANT_SELCFG_TX_DEF] = antcfg_def; | ||||||
|  | 		antsel->ant_config[ANT_SELCFG_TX_UNICAST] = antcfg_def; | ||||||
|  | 		antsel->ant_config[ANT_SELCFG_RX_DEF] = antcfg_def; | ||||||
|  | 		antsel->ant_config[ANT_SELCFG_RX_UNICAST] = antcfg_def; | ||||||
|  | 		antsel->num_antcfg = ANT_SELCFG_NUM_2x3; | ||||||
|  | 
 | ||||||
|  | 	} else if (asi->antsel_type == ANTSEL_2x4) { | ||||||
|  | 
 | ||||||
|  | 		antsel->ant_config[ANT_SELCFG_TX_DEF] = ANT_SELCFG_DEF_2x4; | ||||||
|  | 		antsel->ant_config[ANT_SELCFG_TX_UNICAST] = ANT_SELCFG_DEF_2x4; | ||||||
|  | 		antsel->ant_config[ANT_SELCFG_RX_DEF] = ANT_SELCFG_DEF_2x4; | ||||||
|  | 		antsel->ant_config[ANT_SELCFG_RX_UNICAST] = ANT_SELCFG_DEF_2x4; | ||||||
|  | 		antsel->num_antcfg = ANT_SELCFG_NUM_2x4; | ||||||
|  | 
 | ||||||
|  | 	} else {		/* no antenna selection available */ | ||||||
|  | 
 | ||||||
|  | 		antsel->ant_config[ANT_SELCFG_TX_DEF] = ANT_SELCFG_DEF_2x2; | ||||||
|  | 		antsel->ant_config[ANT_SELCFG_TX_UNICAST] = ANT_SELCFG_DEF_2x2; | ||||||
|  | 		antsel->ant_config[ANT_SELCFG_RX_DEF] = ANT_SELCFG_DEF_2x2; | ||||||
|  | 		antsel->ant_config[ANT_SELCFG_RX_UNICAST] = ANT_SELCFG_DEF_2x2; | ||||||
|  | 		antsel->num_antcfg = 0; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct antsel_info *brcms_c_antsel_attach(struct brcms_c_info *wlc) | ||||||
|  | { | ||||||
|  | 	struct antsel_info *asi; | ||||||
|  | 	struct si_pub *sih = wlc->hw->sih; | ||||||
|  | 
 | ||||||
|  | 	asi = kzalloc(sizeof(struct antsel_info), GFP_ATOMIC); | ||||||
|  | 	if (!asi) | ||||||
|  | 		return NULL; | ||||||
|  | 
 | ||||||
|  | 	asi->wlc = wlc; | ||||||
|  | 	asi->pub = wlc->pub; | ||||||
|  | 	asi->antsel_type = ANTSEL_NA; | ||||||
|  | 	asi->antsel_avail = false; | ||||||
|  | 	asi->antsel_antswitch = (u8) getintvar(sih, BRCMS_SROM_ANTSWITCH); | ||||||
|  | 
 | ||||||
|  | 	if ((asi->pub->sromrev >= 4) && (asi->antsel_antswitch != 0)) { | ||||||
|  | 		switch (asi->antsel_antswitch) { | ||||||
|  | 		case ANTSWITCH_TYPE_1: | ||||||
|  | 		case ANTSWITCH_TYPE_2: | ||||||
|  | 		case ANTSWITCH_TYPE_3: | ||||||
|  | 			/* 4321/2 board with 2x3 switch logic */ | ||||||
|  | 			asi->antsel_type = ANTSEL_2x3; | ||||||
|  | 			/* Antenna selection availability */ | ||||||
|  | 			if (((u16) getintvar(sih, BRCMS_SROM_AA2G) == 7) || | ||||||
|  | 			    ((u16) getintvar(sih, BRCMS_SROM_AA5G) == 7)) { | ||||||
|  | 				asi->antsel_avail = true; | ||||||
|  | 			} else if ( | ||||||
|  | 				(u16) getintvar(sih, BRCMS_SROM_AA2G) == 3 || | ||||||
|  | 				(u16) getintvar(sih, BRCMS_SROM_AA5G) == 3) { | ||||||
|  | 				asi->antsel_avail = false; | ||||||
|  | 			} else { | ||||||
|  | 				asi->antsel_avail = false; | ||||||
|  | 				wiphy_err(wlc->wiphy, "antsel_attach: 2o3 " | ||||||
|  | 					  "board cfg invalid\n"); | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			break; | ||||||
|  | 		default: | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 	} else if ((asi->pub->sromrev == 4) && | ||||||
|  | 		   ((u16) getintvar(sih, BRCMS_SROM_AA2G) == 7) && | ||||||
|  | 		   ((u16) getintvar(sih, BRCMS_SROM_AA5G) == 0)) { | ||||||
|  | 		/* hack to match old 4321CB2 cards with 2of3 antenna switch */ | ||||||
|  | 		asi->antsel_type = ANTSEL_2x3; | ||||||
|  | 		asi->antsel_avail = true; | ||||||
|  | 	} else if (asi->pub->boardflags2 & BFL2_2X4_DIV) { | ||||||
|  | 		asi->antsel_type = ANTSEL_2x4; | ||||||
|  | 		asi->antsel_avail = true; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Set the antenna selection type for the low driver */ | ||||||
|  | 	brcms_b_antsel_type_set(wlc->hw, asi->antsel_type); | ||||||
|  | 
 | ||||||
|  | 	/* Init (auto/manual) antenna selection */ | ||||||
|  | 	brcms_c_antsel_init_cfg(asi, &asi->antcfg_11n, true); | ||||||
|  | 	brcms_c_antsel_init_cfg(asi, &asi->antcfg_cur, true); | ||||||
|  | 
 | ||||||
|  | 	return asi; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void brcms_c_antsel_detach(struct antsel_info *asi) | ||||||
|  | { | ||||||
|  | 	kfree(asi); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * boardlevel antenna selection: | ||||||
|  |  *   convert ant_cfg to mimo_antsel (ucode interface) | ||||||
|  |  */ | ||||||
|  | static u16 brcms_c_antsel_antcfg2antsel(struct antsel_info *asi, u8 ant_cfg) | ||||||
|  | { | ||||||
|  | 	u8 idx = BRCMS_ANTIDX_11N(BRCMS_ANTSEL_11N(ant_cfg)); | ||||||
|  | 	u16 mimo_antsel = 0; | ||||||
|  | 
 | ||||||
|  | 	if (asi->antsel_type == ANTSEL_2x4) { | ||||||
|  | 		/* 2x4 antenna diversity board, 4 cfgs: 0-2 0-3 1-2 1-3 */ | ||||||
|  | 		mimo_antsel = (mimo_2x4_div_antselpat_tbl[idx] & 0xf); | ||||||
|  | 		return mimo_antsel; | ||||||
|  | 
 | ||||||
|  | 	} else if (asi->antsel_type == ANTSEL_2x3) { | ||||||
|  | 		/* 2x3 antenna selection, 3 cfgs: 0-1 0-2 2-1 */ | ||||||
|  | 		mimo_antsel = (mimo_2x3_div_antselpat_tbl[idx] & 0xf); | ||||||
|  | 		return mimo_antsel; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return mimo_antsel; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* boardlevel antenna selection: ucode interface control */ | ||||||
|  | static int brcms_c_antsel_cfgupd(struct antsel_info *asi, | ||||||
|  | 				 struct brcms_antselcfg *antsel) | ||||||
|  | { | ||||||
|  | 	struct brcms_c_info *wlc = asi->wlc; | ||||||
|  | 	u8 ant_cfg; | ||||||
|  | 	u16 mimo_antsel; | ||||||
|  | 
 | ||||||
|  | 	/* 1) Update TX antconfig for all frames that are not unicast data
 | ||||||
|  | 	 *    (aka default TX) | ||||||
|  | 	 */ | ||||||
|  | 	ant_cfg = antsel->ant_config[ANT_SELCFG_TX_DEF]; | ||||||
|  | 	mimo_antsel = brcms_c_antsel_antcfg2antsel(asi, ant_cfg); | ||||||
|  | 	brcms_b_write_shm(wlc->hw, M_MIMO_ANTSEL_TXDFLT, mimo_antsel); | ||||||
|  | 	/*
 | ||||||
|  | 	 * Update driver stats for currently selected | ||||||
|  | 	 * default tx/rx antenna config | ||||||
|  | 	 */ | ||||||
|  | 	asi->antcfg_cur.ant_config[ANT_SELCFG_TX_DEF] = ant_cfg; | ||||||
|  | 
 | ||||||
|  | 	/* 2) Update RX antconfig for all frames that are not unicast data
 | ||||||
|  | 	 *    (aka default RX) | ||||||
|  | 	 */ | ||||||
|  | 	ant_cfg = antsel->ant_config[ANT_SELCFG_RX_DEF]; | ||||||
|  | 	mimo_antsel = brcms_c_antsel_antcfg2antsel(asi, ant_cfg); | ||||||
|  | 	brcms_b_write_shm(wlc->hw, M_MIMO_ANTSEL_RXDFLT, mimo_antsel); | ||||||
|  | 	/*
 | ||||||
|  | 	 * Update driver stats for currently selected | ||||||
|  | 	 * default tx/rx antenna config | ||||||
|  | 	 */ | ||||||
|  | 	asi->antcfg_cur.ant_config[ANT_SELCFG_RX_DEF] = ant_cfg; | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void brcms_c_antsel_init(struct antsel_info *asi) | ||||||
|  | { | ||||||
|  | 	if ((asi->antsel_type == ANTSEL_2x3) || | ||||||
|  | 	    (asi->antsel_type == ANTSEL_2x4)) | ||||||
|  | 		brcms_c_antsel_cfgupd(asi, &asi->antcfg_11n); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* boardlevel antenna selection: convert id to ant_cfg */ | ||||||
|  | static u8 brcms_c_antsel_id2antcfg(struct antsel_info *asi, u8 id) | ||||||
|  | { | ||||||
|  | 	u8 antcfg = ANT_SELCFG_DEF_2x2; | ||||||
|  | 
 | ||||||
|  | 	if (asi->antsel_type == ANTSEL_2x4) { | ||||||
|  | 		/* 2x4 antenna diversity board, 4 cfgs: 0-2 0-3 1-2 1-3 */ | ||||||
|  | 		antcfg = (((id & 0x2) << 3) | ((id & 0x1) + 2)); | ||||||
|  | 		return antcfg; | ||||||
|  | 
 | ||||||
|  | 	} else if (asi->antsel_type == ANTSEL_2x3) { | ||||||
|  | 		/* 2x3 antenna selection, 3 cfgs: 0-1 0-2 2-1 */ | ||||||
|  | 		antcfg = (((id & 0x02) << 4) | ((id & 0x1) + 1)); | ||||||
|  | 		return antcfg; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return antcfg; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | brcms_c_antsel_antcfg_get(struct antsel_info *asi, bool usedef, bool sel, | ||||||
|  | 		      u8 antselid, u8 fbantselid, u8 *antcfg, | ||||||
|  | 		      u8 *fbantcfg) | ||||||
|  | { | ||||||
|  | 	u8 ant; | ||||||
|  | 
 | ||||||
|  | 	/* if use default, assign it and return */ | ||||||
|  | 	if (usedef) { | ||||||
|  | 		*antcfg = asi->antcfg_11n.ant_config[ANT_SELCFG_TX_DEF]; | ||||||
|  | 		*fbantcfg = *antcfg; | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (!sel) { | ||||||
|  | 		*antcfg = asi->antcfg_11n.ant_config[ANT_SELCFG_TX_UNICAST]; | ||||||
|  | 		*fbantcfg = *antcfg; | ||||||
|  | 
 | ||||||
|  | 	} else { | ||||||
|  | 		ant = asi->antcfg_11n.ant_config[ANT_SELCFG_TX_UNICAST]; | ||||||
|  | 		if ((ant & ANT_SELCFG_AUTO) == ANT_SELCFG_AUTO) { | ||||||
|  | 			*antcfg = brcms_c_antsel_id2antcfg(asi, antselid); | ||||||
|  | 			*fbantcfg = brcms_c_antsel_id2antcfg(asi, fbantselid); | ||||||
|  | 		} else { | ||||||
|  | 			*antcfg = | ||||||
|  | 			    asi->antcfg_11n.ant_config[ANT_SELCFG_TX_UNICAST]; | ||||||
|  | 			*fbantcfg = *antcfg; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* boardlevel antenna selection: convert mimo_antsel (ucode interface) to id */ | ||||||
|  | u8 brcms_c_antsel_antsel2id(struct antsel_info *asi, u16 antsel) | ||||||
|  | { | ||||||
|  | 	u8 antselid = 0; | ||||||
|  | 
 | ||||||
|  | 	if (asi->antsel_type == ANTSEL_2x4) { | ||||||
|  | 		/* 2x4 antenna diversity board, 4 cfgs: 0-2 0-3 1-2 1-3 */ | ||||||
|  | 		antselid = mimo_2x4_div_antselid_tbl[(antsel & 0xf)]; | ||||||
|  | 		return antselid; | ||||||
|  | 
 | ||||||
|  | 	} else if (asi->antsel_type == ANTSEL_2x3) { | ||||||
|  | 		/* 2x3 antenna selection, 3 cfgs: 0-1 0-2 2-1 */ | ||||||
|  | 		antselid = mimo_2x3_div_antselid_tbl[(antsel & 0xf)]; | ||||||
|  | 		return antselid; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return antselid; | ||||||
|  | } | ||||||
							
								
								
									
										29
									
								
								drivers/net/wireless/brcm80211/brcmsmac/antsel.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								drivers/net/wireless/brcm80211/brcmsmac/antsel.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,29 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCM_ANTSEL_H_ | ||||||
|  | #define _BRCM_ANTSEL_H_ | ||||||
|  | 
 | ||||||
|  | extern struct antsel_info *brcms_c_antsel_attach(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_antsel_detach(struct antsel_info *asi); | ||||||
|  | extern void brcms_c_antsel_init(struct antsel_info *asi); | ||||||
|  | extern void brcms_c_antsel_antcfg_get(struct antsel_info *asi, bool usedef, | ||||||
|  | 				  bool sel, | ||||||
|  | 				  u8 id, u8 fbid, u8 *antcfg, | ||||||
|  | 				  u8 *fbantcfg); | ||||||
|  | extern u8 brcms_c_antsel_antsel2id(struct antsel_info *asi, u16 antsel); | ||||||
|  | 
 | ||||||
|  | #endif /* _BRCM_ANTSEL_H_ */ | ||||||
							
								
								
									
										23
									
								
								drivers/net/wireless/brcm80211/brcmsmac/brcms_trace_events.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								drivers/net/wireless/brcm80211/brcmsmac/brcms_trace_events.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,23 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2011 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <linux/module.h> /* bug in tracepoint.h, it should include this */ | ||||||
|  | 
 | ||||||
|  | #ifndef __CHECKER__ | ||||||
|  | #include "mac80211_if.h" | ||||||
|  | #define CREATE_TRACE_POINTS | ||||||
|  | #include "brcms_trace_events.h" | ||||||
|  | #endif | ||||||
							
								
								
									
										92
									
								
								drivers/net/wireless/brcm80211/brcmsmac/brcms_trace_events.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										92
									
								
								drivers/net/wireless/brcm80211/brcmsmac/brcms_trace_events.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,92 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2011 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #undef TRACE_SYSTEM | ||||||
|  | #define TRACE_SYSTEM brcmsmac | ||||||
|  | 
 | ||||||
|  | #if !defined(__TRACE_BRCMSMAC_H) || defined(TRACE_HEADER_MULTI_READ) | ||||||
|  | 
 | ||||||
|  | #define __TRACE_BRCMSMAC_H | ||||||
|  | 
 | ||||||
|  | #include <linux/tracepoint.h> | ||||||
|  | #include "mac80211_if.h" | ||||||
|  | 
 | ||||||
|  | #ifndef CONFIG_BRCMDBG | ||||||
|  | #undef TRACE_EVENT | ||||||
|  | #define TRACE_EVENT(name, proto, ...) \ | ||||||
|  | static inline void trace_ ## name(proto) {} | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * We define a tracepoint, its arguments, its printk format and its | ||||||
|  |  * 'fast binary record' layout. | ||||||
|  |  */ | ||||||
|  | TRACE_EVENT(brcms_timer, | ||||||
|  | 	/* TPPROTO is the prototype of the function called by this tracepoint */ | ||||||
|  | 	TP_PROTO(struct brcms_timer *t), | ||||||
|  | 	/*
 | ||||||
|  | 	 * TPARGS(firstarg, p) are the parameters names, same as found in the | ||||||
|  | 	 * prototype. | ||||||
|  | 	 */ | ||||||
|  | 	TP_ARGS(t), | ||||||
|  | 	/*
 | ||||||
|  | 	 * Fast binary tracing: define the trace record via TP_STRUCT__entry(). | ||||||
|  | 	 * You can think about it like a regular C structure local variable | ||||||
|  | 	 * definition. | ||||||
|  | 	 */ | ||||||
|  | 	TP_STRUCT__entry( | ||||||
|  | 		__field(uint, ms) | ||||||
|  | 		__field(uint, set) | ||||||
|  | 		__field(uint, periodic) | ||||||
|  | 	), | ||||||
|  | 	TP_fast_assign( | ||||||
|  | 		__entry->ms = t->ms; | ||||||
|  | 		__entry->set = t->set; | ||||||
|  | 		__entry->periodic = t->periodic; | ||||||
|  | 	), | ||||||
|  | 	TP_printk( | ||||||
|  | 		"ms=%u set=%u periodic=%u", | ||||||
|  | 		__entry->ms, __entry->set, __entry->periodic | ||||||
|  | 	) | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | TRACE_EVENT(brcms_dpc, | ||||||
|  | 	TP_PROTO(unsigned long data), | ||||||
|  | 	TP_ARGS(data), | ||||||
|  | 	TP_STRUCT__entry( | ||||||
|  | 		__field(unsigned long, data) | ||||||
|  | 	), | ||||||
|  | 	TP_fast_assign( | ||||||
|  | 		__entry->data = data; | ||||||
|  | 	), | ||||||
|  | 	TP_printk( | ||||||
|  | 		"data=%p", | ||||||
|  | 		(void *)__entry->data | ||||||
|  | 	) | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | #endif /* __TRACE_BRCMSMAC_H */ | ||||||
|  | 
 | ||||||
|  | #ifdef CONFIG_BRCMDBG | ||||||
|  | 
 | ||||||
|  | #undef TRACE_INCLUDE_PATH | ||||||
|  | #define TRACE_INCLUDE_PATH . | ||||||
|  | #undef TRACE_INCLUDE_FILE | ||||||
|  | #define TRACE_INCLUDE_FILE brcms_trace_events | ||||||
|  | 
 | ||||||
|  | #include <trace/define_trace.h> | ||||||
|  | 
 | ||||||
|  | #endif /* CONFIG_BRCMDBG */ | ||||||
							
								
								
									
										1565
									
								
								drivers/net/wireless/brcm80211/brcmsmac/channel.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1565
									
								
								drivers/net/wireless/brcm80211/brcmsmac/channel.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										53
									
								
								drivers/net/wireless/brcm80211/brcmsmac/channel.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								drivers/net/wireless/brcm80211/brcmsmac/channel.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,53 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCM_CHANNEL_H_ | ||||||
|  | #define _BRCM_CHANNEL_H_ | ||||||
|  | 
 | ||||||
|  | /* conversion for phy txpwr calculations that use .25 dB units */ | ||||||
|  | #define BRCMS_TXPWR_DB_FACTOR 4 | ||||||
|  | 
 | ||||||
|  | /* bits for locale_info flags */ | ||||||
|  | #define BRCMS_PEAK_CONDUCTED	0x00	/* Peak for locals */ | ||||||
|  | #define BRCMS_EIRP		0x01	/* Flag for EIRP */ | ||||||
|  | #define BRCMS_DFS_TPC		0x02	/* Flag for DFS TPC */ | ||||||
|  | #define BRCMS_NO_OFDM		0x04	/* Flag for No OFDM */ | ||||||
|  | #define BRCMS_NO_40MHZ		0x08	/* Flag for No MIMO 40MHz */ | ||||||
|  | #define BRCMS_NO_MIMO		0x10	/* Flag for No MIMO, 20 or 40 MHz */ | ||||||
|  | #define BRCMS_RADAR_TYPE_EU       0x20	/* Flag for EU */ | ||||||
|  | #define BRCMS_DFS_FCC             BRCMS_DFS_TPC	/* Flag for DFS FCC */ | ||||||
|  | 
 | ||||||
|  | #define BRCMS_DFS_EU (BRCMS_DFS_TPC | BRCMS_RADAR_TYPE_EU) /* Flag for DFS EU */ | ||||||
|  | 
 | ||||||
|  | extern struct brcms_cm_info * | ||||||
|  | brcms_c_channel_mgr_attach(struct brcms_c_info *wlc); | ||||||
|  | 
 | ||||||
|  | extern void brcms_c_channel_mgr_detach(struct brcms_cm_info *wlc_cm); | ||||||
|  | 
 | ||||||
|  | extern u8 brcms_c_channel_locale_flags_in_band(struct brcms_cm_info *wlc_cm, | ||||||
|  | 					   uint bandunit); | ||||||
|  | 
 | ||||||
|  | extern bool brcms_c_valid_chanspec_db(struct brcms_cm_info *wlc_cm, | ||||||
|  | 				      u16 chspec); | ||||||
|  | 
 | ||||||
|  | extern void brcms_c_channel_reg_limits(struct brcms_cm_info *wlc_cm, | ||||||
|  | 				   u16 chanspec, | ||||||
|  | 				   struct txpwr_limits *txpwr); | ||||||
|  | extern void brcms_c_channel_set_chanspec(struct brcms_cm_info *wlc_cm, | ||||||
|  | 				     u16 chanspec, | ||||||
|  | 				     u8 local_constraint_qdbm); | ||||||
|  | 
 | ||||||
|  | #endif				/* _WLC_CHANNEL_H */ | ||||||
							
								
								
									
										1898
									
								
								drivers/net/wireless/brcm80211/brcmsmac/d11.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1898
									
								
								drivers/net/wireless/brcm80211/brcmsmac/d11.h
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										1425
									
								
								drivers/net/wireless/brcm80211/brcmsmac/dma.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1425
									
								
								drivers/net/wireless/brcm80211/brcmsmac/dma.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										120
									
								
								drivers/net/wireless/brcm80211/brcmsmac/dma.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										120
									
								
								drivers/net/wireless/brcm80211/brcmsmac/dma.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,120 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef	_BRCM_DMA_H_ | ||||||
|  | #define	_BRCM_DMA_H_ | ||||||
|  | 
 | ||||||
|  | #include <linux/delay.h> | ||||||
|  | #include "types.h"		/* forward structure declarations */ | ||||||
|  | 
 | ||||||
|  | /* map/unmap direction */ | ||||||
|  | #define	DMA_TX	1		/* TX direction for DMA */ | ||||||
|  | #define	DMA_RX	2		/* RX direction for DMA */ | ||||||
|  | 
 | ||||||
|  | /* DMA structure:
 | ||||||
|  |  *  support two DMA engines: 32 bits address or 64 bit addressing | ||||||
|  |  *  basic DMA register set is per channel(transmit or receive) | ||||||
|  |  *  a pair of channels is defined for convenience | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* 32 bits addressing */ | ||||||
|  | 
 | ||||||
|  | struct dma32diag {	/* diag access */ | ||||||
|  | 	u32 fifoaddr;	/* diag address */ | ||||||
|  | 	u32 fifodatalow;	/* low 32bits of data */ | ||||||
|  | 	u32 fifodatahigh;	/* high 32bits of data */ | ||||||
|  | 	u32 pad;		/* reserved */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* 64 bits addressing */ | ||||||
|  | 
 | ||||||
|  | /* dma registers per channel(xmt or rcv) */ | ||||||
|  | struct dma64regs { | ||||||
|  | 	u32 control;	/* enable, et al */ | ||||||
|  | 	u32 ptr;	/* last descriptor posted to chip */ | ||||||
|  | 	u32 addrlow;	/* desc ring base address low 32-bits (8K aligned) */ | ||||||
|  | 	u32 addrhigh;	/* desc ring base address bits 63:32 (8K aligned) */ | ||||||
|  | 	u32 status0;	/* current descriptor, xmt state */ | ||||||
|  | 	u32 status1;	/* active descriptor, xmt error */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* range param for dma_getnexttxp() and dma_txreclaim */ | ||||||
|  | enum txd_range { | ||||||
|  | 	DMA_RANGE_ALL = 1, | ||||||
|  | 	DMA_RANGE_TRANSMITTED, | ||||||
|  | 	DMA_RANGE_TRANSFERED | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Exported data structure (read-only) | ||||||
|  |  */ | ||||||
|  | /* export structure */ | ||||||
|  | struct dma_pub { | ||||||
|  | 	uint txavail;		/* # free tx descriptors */ | ||||||
|  | 	uint dmactrlflags;	/* dma control flags */ | ||||||
|  | 
 | ||||||
|  | 	/* rx error counters */ | ||||||
|  | 	uint rxgiants;		/* rx giant frames */ | ||||||
|  | 	uint rxnobuf;		/* rx out of dma descriptors */ | ||||||
|  | 	/* tx error counters */ | ||||||
|  | 	uint txnobuf;		/* tx out of dma descriptors */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | extern struct dma_pub *dma_attach(char *name, struct si_pub *sih, | ||||||
|  | 			    void __iomem *dmaregstx, void __iomem *dmaregsrx, | ||||||
|  | 			    uint ntxd, uint nrxd, | ||||||
|  | 			    uint rxbufsize, int rxextheadroom, | ||||||
|  | 			    uint nrxpost, uint rxoffset, uint *msg_level); | ||||||
|  | 
 | ||||||
|  | void dma_rxinit(struct dma_pub *pub); | ||||||
|  | struct sk_buff *dma_rx(struct dma_pub *pub); | ||||||
|  | bool dma_rxfill(struct dma_pub *pub); | ||||||
|  | bool dma_rxreset(struct dma_pub *pub); | ||||||
|  | bool dma_txreset(struct dma_pub *pub); | ||||||
|  | void dma_txinit(struct dma_pub *pub); | ||||||
|  | int dma_txfast(struct dma_pub *pub, struct sk_buff *p0, bool commit); | ||||||
|  | void dma_txsuspend(struct dma_pub *pub); | ||||||
|  | bool dma_txsuspended(struct dma_pub *pub); | ||||||
|  | void dma_txresume(struct dma_pub *pub); | ||||||
|  | void dma_txreclaim(struct dma_pub *pub, enum txd_range range); | ||||||
|  | void dma_rxreclaim(struct dma_pub *pub); | ||||||
|  | void dma_detach(struct dma_pub *pub); | ||||||
|  | unsigned long dma_getvar(struct dma_pub *pub, const char *name); | ||||||
|  | struct sk_buff *dma_getnexttxp(struct dma_pub *pub, enum txd_range range); | ||||||
|  | void dma_counterreset(struct dma_pub *pub); | ||||||
|  | 
 | ||||||
|  | void dma_walk_packets(struct dma_pub *dmah, void (*callback_fnc) | ||||||
|  | 		      (void *pkt, void *arg_a), void *arg_a); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * DMA(Bug) on bcm47xx chips seems to declare that the packet is ready, but | ||||||
|  |  * the packet length is not updated yet (by DMA) on the expected time. | ||||||
|  |  * Workaround is to hold processor till DMA updates the length, and stay off | ||||||
|  |  * the bus to allow DMA update the length in buffer | ||||||
|  |  */ | ||||||
|  | static inline void dma_spin_for_len(uint len, struct sk_buff *head) | ||||||
|  | { | ||||||
|  | #if defined(CONFIG_BCM47XX) | ||||||
|  | 	if (!len) { | ||||||
|  | 		while (!(len = *(u16 *) KSEG1ADDR(head->data))) | ||||||
|  | 			udelay(1); | ||||||
|  | 
 | ||||||
|  | 		*(u16 *) (head->data) = cpu_to_le16((u16) len); | ||||||
|  | 	} | ||||||
|  | #endif				/* defined(CONFIG_BCM47XX) */ | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCM_DMA_H_ */ | ||||||
							
								
								
									
										1701
									
								
								drivers/net/wireless/brcm80211/brcmsmac/mac80211_if.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1701
									
								
								drivers/net/wireless/brcm80211/brcmsmac/mac80211_if.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										107
									
								
								drivers/net/wireless/brcm80211/brcmsmac/mac80211_if.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										107
									
								
								drivers/net/wireless/brcm80211/brcmsmac/mac80211_if.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,107 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCM_MAC80211_IF_H_ | ||||||
|  | #define _BRCM_MAC80211_IF_H_ | ||||||
|  | 
 | ||||||
|  | #include <linux/timer.h> | ||||||
|  | #include <linux/interrupt.h> | ||||||
|  | #include "ucode_loader.h" | ||||||
|  | /*
 | ||||||
|  |  * Starting index for 5G rates in the | ||||||
|  |  * legacy rate table. | ||||||
|  |  */ | ||||||
|  | #define BRCMS_LEGACY_5G_RATE_OFFSET	4 | ||||||
|  | 
 | ||||||
|  | /* softmac ioctl definitions */ | ||||||
|  | #define BRCMS_SET_SHORTSLOT_OVERRIDE		146 | ||||||
|  | 
 | ||||||
|  | struct brcms_timer { | ||||||
|  | 	struct timer_list timer; | ||||||
|  | 	struct brcms_info *wl; | ||||||
|  | 	void (*fn) (void *); | ||||||
|  | 	void *arg;		/* argument to fn */ | ||||||
|  | 	uint ms; | ||||||
|  | 	bool periodic; | ||||||
|  | 	bool set; | ||||||
|  | 	struct brcms_timer *next; | ||||||
|  | #ifdef BCMDBG | ||||||
|  | 	char *name;		/* Description of the timer */ | ||||||
|  | #endif | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct brcms_if { | ||||||
|  | 	uint subunit;		/* WDS/BSS unit */ | ||||||
|  | 	struct pci_dev *pci_dev; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #define MAX_FW_IMAGES		4 | ||||||
|  | struct brcms_firmware { | ||||||
|  | 	u32 fw_cnt; | ||||||
|  | 	const struct firmware *fw_bin[MAX_FW_IMAGES]; | ||||||
|  | 	const struct firmware *fw_hdr[MAX_FW_IMAGES]; | ||||||
|  | 	u32 hdr_num_entries[MAX_FW_IMAGES]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct brcms_info { | ||||||
|  | 	struct brcms_pub *pub;		/* pointer to public wlc state */ | ||||||
|  | 	struct brcms_c_info *wlc;	/* pointer to private common data */ | ||||||
|  | 	u32 magic; | ||||||
|  | 
 | ||||||
|  | 	int irq; | ||||||
|  | 
 | ||||||
|  | 	spinlock_t lock;	/* per-device perimeter lock */ | ||||||
|  | 	spinlock_t isr_lock;	/* per-device ISR synchronization lock */ | ||||||
|  | 
 | ||||||
|  | 	/* regsva for unmap in brcms_free() */ | ||||||
|  | 	void __iomem *regsva;	/* opaque chip registers virtual address */ | ||||||
|  | 
 | ||||||
|  | 	/* timer related fields */ | ||||||
|  | 	atomic_t callbacks;	/* # outstanding callback functions */ | ||||||
|  | 	struct brcms_timer *timers;	/* timer cleanup queue */ | ||||||
|  | 
 | ||||||
|  | 	struct tasklet_struct tasklet;	/* dpc tasklet */ | ||||||
|  | 	bool resched;		/* dpc needs to be and is rescheduled */ | ||||||
|  | 	struct brcms_firmware fw; | ||||||
|  | 	struct wiphy *wiphy; | ||||||
|  | 	struct brcms_ucode ucode; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* misc callbacks */ | ||||||
|  | extern void brcms_init(struct brcms_info *wl); | ||||||
|  | extern uint brcms_reset(struct brcms_info *wl); | ||||||
|  | extern void brcms_intrson(struct brcms_info *wl); | ||||||
|  | extern u32 brcms_intrsoff(struct brcms_info *wl); | ||||||
|  | extern void brcms_intrsrestore(struct brcms_info *wl, u32 macintmask); | ||||||
|  | extern int brcms_up(struct brcms_info *wl); | ||||||
|  | extern void brcms_down(struct brcms_info *wl); | ||||||
|  | extern void brcms_txflowcontrol(struct brcms_info *wl, struct brcms_if *wlif, | ||||||
|  | 				bool state, int prio); | ||||||
|  | extern bool brcms_rfkill_set_hw_state(struct brcms_info *wl); | ||||||
|  | 
 | ||||||
|  | /* timer functions */ | ||||||
|  | extern struct brcms_timer *brcms_init_timer(struct brcms_info *wl, | ||||||
|  | 				      void (*fn) (void *arg), void *arg, | ||||||
|  | 				      const char *name); | ||||||
|  | extern void brcms_free_timer(struct brcms_info *wl, struct brcms_timer *timer); | ||||||
|  | extern void brcms_add_timer(struct brcms_info *wl, struct brcms_timer *timer, | ||||||
|  | 			    uint ms, int periodic); | ||||||
|  | extern bool brcms_del_timer(struct brcms_info *wl, struct brcms_timer *timer); | ||||||
|  | extern void brcms_msleep(struct brcms_info *wl, uint ms); | ||||||
|  | extern void brcms_dpc(unsigned long data); | ||||||
|  | extern void brcms_timer(struct brcms_timer *t); | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCM_MAC80211_IF_H_ */ | ||||||
							
								
								
									
										8841
									
								
								drivers/net/wireless/brcm80211/brcmsmac/main.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8841
									
								
								drivers/net/wireless/brcm80211/brcmsmac/main.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										819
									
								
								drivers/net/wireless/brcm80211/brcmsmac/main.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										819
									
								
								drivers/net/wireless/brcm80211/brcmsmac/main.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,819 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCM_MAIN_H_ | ||||||
|  | #define _BRCM_MAIN_H_ | ||||||
|  | 
 | ||||||
|  | #include <linux/etherdevice.h> | ||||||
|  | 
 | ||||||
|  | #include <brcmu_utils.h> | ||||||
|  | #include "types.h" | ||||||
|  | #include "d11.h" | ||||||
|  | #include "scb.h" | ||||||
|  | 
 | ||||||
|  | #define	INVCHANNEL		255	/* invalid channel */ | ||||||
|  | 
 | ||||||
|  | /* max # brcms_c_module_register() calls */ | ||||||
|  | #define BRCMS_MAXMODULES	22 | ||||||
|  | 
 | ||||||
|  | #define SEQNUM_SHIFT		4 | ||||||
|  | #define SEQNUM_MAX		0x1000 | ||||||
|  | 
 | ||||||
|  | #define NTXRATE			64	/* # tx MPDUs rate is reported for */ | ||||||
|  | 
 | ||||||
|  | /* Maximum wait time for a MAC suspend */ | ||||||
|  | /* uS: 83mS is max packet time (64KB ampdu @ 6Mbps) */ | ||||||
|  | #define	BRCMS_MAX_MAC_SUSPEND	83000 | ||||||
|  | 
 | ||||||
|  | /* responses for probe requests older that this are tossed, zero to disable */ | ||||||
|  | #define BRCMS_PRB_RESP_TIMEOUT	0	/* Disable probe response timeout */ | ||||||
|  | 
 | ||||||
|  | /* transmit buffer max headroom for protocol headers */ | ||||||
|  | #define TXOFF (D11_TXH_LEN + D11_PHY_HDR_LEN) | ||||||
|  | 
 | ||||||
|  | #define AC_COUNT		4 | ||||||
|  | 
 | ||||||
|  | /* Macros for doing definition and get/set of bitfields
 | ||||||
|  |  * Usage example, e.g. a three-bit field (bits 4-6): | ||||||
|  |  *    #define <NAME>_M	BITFIELD_MASK(3) | ||||||
|  |  *    #define <NAME>_S	4 | ||||||
|  |  * ... | ||||||
|  |  *    regval = R_REG(osh, ®s->regfoo); | ||||||
|  |  *    field = GFIELD(regval, <NAME>); | ||||||
|  |  *    regval = SFIELD(regval, <NAME>, 1); | ||||||
|  |  *    W_REG(osh, ®s->regfoo, regval); | ||||||
|  |  */ | ||||||
|  | #define BITFIELD_MASK(width) \ | ||||||
|  | 		(((unsigned)1 << (width)) - 1) | ||||||
|  | #define GFIELD(val, field) \ | ||||||
|  | 		(((val) >> field ## _S) & field ## _M) | ||||||
|  | #define SFIELD(val, field, bits) \ | ||||||
|  | 		(((val) & (~(field ## _M << field ## _S))) | \ | ||||||
|  | 		 ((unsigned)(bits) << field ## _S)) | ||||||
|  | 
 | ||||||
|  | #define	SW_TIMER_MAC_STAT_UPD		30	/* periodic MAC stats update */ | ||||||
|  | 
 | ||||||
|  | /* max # supported core revisions (0 .. MAXCOREREV - 1) */ | ||||||
|  | #define	MAXCOREREV		28 | ||||||
|  | 
 | ||||||
|  | /* Double check that unsupported cores are not enabled */ | ||||||
|  | #if CONF_MSK(D11CONF, 0x4f) || CONF_GE(D11CONF, MAXCOREREV) | ||||||
|  | #error "Configuration for D11CONF includes unsupported versions." | ||||||
|  | #endif				/* Bad versions */ | ||||||
|  | 
 | ||||||
|  | /* values for shortslot_override */ | ||||||
|  | #define BRCMS_SHORTSLOT_AUTO	-1 /* Driver will manage Shortslot setting */ | ||||||
|  | #define BRCMS_SHORTSLOT_OFF	0  /* Turn off short slot */ | ||||||
|  | #define BRCMS_SHORTSLOT_ON	1  /* Turn on short slot */ | ||||||
|  | 
 | ||||||
|  | /* value for short/long and mixmode/greenfield preamble */ | ||||||
|  | #define BRCMS_LONG_PREAMBLE	(0) | ||||||
|  | #define BRCMS_SHORT_PREAMBLE	(1 << 0) | ||||||
|  | #define BRCMS_GF_PREAMBLE		(1 << 1) | ||||||
|  | #define BRCMS_MM_PREAMBLE		(1 << 2) | ||||||
|  | #define BRCMS_IS_MIMO_PREAMBLE(_pre) (((_pre) == BRCMS_GF_PREAMBLE) || \ | ||||||
|  | 				      ((_pre) == BRCMS_MM_PREAMBLE)) | ||||||
|  | 
 | ||||||
|  | /* TxFrameID */ | ||||||
|  | /* seq and frag bits: SEQNUM_SHIFT, FRAGNUM_MASK (802.11.h) */ | ||||||
|  | /* rate epoch bits: TXFID_RATE_SHIFT, TXFID_RATE_MASK ((wlc_rate.c) */ | ||||||
|  | #define TXFID_QUEUE_MASK	0x0007	/* Bits 0-2 */ | ||||||
|  | #define TXFID_SEQ_MASK		0x7FE0	/* Bits 5-15 */ | ||||||
|  | #define TXFID_SEQ_SHIFT		5	/* Number of bit shifts */ | ||||||
|  | #define	TXFID_RATE_PROBE_MASK	0x8000	/* Bit 15 for rate probe */ | ||||||
|  | #define TXFID_RATE_MASK		0x0018	/* Mask for bits 3 and 4 */ | ||||||
|  | #define TXFID_RATE_SHIFT	3	/* Shift 3 bits for rate mask */ | ||||||
|  | 
 | ||||||
|  | /* promote boardrev */ | ||||||
|  | #define BOARDREV_PROMOTABLE	0xFF	/* from */ | ||||||
|  | #define BOARDREV_PROMOTED	1	/* to */ | ||||||
|  | 
 | ||||||
|  | #define DATA_BLOCK_TX_SUPR	(1 << 4) | ||||||
|  | 
 | ||||||
|  | /* 802.1D Priority to TX FIFO number for wme */ | ||||||
|  | extern const u8 prio2fifo[]; | ||||||
|  | 
 | ||||||
|  | /* Ucode MCTL_WAKE override bits */ | ||||||
|  | #define BRCMS_WAKE_OVERRIDE_CLKCTL	0x01 | ||||||
|  | #define BRCMS_WAKE_OVERRIDE_PHYREG	0x02 | ||||||
|  | #define BRCMS_WAKE_OVERRIDE_MACSUSPEND	0x04 | ||||||
|  | #define BRCMS_WAKE_OVERRIDE_TXFIFO	0x08 | ||||||
|  | #define BRCMS_WAKE_OVERRIDE_FORCEFAST	0x10 | ||||||
|  | 
 | ||||||
|  | /* stuff pulled in from wlc.c */ | ||||||
|  | 
 | ||||||
|  | /* Interrupt bit error summary.  Don't include I_RU: we refill DMA at other
 | ||||||
|  |  * times; and if we run out, constant I_RU interrupts may cause lockup.  We | ||||||
|  |  * will still get error counts from rx0ovfl. | ||||||
|  |  */ | ||||||
|  | #define	I_ERRORS	(I_PC | I_PD | I_DE | I_RO | I_XU) | ||||||
|  | /* default software intmasks */ | ||||||
|  | #define	DEF_RXINTMASK	(I_RI)	/* enable rx int on rxfifo only */ | ||||||
|  | #define	DEF_MACINTMASK	(MI_TXSTOP | MI_TBTT | MI_ATIMWINEND | MI_PMQ | \ | ||||||
|  | 			 MI_PHYTXERR | MI_DMAINT | MI_TFS | MI_BG_NOISE | \ | ||||||
|  | 			 MI_CCA | MI_TO | MI_GP0 | MI_RFDISABLE | MI_PWRUP) | ||||||
|  | 
 | ||||||
|  | #define	MAXTXPKTS		6	/* max # pkts pending */ | ||||||
|  | 
 | ||||||
|  | /* frameburst */ | ||||||
|  | #define	MAXTXFRAMEBURST		8 /* vanilla xpress mode: max frames/burst */ | ||||||
|  | #define	MAXFRAMEBURST_TXOP	10000	/* Frameburst TXOP in usec */ | ||||||
|  | 
 | ||||||
|  | #define	NFIFO			6	/* # tx/rx fifopairs */ | ||||||
|  | 
 | ||||||
|  | /* PLL requests */ | ||||||
|  | 
 | ||||||
|  | /* pll is shared on old chips */ | ||||||
|  | #define BRCMS_PLLREQ_SHARED	0x1 | ||||||
|  | /* hold pll for radio monitor register checking */ | ||||||
|  | #define BRCMS_PLLREQ_RADIO_MON	0x2 | ||||||
|  | /* hold/release pll for some short operation */ | ||||||
|  | #define BRCMS_PLLREQ_FLIP		0x4 | ||||||
|  | 
 | ||||||
|  | #define	CHANNEL_BANDUNIT(wlc, ch) \ | ||||||
|  | 	(((ch) <= CH_MAX_2G_CHANNEL) ? BAND_2G_INDEX : BAND_5G_INDEX) | ||||||
|  | 
 | ||||||
|  | #define	OTHERBANDUNIT(wlc) \ | ||||||
|  | 	((uint)((wlc)->band->bandunit ? BAND_2G_INDEX : BAND_5G_INDEX)) | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * 802.11 protection information | ||||||
|  |  * | ||||||
|  |  * _g: use g spec protection, driver internal. | ||||||
|  |  * g_override: override for use of g spec protection. | ||||||
|  |  * gmode_user: user config gmode, operating band->gmode is different. | ||||||
|  |  * overlap: Overlap BSS/IBSS protection for both 11g and 11n. | ||||||
|  |  * nmode_user: user config nmode, operating pub->nmode is different. | ||||||
|  |  * n_cfg: use OFDM protection on MIMO frames. | ||||||
|  |  * n_cfg_override: override for use of N protection. | ||||||
|  |  * nongf: non-GF present protection. | ||||||
|  |  * nongf_override: override for use of GF protection. | ||||||
|  |  * n_pam_override: override for preamble: MM or GF. | ||||||
|  |  * n_obss: indicated OBSS Non-HT STA present. | ||||||
|  | */ | ||||||
|  | struct brcms_protection { | ||||||
|  | 	bool _g; | ||||||
|  | 	s8 g_override; | ||||||
|  | 	u8 gmode_user; | ||||||
|  | 	s8 overlap; | ||||||
|  | 	s8 nmode_user; | ||||||
|  | 	s8 n_cfg; | ||||||
|  | 	s8 n_cfg_override; | ||||||
|  | 	bool nongf; | ||||||
|  | 	s8 nongf_override; | ||||||
|  | 	s8 n_pam_override; | ||||||
|  | 	bool n_obss; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * anything affecting the single/dual streams/antenna operation | ||||||
|  |  * | ||||||
|  |  * hw_txchain: HW txchain bitmap cfg. | ||||||
|  |  * txchain: txchain bitmap being used. | ||||||
|  |  * txstreams: number of txchains being used. | ||||||
|  |  * hw_rxchain: HW rxchain bitmap cfg. | ||||||
|  |  * rxchain: rxchain bitmap being used. | ||||||
|  |  * rxstreams: number of rxchains being used. | ||||||
|  |  * ant_rx_ovr: rx antenna override. | ||||||
|  |  * txant: userTx antenna setting. | ||||||
|  |  * phytxant: phyTx antenna setting in txheader. | ||||||
|  |  * ss_opmode: singlestream Operational mode, 0:siso; 1:cdd. | ||||||
|  |  * ss_algosel_auto: if true, use wlc->stf->ss_algo_channel; | ||||||
|  |  *			else use wlc->band->stf->ss_mode_band. | ||||||
|  |  * ss_algo_channel: ss based on per-channel algo: 0: SISO, 1: CDD 2: STBC. | ||||||
|  |  * rxchain_restore_delay: delay time to restore default rxchain. | ||||||
|  |  * ldpc: AUTO/ON/OFF ldpc cap supported. | ||||||
|  |  * txcore[MAX_STREAMS_SUPPORTED + 1]: bitmap of selected core for each Nsts. | ||||||
|  |  * spatial_policy: | ||||||
|  |  */ | ||||||
|  | struct brcms_stf { | ||||||
|  | 	u8 hw_txchain; | ||||||
|  | 	u8 txchain; | ||||||
|  | 	u8 txstreams; | ||||||
|  | 	u8 hw_rxchain; | ||||||
|  | 	u8 rxchain; | ||||||
|  | 	u8 rxstreams; | ||||||
|  | 	u8 ant_rx_ovr; | ||||||
|  | 	s8 txant; | ||||||
|  | 	u16 phytxant; | ||||||
|  | 	u8 ss_opmode; | ||||||
|  | 	bool ss_algosel_auto; | ||||||
|  | 	u16 ss_algo_channel; | ||||||
|  | 	u8 rxchain_restore_delay; | ||||||
|  | 	s8 ldpc; | ||||||
|  | 	u8 txcore[MAX_STREAMS_SUPPORTED + 1]; | ||||||
|  | 	s8 spatial_policy; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #define BRCMS_STF_SS_STBC_TX(wlc, scb) \ | ||||||
|  | 	(((wlc)->stf->txstreams > 1) && (((wlc)->band->band_stf_stbc_tx == ON) \ | ||||||
|  | 	 || (((scb)->flags & SCB_STBCCAP) && \ | ||||||
|  | 	     (wlc)->band->band_stf_stbc_tx == AUTO && \ | ||||||
|  | 	     isset(&((wlc)->stf->ss_algo_channel), PHY_TXC1_MODE_STBC)))) | ||||||
|  | 
 | ||||||
|  | #define BRCMS_STBC_CAP_PHY(wlc) (BRCMS_ISNPHY(wlc->band) && \ | ||||||
|  | 				 NREV_GE(wlc->band->phyrev, 3)) | ||||||
|  | 
 | ||||||
|  | #define BRCMS_SGI_CAP_PHY(wlc) ((BRCMS_ISNPHY(wlc->band) && \ | ||||||
|  | 				 NREV_GE(wlc->band->phyrev, 3)) || \ | ||||||
|  | 				BRCMS_ISLCNPHY(wlc->band)) | ||||||
|  | 
 | ||||||
|  | #define BRCMS_CHAN_PHYTYPE(x)     (((x) & RXS_CHAN_PHYTYPE_MASK) \ | ||||||
|  | 				   >> RXS_CHAN_PHYTYPE_SHIFT) | ||||||
|  | #define BRCMS_CHAN_CHANNEL(x)     (((x) & RXS_CHAN_ID_MASK) \ | ||||||
|  | 				   >> RXS_CHAN_ID_SHIFT) | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * core state (mac) | ||||||
|  |  */ | ||||||
|  | struct brcms_core { | ||||||
|  | 	uint coreidx;		/* # sb enumerated core */ | ||||||
|  | 
 | ||||||
|  | 	/* fifo */ | ||||||
|  | 	uint *txavail[NFIFO];	/* # tx descriptors available */ | ||||||
|  | 	s16 txpktpend[NFIFO];	/* tx admission control */ | ||||||
|  | 
 | ||||||
|  | 	struct macstat *macstat_snapshot;	/* mac hw prev read values */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * band state (phy+ana+radio) | ||||||
|  |  */ | ||||||
|  | struct brcms_band { | ||||||
|  | 	int bandtype;		/* BRCM_BAND_2G, BRCM_BAND_5G */ | ||||||
|  | 	uint bandunit;		/* bandstate[] index */ | ||||||
|  | 
 | ||||||
|  | 	u16 phytype;		/* phytype */ | ||||||
|  | 	u16 phyrev; | ||||||
|  | 	u16 radioid; | ||||||
|  | 	u16 radiorev; | ||||||
|  | 	struct brcms_phy_pub *pi; /* pointer to phy specific information */ | ||||||
|  | 	bool abgphy_encore; | ||||||
|  | 
 | ||||||
|  | 	u8 gmode;		/* currently active gmode */ | ||||||
|  | 
 | ||||||
|  | 	struct scb *hwrs_scb;	/* permanent scb for hw rateset */ | ||||||
|  | 
 | ||||||
|  | 	/* band-specific copy of default_bss.rateset */ | ||||||
|  | 	struct brcms_c_rateset defrateset; | ||||||
|  | 
 | ||||||
|  | 	u8 band_stf_ss_mode;	/* Configured STF type, 0:siso; 1:cdd */ | ||||||
|  | 	s8 band_stf_stbc_tx;	/* STBC TX 0:off; 1:force on; -1:auto */ | ||||||
|  | 	/* rates supported by chip (phy-specific) */ | ||||||
|  | 	struct brcms_c_rateset hw_rateset; | ||||||
|  | 	u8 basic_rate[BRCM_MAXRATE + 1]; /* basic rates indexed by rate */ | ||||||
|  | 	bool mimo_cap_40;	/* 40 MHz cap enabled on this band */ | ||||||
|  | 	s8 antgain;		/* antenna gain from srom */ | ||||||
|  | 
 | ||||||
|  | 	u16 CWmin; /* minimum size of contention window, in unit of aSlotTime */ | ||||||
|  | 	u16 CWmax; /* maximum size of contention window, in unit of aSlotTime */ | ||||||
|  | 	struct ieee80211_supported_band band; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* module control blocks */ | ||||||
|  | struct modulecb { | ||||||
|  | 	/* module name : NULL indicates empty array member */ | ||||||
|  | 	char name[32]; | ||||||
|  | 	/* handle passed when handler 'doiovar' is called */ | ||||||
|  | 	struct brcms_info *hdl; | ||||||
|  | 
 | ||||||
|  | 	int (*down_fn)(void *handle); /* down handler. Note: the int returned
 | ||||||
|  | 				       * by the down function is a count of the | ||||||
|  | 				       * number of timers that could not be | ||||||
|  | 				       * freed. | ||||||
|  | 				       */ | ||||||
|  | 
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct brcms_hw_band { | ||||||
|  | 	int bandtype;		/* BRCM_BAND_2G, BRCM_BAND_5G */ | ||||||
|  | 	uint bandunit;		/* bandstate[] index */ | ||||||
|  | 	u16 mhfs[MHFMAX];	/* MHF array shadow */ | ||||||
|  | 	u8 bandhw_stf_ss_mode;	/* HW configured STF type, 0:siso; 1:cdd */ | ||||||
|  | 	u16 CWmin; | ||||||
|  | 	u16 CWmax; | ||||||
|  | 	u32 core_flags; | ||||||
|  | 
 | ||||||
|  | 	u16 phytype;		/* phytype */ | ||||||
|  | 	u16 phyrev; | ||||||
|  | 	u16 radioid; | ||||||
|  | 	u16 radiorev; | ||||||
|  | 	struct brcms_phy_pub *pi; /* pointer to phy specific information */ | ||||||
|  | 	bool abgphy_encore; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct brcms_hardware { | ||||||
|  | 	bool _piomode;		/* true if pio mode */ | ||||||
|  | 	struct brcms_c_info *wlc; | ||||||
|  | 
 | ||||||
|  | 	/* fifo */ | ||||||
|  | 	struct dma_pub *di[NFIFO];	/* dma handles, per fifo */ | ||||||
|  | 
 | ||||||
|  | 	uint unit;		/* device instance number */ | ||||||
|  | 
 | ||||||
|  | 	/* version info */ | ||||||
|  | 	u16 vendorid;	/* PCI vendor id */ | ||||||
|  | 	u16 deviceid;	/* PCI device id */ | ||||||
|  | 	uint corerev;		/* core revision */ | ||||||
|  | 	u8 sromrev;		/* version # of the srom */ | ||||||
|  | 	u16 boardrev;	/* version # of particular board */ | ||||||
|  | 	u32 boardflags;	/* Board specific flags from srom */ | ||||||
|  | 	u32 boardflags2;	/* More board flags if sromrev >= 4 */ | ||||||
|  | 	u32 machwcap;	/* MAC capabilities */ | ||||||
|  | 	u32 machwcap_backup;	/* backup of machwcap */ | ||||||
|  | 
 | ||||||
|  | 	struct si_pub *sih;	/* SI handle (cookie for siutils calls) */ | ||||||
|  | 	struct d11regs __iomem *regs;	/* pointer to device registers */ | ||||||
|  | 	struct phy_shim_info *physhim; /* phy shim layer handler */ | ||||||
|  | 	struct shared_phy *phy_sh;	/* pointer to shared phy state */ | ||||||
|  | 	struct brcms_hw_band *band;/* pointer to active per-band state */ | ||||||
|  | 	/* band state per phy/radio */ | ||||||
|  | 	struct brcms_hw_band *bandstate[MAXBANDS]; | ||||||
|  | 	u16 bmac_phytxant;	/* cache of high phytxant state */ | ||||||
|  | 	bool shortslot;		/* currently using 11g ShortSlot timing */ | ||||||
|  | 	u16 SRL;		/* 802.11 dot11ShortRetryLimit */ | ||||||
|  | 	u16 LRL;		/* 802.11 dot11LongRetryLimit */ | ||||||
|  | 	u16 SFBL;		/* Short Frame Rate Fallback Limit */ | ||||||
|  | 	u16 LFBL;		/* Long Frame Rate Fallback Limit */ | ||||||
|  | 
 | ||||||
|  | 	bool up;		/* d11 hardware up and running */ | ||||||
|  | 	uint now;		/* # elapsed seconds */ | ||||||
|  | 	uint _nbands;		/* # bands supported */ | ||||||
|  | 	u16 chanspec;	/* bmac chanspec shadow */ | ||||||
|  | 
 | ||||||
|  | 	uint *txavail[NFIFO];	/* # tx descriptors available */ | ||||||
|  | 	const u16 *xmtfifo_sz;	/* fifo size in 256B for each xmt fifo */ | ||||||
|  | 
 | ||||||
|  | 	u32 pllreq;		/* pll requests to keep PLL on */ | ||||||
|  | 
 | ||||||
|  | 	u8 suspended_fifos;	/* Which TX fifo to remain awake for */ | ||||||
|  | 	u32 maccontrol;	/* Cached value of maccontrol */ | ||||||
|  | 	uint mac_suspend_depth;	/* current depth of mac_suspend levels */ | ||||||
|  | 	u32 wake_override;	/* bit flags to force MAC to WAKE mode */ | ||||||
|  | 	u32 mute_override;	/* Prevent ucode from sending beacons */ | ||||||
|  | 	u8 etheraddr[ETH_ALEN];	/* currently configured ethernet address */ | ||||||
|  | 	bool noreset;		/* true= do not reset hw, used by WLC_OUT */ | ||||||
|  | 	bool forcefastclk;	/* true if h/w is forcing to use fast clk */ | ||||||
|  | 	bool clk;		/* core is out of reset and has clock */ | ||||||
|  | 	bool sbclk;		/* sb has clock */ | ||||||
|  | 	bool phyclk;		/* phy is out of reset and has clock */ | ||||||
|  | 
 | ||||||
|  | 	bool ucode_loaded;	/* true after ucode downloaded */ | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 	u8 hw_stf_ss_opmode;	/* STF single stream operation mode */ | ||||||
|  | 	u8 antsel_type;	/* Type of boardlevel mimo antenna switch-logic
 | ||||||
|  | 				 * 0 = N/A, 1 = 2x4 board, 2 = 2x3 CB2 board | ||||||
|  | 				 */ | ||||||
|  | 	u32 antsel_avail;	/*
 | ||||||
|  | 				 * put struct antsel_info here if more info is | ||||||
|  | 				 * needed | ||||||
|  | 				 */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* TX Queue information
 | ||||||
|  |  * | ||||||
|  |  * Each flow of traffic out of the device has a TX Queue with independent | ||||||
|  |  * flow control. Several interfaces may be associated with a single TX Queue | ||||||
|  |  * if they belong to the same flow of traffic from the device. For multi-channel | ||||||
|  |  * operation there are independent TX Queues for each channel. | ||||||
|  |  */ | ||||||
|  | struct brcms_txq_info { | ||||||
|  | 	struct brcms_txq_info *next; | ||||||
|  | 	struct pktq q; | ||||||
|  | 	uint stopped;		/* tx flow control bits */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Principal common driver data structure. | ||||||
|  |  * | ||||||
|  |  * pub: pointer to driver public state. | ||||||
|  |  * wl: pointer to specific private state. | ||||||
|  |  * regs: pointer to device registers. | ||||||
|  |  * hw: HW related state. | ||||||
|  |  * clkreq_override: setting for clkreq for PCIE : Auto, 0, 1. | ||||||
|  |  * fastpwrup_dly: time in us needed to bring up d11 fast clock. | ||||||
|  |  * macintstatus: bit channel between isr and dpc. | ||||||
|  |  * macintmask: sw runtime master macintmask value. | ||||||
|  |  * defmacintmask: default "on" macintmask value. | ||||||
|  |  * clk: core is out of reset and has clock. | ||||||
|  |  * core: pointer to active io core. | ||||||
|  |  * band: pointer to active per-band state. | ||||||
|  |  * corestate: per-core state (one per hw core). | ||||||
|  |  * bandstate: per-band state (one per phy/radio). | ||||||
|  |  * qvalid: DirFrmQValid and BcMcFrmQValid. | ||||||
|  |  * ampdu: ampdu module handler. | ||||||
|  |  * asi: antsel module handler. | ||||||
|  |  * cmi: channel manager module handler. | ||||||
|  |  * vendorid: PCI vendor id. | ||||||
|  |  * deviceid: PCI device id. | ||||||
|  |  * ucode_rev: microcode revision. | ||||||
|  |  * machwcap: MAC capabilities, BMAC shadow. | ||||||
|  |  * perm_etheraddr: original sprom local ethernet address. | ||||||
|  |  * bandlocked: disable auto multi-band switching. | ||||||
|  |  * bandinit_pending: track band init in auto band. | ||||||
|  |  * radio_monitor: radio timer is running. | ||||||
|  |  * going_down: down path intermediate variable. | ||||||
|  |  * mpc: enable minimum power consumption. | ||||||
|  |  * mpc_dlycnt: # of watchdog cnt before turn disable radio. | ||||||
|  |  * mpc_offcnt: # of watchdog cnt that radio is disabled. | ||||||
|  |  * mpc_delay_off: delay radio disable by # of watchdog cnt. | ||||||
|  |  * prev_non_delay_mpc: prev state brcms_c_is_non_delay_mpc. | ||||||
|  |  * wdtimer: timer for watchdog routine. | ||||||
|  |  * radio_timer: timer for hw radio button monitor routine. | ||||||
|  |  * monitor: monitor (MPDU sniffing) mode. | ||||||
|  |  * bcnmisc_monitor: bcns promisc mode override for monitor. | ||||||
|  |  * _rifs: enable per-packet rifs. | ||||||
|  |  * bcn_li_bcn: beacon listen interval in # beacons. | ||||||
|  |  * bcn_li_dtim: beacon listen interval in # dtims. | ||||||
|  |  * WDarmed: watchdog timer is armed. | ||||||
|  |  * WDlast: last time wlc_watchdog() was called. | ||||||
|  |  * edcf_txop[AC_COUNT]: current txop for each ac. | ||||||
|  |  * wme_retries: per-AC retry limits. | ||||||
|  |  * tx_prec_map: Precedence map based on HW FIFO space. | ||||||
|  |  * fifo2prec_map[NFIFO]: pointer to fifo2_prec map based on WME. | ||||||
|  |  * bsscfg: set of BSS configurations, idx 0 is default and always valid. | ||||||
|  |  * cfg: the primary bsscfg (can be AP or STA). | ||||||
|  |  * tx_queues: common TX Queue list. | ||||||
|  |  * modulecb: | ||||||
|  |  * mimoft: SIGN or 11N. | ||||||
|  |  * cck_40txbw: 11N, cck tx b/w override when in 40MHZ mode. | ||||||
|  |  * ofdm_40txbw: 11N, ofdm tx b/w override when in 40MHZ mode. | ||||||
|  |  * mimo_40txbw: 11N, mimo tx b/w override when in 40MHZ mode. | ||||||
|  |  * default_bss: configured BSS parameters. | ||||||
|  |  * mc_fid_counter: BC/MC FIFO frame ID counter. | ||||||
|  |  * country_default: saved country for leaving 802.11d auto-country mode. | ||||||
|  |  * autocountry_default: initial country for 802.11d auto-country mode. | ||||||
|  |  * prb_resp_timeout: do not send prb resp if request older | ||||||
|  |  *		     than this, 0 = disable. | ||||||
|  |  * home_chanspec: shared home chanspec. | ||||||
|  |  * chanspec: target operational channel. | ||||||
|  |  * usr_fragthresh: user configured fragmentation threshold. | ||||||
|  |  * fragthresh[NFIFO]: per-fifo fragmentation thresholds. | ||||||
|  |  * RTSThresh: 802.11 dot11RTSThreshold. | ||||||
|  |  * SRL: 802.11 dot11ShortRetryLimit. | ||||||
|  |  * LRL: 802.11 dot11LongRetryLimit. | ||||||
|  |  * SFBL: Short Frame Rate Fallback Limit. | ||||||
|  |  * LFBL: Long Frame Rate Fallback Limit. | ||||||
|  |  * shortslot: currently using 11g ShortSlot timing. | ||||||
|  |  * shortslot_override: 11g ShortSlot override. | ||||||
|  |  * include_legacy_erp: include Legacy ERP info elt ID 47 as well as g ID 42. | ||||||
|  |  * PLCPHdr_override: 802.11b Preamble Type override. | ||||||
|  |  * stf: | ||||||
|  |  * bcn_rspec: save bcn ratespec purpose. | ||||||
|  |  * tempsense_lasttime; | ||||||
|  |  * tx_duty_cycle_ofdm: maximum allowed duty cycle for OFDM. | ||||||
|  |  * tx_duty_cycle_cck: maximum allowed duty cycle for CCK. | ||||||
|  |  * pkt_queue: txq for transmit packets. | ||||||
|  |  * wiphy: | ||||||
|  |  * pri_scb: primary Station Control Block | ||||||
|  |  */ | ||||||
|  | struct brcms_c_info { | ||||||
|  | 	struct brcms_pub *pub; | ||||||
|  | 	struct brcms_info *wl; | ||||||
|  | 	struct d11regs __iomem *regs; | ||||||
|  | 	struct brcms_hardware *hw; | ||||||
|  | 
 | ||||||
|  | 	/* clock */ | ||||||
|  | 	u16 fastpwrup_dly; | ||||||
|  | 
 | ||||||
|  | 	/* interrupt */ | ||||||
|  | 	u32 macintstatus; | ||||||
|  | 	u32 macintmask; | ||||||
|  | 	u32 defmacintmask; | ||||||
|  | 
 | ||||||
|  | 	bool clk; | ||||||
|  | 
 | ||||||
|  | 	/* multiband */ | ||||||
|  | 	struct brcms_core *core; | ||||||
|  | 	struct brcms_band *band; | ||||||
|  | 	struct brcms_core *corestate; | ||||||
|  | 	struct brcms_band *bandstate[MAXBANDS]; | ||||||
|  | 
 | ||||||
|  | 	/* packet queue */ | ||||||
|  | 	uint qvalid; | ||||||
|  | 
 | ||||||
|  | 	struct ampdu_info *ampdu; | ||||||
|  | 	struct antsel_info *asi; | ||||||
|  | 	struct brcms_cm_info *cmi; | ||||||
|  | 
 | ||||||
|  | 	u16 vendorid; | ||||||
|  | 	u16 deviceid; | ||||||
|  | 	uint ucode_rev; | ||||||
|  | 
 | ||||||
|  | 	u8 perm_etheraddr[ETH_ALEN]; | ||||||
|  | 
 | ||||||
|  | 	bool bandlocked; | ||||||
|  | 	bool bandinit_pending; | ||||||
|  | 
 | ||||||
|  | 	bool radio_monitor; | ||||||
|  | 	bool going_down; | ||||||
|  | 
 | ||||||
|  | 	bool mpc; | ||||||
|  | 	u8 mpc_dlycnt; | ||||||
|  | 	u8 mpc_offcnt; | ||||||
|  | 	u8 mpc_delay_off; | ||||||
|  | 	u8 prev_non_delay_mpc; | ||||||
|  | 
 | ||||||
|  | 	struct brcms_timer *wdtimer; | ||||||
|  | 	struct brcms_timer *radio_timer; | ||||||
|  | 
 | ||||||
|  | 	/* promiscuous */ | ||||||
|  | 	bool monitor; | ||||||
|  | 	bool bcnmisc_monitor; | ||||||
|  | 
 | ||||||
|  | 	/* driver feature */ | ||||||
|  | 	bool _rifs; | ||||||
|  | 
 | ||||||
|  | 	/* AP-STA synchronization, power save */ | ||||||
|  | 	u8 bcn_li_bcn; | ||||||
|  | 	u8 bcn_li_dtim; | ||||||
|  | 
 | ||||||
|  | 	bool WDarmed; | ||||||
|  | 	u32 WDlast; | ||||||
|  | 
 | ||||||
|  | 	/* WME */ | ||||||
|  | 	u16 edcf_txop[AC_COUNT]; | ||||||
|  | 
 | ||||||
|  | 	u16 wme_retries[AC_COUNT]; | ||||||
|  | 	u16 tx_prec_map; | ||||||
|  | 	u16 fifo2prec_map[NFIFO]; | ||||||
|  | 
 | ||||||
|  | 	struct brcms_bss_cfg *bsscfg; | ||||||
|  | 
 | ||||||
|  | 	/* tx queue */ | ||||||
|  | 	struct brcms_txq_info *tx_queues; | ||||||
|  | 
 | ||||||
|  | 	struct modulecb *modulecb; | ||||||
|  | 
 | ||||||
|  | 	u8 mimoft; | ||||||
|  | 	s8 cck_40txbw; | ||||||
|  | 	s8 ofdm_40txbw; | ||||||
|  | 	s8 mimo_40txbw; | ||||||
|  | 
 | ||||||
|  | 	struct brcms_bss_info *default_bss; | ||||||
|  | 
 | ||||||
|  | 	u16 mc_fid_counter; | ||||||
|  | 
 | ||||||
|  | 	char country_default[BRCM_CNTRY_BUF_SZ]; | ||||||
|  | 	char autocountry_default[BRCM_CNTRY_BUF_SZ]; | ||||||
|  | 	u16 prb_resp_timeout; | ||||||
|  | 
 | ||||||
|  | 	u16 home_chanspec; | ||||||
|  | 
 | ||||||
|  | 	/* PHY parameters */ | ||||||
|  | 	u16 chanspec; | ||||||
|  | 	u16 usr_fragthresh; | ||||||
|  | 	u16 fragthresh[NFIFO]; | ||||||
|  | 	u16 RTSThresh; | ||||||
|  | 	u16 SRL; | ||||||
|  | 	u16 LRL; | ||||||
|  | 	u16 SFBL; | ||||||
|  | 	u16 LFBL; | ||||||
|  | 
 | ||||||
|  | 	/* network config */ | ||||||
|  | 	bool shortslot; | ||||||
|  | 	s8 shortslot_override; | ||||||
|  | 	bool include_legacy_erp; | ||||||
|  | 
 | ||||||
|  | 	struct brcms_protection *protection; | ||||||
|  | 	s8 PLCPHdr_override; | ||||||
|  | 
 | ||||||
|  | 	struct brcms_stf *stf; | ||||||
|  | 
 | ||||||
|  | 	u32 bcn_rspec; | ||||||
|  | 
 | ||||||
|  | 	uint tempsense_lasttime; | ||||||
|  | 
 | ||||||
|  | 	u16 tx_duty_cycle_ofdm; | ||||||
|  | 	u16 tx_duty_cycle_cck; | ||||||
|  | 
 | ||||||
|  | 	struct brcms_txq_info *pkt_queue; | ||||||
|  | 	struct wiphy *wiphy; | ||||||
|  | 	struct scb pri_scb; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* antsel module specific state */ | ||||||
|  | struct antsel_info { | ||||||
|  | 	struct brcms_c_info *wlc;	/* pointer to main wlc structure */ | ||||||
|  | 	struct brcms_pub *pub;		/* pointer to public fn */ | ||||||
|  | 	u8 antsel_type;	/* Type of boardlevel mimo antenna switch-logic
 | ||||||
|  | 				 * 0 = N/A, 1 = 2x4 board, 2 = 2x3 CB2 board | ||||||
|  | 				 */ | ||||||
|  | 	u8 antsel_antswitch;	/* board level antenna switch type */ | ||||||
|  | 	bool antsel_avail;	/* Ant selection availability (SROM based) */ | ||||||
|  | 	struct brcms_antselcfg antcfg_11n; /* antenna configuration */ | ||||||
|  | 	struct brcms_antselcfg antcfg_cur; /* current antenna config (auto) */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * BSS configuration state | ||||||
|  |  * | ||||||
|  |  * wlc: wlc to which this bsscfg belongs to. | ||||||
|  |  * up: is this configuration up operational | ||||||
|  |  * enable: is this configuration enabled | ||||||
|  |  * associated: is BSS in ASSOCIATED state | ||||||
|  |  * BSS: infraustructure or adhoc | ||||||
|  |  * SSID_len: the length of SSID | ||||||
|  |  * SSID: SSID string | ||||||
|  |  * | ||||||
|  |  * | ||||||
|  |  * BSSID: BSSID (associated) | ||||||
|  |  * cur_etheraddr: h/w address | ||||||
|  |  * flags: BSSCFG flags; see below | ||||||
|  |  * | ||||||
|  |  * current_bss: BSS parms in ASSOCIATED state | ||||||
|  |  * | ||||||
|  |  * | ||||||
|  |  * ID: 'unique' ID of this bsscfg, assigned at bsscfg allocation | ||||||
|  |  */ | ||||||
|  | struct brcms_bss_cfg { | ||||||
|  | 	struct brcms_c_info *wlc; | ||||||
|  | 	bool up; | ||||||
|  | 	bool enable; | ||||||
|  | 	bool associated; | ||||||
|  | 	bool BSS; | ||||||
|  | 	u8 SSID_len; | ||||||
|  | 	u8 SSID[IEEE80211_MAX_SSID_LEN]; | ||||||
|  | 	u8 BSSID[ETH_ALEN]; | ||||||
|  | 	u8 cur_etheraddr[ETH_ALEN]; | ||||||
|  | 	struct brcms_bss_info *current_bss; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | extern void brcms_c_fatal_error(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_b_rpc_watchdog(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_recv(struct brcms_c_info *wlc, struct sk_buff *p); | ||||||
|  | extern void brcms_c_txfifo(struct brcms_c_info *wlc, uint fifo, | ||||||
|  | 			   struct sk_buff *p, | ||||||
|  | 			   bool commit, s8 txpktpend); | ||||||
|  | extern void brcms_c_txfifo_complete(struct brcms_c_info *wlc, uint fifo, | ||||||
|  | 				    s8 txpktpend); | ||||||
|  | extern void brcms_c_txq_enq(struct brcms_c_info *wlc, struct scb *scb, | ||||||
|  | 			    struct sk_buff *sdu, uint prec); | ||||||
|  | extern void brcms_c_info_init(struct brcms_c_info *wlc, int unit); | ||||||
|  | extern void brcms_c_print_txstatus(struct tx_status *txs); | ||||||
|  | extern int brcms_b_xmtfifo_sz_get(struct brcms_hardware *wlc_hw, uint fifo, | ||||||
|  | 		   uint *blocks); | ||||||
|  | 
 | ||||||
|  | #if defined(BCMDBG) | ||||||
|  | extern void brcms_c_print_rxh(struct d11rxhdr *rxh); | ||||||
|  | extern void brcms_c_print_txdesc(struct d11txh *txh); | ||||||
|  | #else | ||||||
|  | #define brcms_c_print_txdesc(a) | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | extern void brcms_c_setxband(struct brcms_hardware *wlc_hw, uint bandunit); | ||||||
|  | extern void brcms_c_coredisable(struct brcms_hardware *wlc_hw); | ||||||
|  | 
 | ||||||
|  | extern bool brcms_c_valid_rate(struct brcms_c_info *wlc, u32 rate, | ||||||
|  | 			       int band, bool verbose); | ||||||
|  | extern void brcms_c_ap_upd(struct brcms_c_info *wlc); | ||||||
|  | 
 | ||||||
|  | /* helper functions */ | ||||||
|  | extern void brcms_c_shm_ssid_upd(struct brcms_c_info *wlc, | ||||||
|  | 				 struct brcms_bss_cfg *cfg); | ||||||
|  | extern int brcms_c_set_gmode(struct brcms_c_info *wlc, u8 gmode, bool config); | ||||||
|  | 
 | ||||||
|  | extern void brcms_c_mac_bcn_promisc_change(struct brcms_c_info *wlc, | ||||||
|  | 					   bool promisc); | ||||||
|  | extern void brcms_c_mac_bcn_promisc(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_mac_promisc(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_txflowcontrol(struct brcms_c_info *wlc, | ||||||
|  | 				  struct brcms_txq_info *qi, | ||||||
|  | 				  bool on, int prio); | ||||||
|  | extern void brcms_c_txflowcontrol_override(struct brcms_c_info *wlc, | ||||||
|  | 				       struct brcms_txq_info *qi, | ||||||
|  | 				       bool on, uint override); | ||||||
|  | extern bool brcms_c_txflowcontrol_prio_isset(struct brcms_c_info *wlc, | ||||||
|  | 					     struct brcms_txq_info *qi, | ||||||
|  | 					     int prio); | ||||||
|  | extern void brcms_c_send_q(struct brcms_c_info *wlc); | ||||||
|  | extern int brcms_c_prep_pdu(struct brcms_c_info *wlc, struct sk_buff *pdu, | ||||||
|  | 			    uint *fifo); | ||||||
|  | 
 | ||||||
|  | extern u16 brcms_c_calc_lsig_len(struct brcms_c_info *wlc, u32 ratespec, | ||||||
|  | 				uint mac_len); | ||||||
|  | extern u32 brcms_c_rspec_to_rts_rspec(struct brcms_c_info *wlc, | ||||||
|  | 					     u32 rspec, | ||||||
|  | 					     bool use_rspec, u16 mimo_ctlchbw); | ||||||
|  | extern u16 brcms_c_compute_rtscts_dur(struct brcms_c_info *wlc, bool cts_only, | ||||||
|  | 				      u32 rts_rate, | ||||||
|  | 				      u32 frame_rate, | ||||||
|  | 				      u8 rts_preamble_type, | ||||||
|  | 				      u8 frame_preamble_type, uint frame_len, | ||||||
|  | 				      bool ba); | ||||||
|  | 
 | ||||||
|  | extern void brcms_c_tbtt(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_inval_dma_pkts(struct brcms_hardware *hw, | ||||||
|  | 			       struct ieee80211_sta *sta, | ||||||
|  | 			       void (*dma_callback_fn)); | ||||||
|  | 
 | ||||||
|  | /* Shared memory access */ | ||||||
|  | extern void brcms_c_copyto_shm(struct brcms_c_info *wlc, uint offset, | ||||||
|  | 			       const void *buf, int len); | ||||||
|  | 
 | ||||||
|  | extern void brcms_c_update_beacon(struct brcms_c_info *wlc); | ||||||
|  | 
 | ||||||
|  | extern void brcms_c_update_probe_resp(struct brcms_c_info *wlc, bool suspend); | ||||||
|  | extern void brcms_c_bss_update_probe_resp(struct brcms_c_info *wlc, | ||||||
|  | 					  struct brcms_bss_cfg *cfg, | ||||||
|  | 					  bool suspend); | ||||||
|  | extern bool brcms_c_ismpc(struct brcms_c_info *wlc); | ||||||
|  | extern bool brcms_c_is_non_delay_mpc(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_radio_mpc_upd(struct brcms_c_info *wlc); | ||||||
|  | extern bool brcms_c_prec_enq_head(struct brcms_c_info *wlc, struct pktq *q, | ||||||
|  | 			      struct sk_buff *pkt, int prec, bool head); | ||||||
|  | extern u16 brcms_c_phytxctl1_calc(struct brcms_c_info *wlc, u32 rspec); | ||||||
|  | extern void brcms_c_compute_plcp(struct brcms_c_info *wlc, u32 rate, | ||||||
|  | 				 uint length, u8 *plcp); | ||||||
|  | extern uint brcms_c_calc_frame_time(struct brcms_c_info *wlc, | ||||||
|  | 				    u32 ratespec, | ||||||
|  | 				    u8 preamble_type, uint mac_len); | ||||||
|  | 
 | ||||||
|  | extern void brcms_c_set_chanspec(struct brcms_c_info *wlc, | ||||||
|  | 				 u16 chanspec); | ||||||
|  | 
 | ||||||
|  | extern bool brcms_c_timers_init(struct brcms_c_info *wlc, int unit); | ||||||
|  | 
 | ||||||
|  | extern int brcms_c_set_nmode(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_mimops_action_ht_send(struct brcms_c_info *wlc, | ||||||
|  | 				      struct brcms_bss_cfg *bsscfg, | ||||||
|  | 				      u8 mimops_mode); | ||||||
|  | 
 | ||||||
|  | extern void brcms_c_switch_shortslot(struct brcms_c_info *wlc, bool shortslot); | ||||||
|  | extern void brcms_c_set_bssid(struct brcms_bss_cfg *cfg); | ||||||
|  | extern void brcms_c_edcf_setparams(struct brcms_c_info *wlc, bool suspend); | ||||||
|  | 
 | ||||||
|  | extern void brcms_c_set_ratetable(struct brcms_c_info *wlc); | ||||||
|  | extern int brcms_c_set_mac(struct brcms_bss_cfg *cfg); | ||||||
|  | extern void brcms_c_beacon_phytxctl_txant_upd(struct brcms_c_info *wlc, | ||||||
|  | 					  u32 bcn_rate); | ||||||
|  | extern void brcms_c_mod_prb_rsp_rate_table(struct brcms_c_info *wlc, | ||||||
|  | 					   uint frame_len); | ||||||
|  | extern u32 brcms_c_lowest_basic_rspec(struct brcms_c_info *wlc, | ||||||
|  | 					     struct brcms_c_rateset *rs); | ||||||
|  | extern void brcms_c_radio_disable(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_bcn_li_upd(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_set_home_chanspec(struct brcms_c_info *wlc, | ||||||
|  | 				      u16 chanspec); | ||||||
|  | extern bool brcms_c_ps_allowed(struct brcms_c_info *wlc); | ||||||
|  | extern bool brcms_c_stay_awake(struct brcms_c_info *wlc); | ||||||
|  | 
 | ||||||
|  | extern void brcms_b_antsel_type_set(struct brcms_hardware *wlc_hw, | ||||||
|  | 				     u8 antsel_type); | ||||||
|  | 
 | ||||||
|  | /* chanspec, ucode interface */ | ||||||
|  | extern void brcms_b_set_chanspec(struct brcms_hardware *wlc_hw, | ||||||
|  | 				  u16 chanspec, | ||||||
|  | 				  bool mute, struct txpwr_limits *txpwr); | ||||||
|  | 
 | ||||||
|  | extern void brcms_b_write_shm(struct brcms_hardware *wlc_hw, uint offset, | ||||||
|  | 			      u16 v); | ||||||
|  | extern u16 brcms_b_read_shm(struct brcms_hardware *wlc_hw, uint offset); | ||||||
|  | 
 | ||||||
|  | extern void brcms_b_mhf(struct brcms_hardware *wlc_hw, u8 idx, u16 mask, | ||||||
|  | 			u16 val, int bands); | ||||||
|  | 
 | ||||||
|  | extern void brcms_b_corereset(struct brcms_hardware *wlc_hw, u32 flags); | ||||||
|  | 
 | ||||||
|  | extern void brcms_b_mctrl(struct brcms_hardware *wlc_hw, u32 mask, u32 val); | ||||||
|  | 
 | ||||||
|  | extern void brcms_b_phy_reset(struct brcms_hardware *wlc_hw); | ||||||
|  | extern void brcms_b_bw_set(struct brcms_hardware *wlc_hw, u16 bw); | ||||||
|  | extern void brcms_b_core_phypll_reset(struct brcms_hardware *wlc_hw); | ||||||
|  | extern void brcms_c_ucode_wake_override_set(struct brcms_hardware *wlc_hw, | ||||||
|  | 					u32 override_bit); | ||||||
|  | extern void brcms_c_ucode_wake_override_clear(struct brcms_hardware *wlc_hw, | ||||||
|  | 					  u32 override_bit); | ||||||
|  | extern void brcms_b_write_template_ram(struct brcms_hardware *wlc_hw, | ||||||
|  | 				       int offset, int len, void *buf); | ||||||
|  | extern u16 brcms_b_rate_shm_offset(struct brcms_hardware *wlc_hw, u8 rate); | ||||||
|  | extern void brcms_b_copyto_objmem(struct brcms_hardware *wlc_hw, | ||||||
|  | 				   uint offset, const void *buf, int len, | ||||||
|  | 				   u32 sel); | ||||||
|  | extern void brcms_b_copyfrom_objmem(struct brcms_hardware *wlc_hw, uint offset, | ||||||
|  | 				     void *buf, int len, u32 sel); | ||||||
|  | extern void brcms_b_switch_macfreq(struct brcms_hardware *wlc_hw, u8 spurmode); | ||||||
|  | extern u16 brcms_b_get_txant(struct brcms_hardware *wlc_hw); | ||||||
|  | extern void brcms_b_phyclk_fgc(struct brcms_hardware *wlc_hw, bool clk); | ||||||
|  | extern void brcms_b_macphyclk_set(struct brcms_hardware *wlc_hw, bool clk); | ||||||
|  | extern void brcms_b_core_phypll_ctl(struct brcms_hardware *wlc_hw, bool on); | ||||||
|  | extern void brcms_b_txant_set(struct brcms_hardware *wlc_hw, u16 phytxant); | ||||||
|  | extern void brcms_b_band_stf_ss_set(struct brcms_hardware *wlc_hw, | ||||||
|  | 				    u8 stf_mode); | ||||||
|  | extern void brcms_c_init_scb(struct scb *scb); | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCM_MAIN_H_ */ | ||||||
							
								
								
									
										835
									
								
								drivers/net/wireless/brcm80211/brcmsmac/nicpci.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										835
									
								
								drivers/net/wireless/brcm80211/brcmsmac/nicpci.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,835 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <linux/slab.h> | ||||||
|  | #include <linux/delay.h> | ||||||
|  | #include <linux/pci.h> | ||||||
|  | 
 | ||||||
|  | #include <defs.h> | ||||||
|  | #include <soc.h> | ||||||
|  | #include <chipcommon.h> | ||||||
|  | #include "aiutils.h" | ||||||
|  | #include "pub.h" | ||||||
|  | #include "nicpci.h" | ||||||
|  | 
 | ||||||
|  | /* SPROM offsets */ | ||||||
|  | #define SRSH_ASPM_OFFSET		4	/* word 4 */ | ||||||
|  | #define SRSH_ASPM_ENB			0x18	/* bit 3, 4 */ | ||||||
|  | #define SRSH_ASPM_L1_ENB		0x10	/* bit 4 */ | ||||||
|  | #define SRSH_ASPM_L0s_ENB		0x8	/* bit 3 */ | ||||||
|  | 
 | ||||||
|  | #define SRSH_PCIE_MISC_CONFIG		5	/* word 5 */ | ||||||
|  | #define SRSH_L23READY_EXIT_NOPERST	0x8000	/* bit 15 */ | ||||||
|  | #define SRSH_CLKREQ_OFFSET_REV5		20	/* word 20 for srom rev <= 5 */ | ||||||
|  | #define SRSH_CLKREQ_ENB			0x0800	/* bit 11 */ | ||||||
|  | #define SRSH_BD_OFFSET                  6	/* word 6 */ | ||||||
|  | 
 | ||||||
|  | /* chipcontrol */ | ||||||
|  | #define CHIPCTRL_4321_PLL_DOWN		0x800000/* serdes PLL down override */ | ||||||
|  | 
 | ||||||
|  | /* MDIO control */ | ||||||
|  | #define MDIOCTL_DIVISOR_MASK		0x7f	/* clock to be used on MDIO */ | ||||||
|  | #define MDIOCTL_DIVISOR_VAL		0x2 | ||||||
|  | #define MDIOCTL_PREAM_EN		0x80	/* Enable preamble sequnce */ | ||||||
|  | #define MDIOCTL_ACCESS_DONE		0x100	/* Transaction complete */ | ||||||
|  | 
 | ||||||
|  | /* MDIO Data */ | ||||||
|  | #define MDIODATA_MASK			0x0000ffff	/* data 2 bytes */ | ||||||
|  | #define MDIODATA_TA			0x00020000	/* Turnaround */ | ||||||
|  | 
 | ||||||
|  | #define MDIODATA_REGADDR_SHF		18		/* Regaddr shift */ | ||||||
|  | #define MDIODATA_REGADDR_MASK		0x007c0000	/* Regaddr Mask */ | ||||||
|  | #define MDIODATA_DEVADDR_SHF		23	/* Physmedia devaddr shift */ | ||||||
|  | #define MDIODATA_DEVADDR_MASK		0x0f800000 | ||||||
|  | 						/* Physmedia devaddr Mask */ | ||||||
|  | 
 | ||||||
|  | /* MDIO Data for older revisions < 10 */ | ||||||
|  | #define MDIODATA_REGADDR_SHF_OLD	18	/* Regaddr shift */ | ||||||
|  | #define MDIODATA_REGADDR_MASK_OLD	0x003c0000 | ||||||
|  | 						/* Regaddr Mask */ | ||||||
|  | #define MDIODATA_DEVADDR_SHF_OLD	22	/* Physmedia devaddr shift  */ | ||||||
|  | #define MDIODATA_DEVADDR_MASK_OLD	0x0fc00000 | ||||||
|  | 						/* Physmedia devaddr Mask */ | ||||||
|  | 
 | ||||||
|  | /* Transactions flags */ | ||||||
|  | #define MDIODATA_WRITE			0x10000000 | ||||||
|  | #define MDIODATA_READ			0x20000000 | ||||||
|  | #define MDIODATA_START			0x40000000 | ||||||
|  | 
 | ||||||
|  | #define MDIODATA_DEV_ADDR		0x0	/* dev address for serdes */ | ||||||
|  | #define	MDIODATA_BLK_ADDR		0x1F	/* blk address for serdes */ | ||||||
|  | 
 | ||||||
|  | /* serdes regs (rev < 10) */ | ||||||
|  | #define MDIODATA_DEV_PLL		0x1d	/* SERDES PLL Dev */ | ||||||
|  | #define MDIODATA_DEV_TX			0x1e	/* SERDES TX Dev */ | ||||||
|  | #define MDIODATA_DEV_RX			0x1f	/* SERDES RX Dev */ | ||||||
|  | 
 | ||||||
|  | /* SERDES RX registers */ | ||||||
|  | #define SERDES_RX_CTRL			1	/* Rx cntrl */ | ||||||
|  | #define SERDES_RX_TIMER1		2	/* Rx Timer1 */ | ||||||
|  | #define SERDES_RX_CDR			6	/* CDR */ | ||||||
|  | #define SERDES_RX_CDRBW			7	/* CDR BW */ | ||||||
|  | /* SERDES RX control register */ | ||||||
|  | #define SERDES_RX_CTRL_FORCE		0x80	/* rxpolarity_force */ | ||||||
|  | #define SERDES_RX_CTRL_POLARITY		0x40	/* rxpolarity_value */ | ||||||
|  | 
 | ||||||
|  | /* SERDES PLL registers */ | ||||||
|  | #define SERDES_PLL_CTRL                 1	/* PLL control reg */ | ||||||
|  | #define PLL_CTRL_FREQDET_EN             0x4000	/* bit 14 is FREQDET on */ | ||||||
|  | 
 | ||||||
|  | /* Linkcontrol reg offset in PCIE Cap */ | ||||||
|  | #define PCIE_CAP_LINKCTRL_OFFSET	16	/* offset in pcie cap */ | ||||||
|  | #define PCIE_CAP_LCREG_ASPML0s		0x01	/* ASPM L0s in linkctrl */ | ||||||
|  | #define PCIE_CAP_LCREG_ASPML1		0x02	/* ASPM L1 in linkctrl */ | ||||||
|  | #define PCIE_CLKREQ_ENAB		0x100	/* CLKREQ Enab in linkctrl */ | ||||||
|  | 
 | ||||||
|  | #define PCIE_ASPM_ENAB			3	/* ASPM L0s & L1 in linkctrl */ | ||||||
|  | #define PCIE_ASPM_L1_ENAB		2	/* ASPM L0s & L1 in linkctrl */ | ||||||
|  | #define PCIE_ASPM_L0s_ENAB		1	/* ASPM L0s & L1 in linkctrl */ | ||||||
|  | #define PCIE_ASPM_DISAB			0	/* ASPM L0s & L1 in linkctrl */ | ||||||
|  | 
 | ||||||
|  | /* Power management threshold */ | ||||||
|  | #define PCIE_L1THRESHOLDTIME_MASK       0xFF00	/* bits 8 - 15 */ | ||||||
|  | #define PCIE_L1THRESHOLDTIME_SHIFT      8	/* PCIE_L1THRESHOLDTIME_SHIFT */ | ||||||
|  | #define PCIE_L1THRESHOLD_WARVAL         0x72	/* WAR value */ | ||||||
|  | #define PCIE_ASPMTIMER_EXTEND		0x01000000 | ||||||
|  | 						/* > rev7:
 | ||||||
|  | 						 * enable extend ASPM timer | ||||||
|  | 						 */ | ||||||
|  | 
 | ||||||
|  | /* different register spaces to access thru pcie indirect access */ | ||||||
|  | #define PCIE_CONFIGREGS		1	/* Access to config space */ | ||||||
|  | #define PCIE_PCIEREGS		2	/* Access to pcie registers */ | ||||||
|  | 
 | ||||||
|  | /* PCIE protocol PHY diagnostic registers */ | ||||||
|  | #define	PCIE_PLP_STATUSREG		0x204	/* Status */ | ||||||
|  | 
 | ||||||
|  | /* Status reg PCIE_PLP_STATUSREG */ | ||||||
|  | #define PCIE_PLP_POLARITYINV_STAT	0x10 | ||||||
|  | 
 | ||||||
|  | /* PCIE protocol DLLP diagnostic registers */ | ||||||
|  | #define PCIE_DLLP_LCREG			0x100	/* Link Control */ | ||||||
|  | #define PCIE_DLLP_PMTHRESHREG		0x128	/* Power Management Threshold */ | ||||||
|  | 
 | ||||||
|  | /* PCIE protocol TLP diagnostic registers */ | ||||||
|  | #define PCIE_TLP_WORKAROUNDSREG		0x004	/* TLP Workarounds */ | ||||||
|  | 
 | ||||||
|  | /* Sonics to PCI translation types */ | ||||||
|  | #define	SBTOPCI_PREF	0x4		/* prefetch enable */ | ||||||
|  | #define	SBTOPCI_BURST	0x8		/* burst enable */ | ||||||
|  | #define	SBTOPCI_RC_READMULTI	0x20	/* memory read multiple */ | ||||||
|  | 
 | ||||||
|  | #define PCI_CLKRUN_DSBL	0x8000	/* Bit 15 forceClkrun */ | ||||||
|  | 
 | ||||||
|  | /* PCI core index in SROM shadow area */ | ||||||
|  | #define SRSH_PI_OFFSET	0	/* first word */ | ||||||
|  | #define SRSH_PI_MASK	0xf000	/* bit 15:12 */ | ||||||
|  | #define SRSH_PI_SHIFT	12	/* bit 15:12 */ | ||||||
|  | 
 | ||||||
|  | /* Sonics side: PCI core and host control registers */ | ||||||
|  | struct sbpciregs { | ||||||
|  | 	u32 control;		/* PCI control */ | ||||||
|  | 	u32 PAD[3]; | ||||||
|  | 	u32 arbcontrol;		/* PCI arbiter control */ | ||||||
|  | 	u32 clkrun;		/* Clkrun Control (>=rev11) */ | ||||||
|  | 	u32 PAD[2]; | ||||||
|  | 	u32 intstatus;		/* Interrupt status */ | ||||||
|  | 	u32 intmask;		/* Interrupt mask */ | ||||||
|  | 	u32 sbtopcimailbox;	/* Sonics to PCI mailbox */ | ||||||
|  | 	u32 PAD[9]; | ||||||
|  | 	u32 bcastaddr;		/* Sonics broadcast address */ | ||||||
|  | 	u32 bcastdata;		/* Sonics broadcast data */ | ||||||
|  | 	u32 PAD[2]; | ||||||
|  | 	u32 gpioin;		/* ro: gpio input (>=rev2) */ | ||||||
|  | 	u32 gpioout;		/* rw: gpio output (>=rev2) */ | ||||||
|  | 	u32 gpioouten;		/* rw: gpio output enable (>= rev2) */ | ||||||
|  | 	u32 gpiocontrol;	/* rw: gpio control (>= rev2) */ | ||||||
|  | 	u32 PAD[36]; | ||||||
|  | 	u32 sbtopci0;		/* Sonics to PCI translation 0 */ | ||||||
|  | 	u32 sbtopci1;		/* Sonics to PCI translation 1 */ | ||||||
|  | 	u32 sbtopci2;		/* Sonics to PCI translation 2 */ | ||||||
|  | 	u32 PAD[189]; | ||||||
|  | 	u32 pcicfg[4][64];	/* 0x400 - 0x7FF, PCI Cfg Space (>=rev8) */ | ||||||
|  | 	u16 sprom[36];		/* SPROM shadow Area */ | ||||||
|  | 	u32 PAD[46]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* SB side: PCIE core and host control registers */ | ||||||
|  | struct sbpcieregs { | ||||||
|  | 	u32 control;		/* host mode only */ | ||||||
|  | 	u32 PAD[2]; | ||||||
|  | 	u32 biststatus;		/* bist Status: 0x00C */ | ||||||
|  | 	u32 gpiosel;		/* PCIE gpio sel: 0x010 */ | ||||||
|  | 	u32 gpioouten;		/* PCIE gpio outen: 0x14 */ | ||||||
|  | 	u32 PAD[2]; | ||||||
|  | 	u32 intstatus;		/* Interrupt status: 0x20 */ | ||||||
|  | 	u32 intmask;		/* Interrupt mask: 0x24 */ | ||||||
|  | 	u32 sbtopcimailbox;	/* sb to pcie mailbox: 0x028 */ | ||||||
|  | 	u32 PAD[53]; | ||||||
|  | 	u32 sbtopcie0;		/* sb to pcie translation 0: 0x100 */ | ||||||
|  | 	u32 sbtopcie1;		/* sb to pcie translation 1: 0x104 */ | ||||||
|  | 	u32 sbtopcie2;		/* sb to pcie translation 2: 0x108 */ | ||||||
|  | 	u32 PAD[5]; | ||||||
|  | 
 | ||||||
|  | 	/* pcie core supports in direct access to config space */ | ||||||
|  | 	u32 configaddr;	/* pcie config space access: Address field: 0x120 */ | ||||||
|  | 	u32 configdata;	/* pcie config space access: Data field: 0x124 */ | ||||||
|  | 
 | ||||||
|  | 	/* mdio access to serdes */ | ||||||
|  | 	u32 mdiocontrol;	/* controls the mdio access: 0x128 */ | ||||||
|  | 	u32 mdiodata;		/* Data to the mdio access: 0x12c */ | ||||||
|  | 
 | ||||||
|  | 	/* pcie protocol phy/dllp/tlp register indirect access mechanism */ | ||||||
|  | 	u32 pcieindaddr;	/* indirect access to
 | ||||||
|  | 				 * the internal register: 0x130 | ||||||
|  | 				 */ | ||||||
|  | 	u32 pcieinddata;	/* Data to/from the internal regsiter: 0x134 */ | ||||||
|  | 
 | ||||||
|  | 	u32 clkreqenctrl;	/* >= rev 6, Clkreq rdma control : 0x138 */ | ||||||
|  | 	u32 PAD[177]; | ||||||
|  | 	u32 pciecfg[4][64];	/* 0x400 - 0x7FF, PCIE Cfg Space */ | ||||||
|  | 	u16 sprom[64];		/* SPROM shadow Area */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct pcicore_info { | ||||||
|  | 	union { | ||||||
|  | 		struct sbpcieregs __iomem *pcieregs; | ||||||
|  | 		struct sbpciregs __iomem *pciregs; | ||||||
|  | 	} regs;			/* Memory mapped register to the core */ | ||||||
|  | 
 | ||||||
|  | 	struct si_pub *sih;	/* System interconnect handle */ | ||||||
|  | 	struct pci_dev *dev; | ||||||
|  | 	u8 pciecap_lcreg_offset;/* PCIE capability LCreg offset
 | ||||||
|  | 				 * in the config space | ||||||
|  | 				 */ | ||||||
|  | 	bool pcie_pr42767; | ||||||
|  | 	u8 pcie_polarity; | ||||||
|  | 	u8 pcie_war_aspm_ovr;	/* Override ASPM/Clkreq settings */ | ||||||
|  | 
 | ||||||
|  | 	u8 pmecap_offset;	/* PM Capability offset in the config space */ | ||||||
|  | 	bool pmecap;		/* Capable of generating PME */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #define PCIE_ASPM(sih)							\ | ||||||
|  | 	(((sih)->buscoretype == PCIE_CORE_ID) &&			\ | ||||||
|  | 	 (((sih)->buscorerev >= 3) &&					\ | ||||||
|  | 	  ((sih)->buscorerev <= 5))) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /* delay needed between the mdio control/ mdiodata register data access */ | ||||||
|  | static void pr28829_delay(void) | ||||||
|  | { | ||||||
|  | 	udelay(10); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Initialize the PCI core.
 | ||||||
|  |  * It's caller's responsibility to make sure that this is done only once | ||||||
|  |  */ | ||||||
|  | struct pcicore_info *pcicore_init(struct si_pub *sih, struct pci_dev *pdev, | ||||||
|  | 				  void __iomem *regs) | ||||||
|  | { | ||||||
|  | 	struct pcicore_info *pi; | ||||||
|  | 
 | ||||||
|  | 	/* alloc struct pcicore_info */ | ||||||
|  | 	pi = kzalloc(sizeof(struct pcicore_info), GFP_ATOMIC); | ||||||
|  | 	if (pi == NULL) | ||||||
|  | 		return NULL; | ||||||
|  | 
 | ||||||
|  | 	pi->sih = sih; | ||||||
|  | 	pi->dev = pdev; | ||||||
|  | 
 | ||||||
|  | 	if (sih->buscoretype == PCIE_CORE_ID) { | ||||||
|  | 		u8 cap_ptr; | ||||||
|  | 		pi->regs.pcieregs = regs; | ||||||
|  | 		cap_ptr = pcicore_find_pci_capability(pi->dev, PCI_CAP_ID_EXP, | ||||||
|  | 						      NULL, NULL); | ||||||
|  | 		pi->pciecap_lcreg_offset = cap_ptr + PCIE_CAP_LINKCTRL_OFFSET; | ||||||
|  | 	} else | ||||||
|  | 		pi->regs.pciregs = regs; | ||||||
|  | 
 | ||||||
|  | 	return pi; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void pcicore_deinit(struct pcicore_info *pch) | ||||||
|  | { | ||||||
|  | 	kfree(pch); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* return cap_offset if requested capability exists in the PCI config space */ | ||||||
|  | /* Note that it's caller's responsibility to make sure it's a pci bus */ | ||||||
|  | u8 | ||||||
|  | pcicore_find_pci_capability(struct pci_dev *dev, u8 req_cap_id, | ||||||
|  | 			    unsigned char *buf, u32 *buflen) | ||||||
|  | { | ||||||
|  | 	u8 cap_id; | ||||||
|  | 	u8 cap_ptr = 0; | ||||||
|  | 	u32 bufsize; | ||||||
|  | 	u8 byte_val; | ||||||
|  | 
 | ||||||
|  | 	/* check for Header type 0 */ | ||||||
|  | 	pci_read_config_byte(dev, PCI_HEADER_TYPE, &byte_val); | ||||||
|  | 	if ((byte_val & 0x7f) != PCI_HEADER_TYPE_NORMAL) | ||||||
|  | 		goto end; | ||||||
|  | 
 | ||||||
|  | 	/* check if the capability pointer field exists */ | ||||||
|  | 	pci_read_config_byte(dev, PCI_STATUS, &byte_val); | ||||||
|  | 	if (!(byte_val & PCI_STATUS_CAP_LIST)) | ||||||
|  | 		goto end; | ||||||
|  | 
 | ||||||
|  | 	pci_read_config_byte(dev, PCI_CAPABILITY_LIST, &cap_ptr); | ||||||
|  | 	/* check if the capability pointer is 0x00 */ | ||||||
|  | 	if (cap_ptr == 0x00) | ||||||
|  | 		goto end; | ||||||
|  | 
 | ||||||
|  | 	/* loop thru the capability list
 | ||||||
|  | 	 * and see if the pcie capability exists | ||||||
|  | 	 */ | ||||||
|  | 
 | ||||||
|  | 	pci_read_config_byte(dev, cap_ptr, &cap_id); | ||||||
|  | 
 | ||||||
|  | 	while (cap_id != req_cap_id) { | ||||||
|  | 		pci_read_config_byte(dev, cap_ptr + 1, &cap_ptr); | ||||||
|  | 		if (cap_ptr == 0x00) | ||||||
|  | 			break; | ||||||
|  | 		pci_read_config_byte(dev, cap_ptr, &cap_id); | ||||||
|  | 	} | ||||||
|  | 	if (cap_id != req_cap_id) | ||||||
|  | 		goto end; | ||||||
|  | 
 | ||||||
|  | 	/* found the caller requested capability */ | ||||||
|  | 	if (buf != NULL && buflen != NULL) { | ||||||
|  | 		u8 cap_data; | ||||||
|  | 
 | ||||||
|  | 		bufsize = *buflen; | ||||||
|  | 		if (!bufsize) | ||||||
|  | 			goto end; | ||||||
|  | 		*buflen = 0; | ||||||
|  | 		/* copy the capability data excluding cap ID and next ptr */ | ||||||
|  | 		cap_data = cap_ptr + 2; | ||||||
|  | 		if ((bufsize + cap_data) > PCI_SZPCR) | ||||||
|  | 			bufsize = PCI_SZPCR - cap_data; | ||||||
|  | 		*buflen = bufsize; | ||||||
|  | 		while (bufsize--) { | ||||||
|  | 			pci_read_config_byte(dev, cap_data, buf); | ||||||
|  | 			cap_data++; | ||||||
|  | 			buf++; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | end: | ||||||
|  | 	return cap_ptr; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* ***** Register Access API */ | ||||||
|  | static uint | ||||||
|  | pcie_readreg(struct sbpcieregs __iomem *pcieregs, uint addrtype, uint offset) | ||||||
|  | { | ||||||
|  | 	uint retval = 0xFFFFFFFF; | ||||||
|  | 
 | ||||||
|  | 	switch (addrtype) { | ||||||
|  | 	case PCIE_CONFIGREGS: | ||||||
|  | 		W_REG(&pcieregs->configaddr, offset); | ||||||
|  | 		(void)R_REG((&pcieregs->configaddr)); | ||||||
|  | 		retval = R_REG(&pcieregs->configdata); | ||||||
|  | 		break; | ||||||
|  | 	case PCIE_PCIEREGS: | ||||||
|  | 		W_REG(&pcieregs->pcieindaddr, offset); | ||||||
|  | 		(void)R_REG(&pcieregs->pcieindaddr); | ||||||
|  | 		retval = R_REG(&pcieregs->pcieinddata); | ||||||
|  | 		break; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return retval; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static uint pcie_writereg(struct sbpcieregs __iomem *pcieregs, uint addrtype, | ||||||
|  | 			  uint offset, uint val) | ||||||
|  | { | ||||||
|  | 	switch (addrtype) { | ||||||
|  | 	case PCIE_CONFIGREGS: | ||||||
|  | 		W_REG((&pcieregs->configaddr), offset); | ||||||
|  | 		W_REG((&pcieregs->configdata), val); | ||||||
|  | 		break; | ||||||
|  | 	case PCIE_PCIEREGS: | ||||||
|  | 		W_REG((&pcieregs->pcieindaddr), offset); | ||||||
|  | 		W_REG((&pcieregs->pcieinddata), val); | ||||||
|  | 		break; | ||||||
|  | 	default: | ||||||
|  | 		break; | ||||||
|  | 	} | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static bool pcie_mdiosetblock(struct pcicore_info *pi, uint blk) | ||||||
|  | { | ||||||
|  | 	struct sbpcieregs __iomem *pcieregs = pi->regs.pcieregs; | ||||||
|  | 	uint mdiodata, i = 0; | ||||||
|  | 	uint pcie_serdes_spinwait = 200; | ||||||
|  | 
 | ||||||
|  | 	mdiodata = (MDIODATA_START | MDIODATA_WRITE | MDIODATA_TA | | ||||||
|  | 		    (MDIODATA_DEV_ADDR << MDIODATA_DEVADDR_SHF) | | ||||||
|  | 		    (MDIODATA_BLK_ADDR << MDIODATA_REGADDR_SHF) | | ||||||
|  | 		    (blk << 4)); | ||||||
|  | 	W_REG(&pcieregs->mdiodata, mdiodata); | ||||||
|  | 
 | ||||||
|  | 	pr28829_delay(); | ||||||
|  | 	/* retry till the transaction is complete */ | ||||||
|  | 	while (i < pcie_serdes_spinwait) { | ||||||
|  | 		if (R_REG(&pcieregs->mdiocontrol) & MDIOCTL_ACCESS_DONE) | ||||||
|  | 			break; | ||||||
|  | 
 | ||||||
|  | 		udelay(1000); | ||||||
|  | 		i++; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (i >= pcie_serdes_spinwait) | ||||||
|  | 		return false; | ||||||
|  | 
 | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | pcie_mdioop(struct pcicore_info *pi, uint physmedia, uint regaddr, bool write, | ||||||
|  | 	    uint *val) | ||||||
|  | { | ||||||
|  | 	struct sbpcieregs __iomem *pcieregs = pi->regs.pcieregs; | ||||||
|  | 	uint mdiodata; | ||||||
|  | 	uint i = 0; | ||||||
|  | 	uint pcie_serdes_spinwait = 10; | ||||||
|  | 
 | ||||||
|  | 	/* enable mdio access to SERDES */ | ||||||
|  | 	W_REG(&pcieregs->mdiocontrol, MDIOCTL_PREAM_EN | MDIOCTL_DIVISOR_VAL); | ||||||
|  | 
 | ||||||
|  | 	if (pi->sih->buscorerev >= 10) { | ||||||
|  | 		/* new serdes is slower in rw,
 | ||||||
|  | 		 * using two layers of reg address mapping | ||||||
|  | 		 */ | ||||||
|  | 		if (!pcie_mdiosetblock(pi, physmedia)) | ||||||
|  | 			return 1; | ||||||
|  | 		mdiodata = ((MDIODATA_DEV_ADDR << MDIODATA_DEVADDR_SHF) | | ||||||
|  | 			    (regaddr << MDIODATA_REGADDR_SHF)); | ||||||
|  | 		pcie_serdes_spinwait *= 20; | ||||||
|  | 	} else { | ||||||
|  | 		mdiodata = ((physmedia << MDIODATA_DEVADDR_SHF_OLD) | | ||||||
|  | 			    (regaddr << MDIODATA_REGADDR_SHF_OLD)); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (!write) | ||||||
|  | 		mdiodata |= (MDIODATA_START | MDIODATA_READ | MDIODATA_TA); | ||||||
|  | 	else | ||||||
|  | 		mdiodata |= (MDIODATA_START | MDIODATA_WRITE | MDIODATA_TA | | ||||||
|  | 			     *val); | ||||||
|  | 
 | ||||||
|  | 	W_REG(&pcieregs->mdiodata, mdiodata); | ||||||
|  | 
 | ||||||
|  | 	pr28829_delay(); | ||||||
|  | 
 | ||||||
|  | 	/* retry till the transaction is complete */ | ||||||
|  | 	while (i < pcie_serdes_spinwait) { | ||||||
|  | 		if (R_REG(&pcieregs->mdiocontrol) & MDIOCTL_ACCESS_DONE) { | ||||||
|  | 			if (!write) { | ||||||
|  | 				pr28829_delay(); | ||||||
|  | 				*val = (R_REG(&pcieregs->mdiodata) & | ||||||
|  | 					MDIODATA_MASK); | ||||||
|  | 			} | ||||||
|  | 			/* Disable mdio access to SERDES */ | ||||||
|  | 			W_REG(&pcieregs->mdiocontrol, 0); | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  | 		udelay(1000); | ||||||
|  | 		i++; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Timed out. Disable mdio access to SERDES. */ | ||||||
|  | 	W_REG(&pcieregs->mdiocontrol, 0); | ||||||
|  | 	return 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* use the mdio interface to read from mdio slaves */ | ||||||
|  | static int | ||||||
|  | pcie_mdioread(struct pcicore_info *pi, uint physmedia, uint regaddr, | ||||||
|  | 	      uint *regval) | ||||||
|  | { | ||||||
|  | 	return pcie_mdioop(pi, physmedia, regaddr, false, regval); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* use the mdio interface to write to mdio slaves */ | ||||||
|  | static int | ||||||
|  | pcie_mdiowrite(struct pcicore_info *pi, uint physmedia, uint regaddr, uint val) | ||||||
|  | { | ||||||
|  | 	return pcie_mdioop(pi, physmedia, regaddr, true, &val); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* ***** Support functions ***** */ | ||||||
|  | static u8 pcie_clkreq(struct pcicore_info *pi, u32 mask, u32 val) | ||||||
|  | { | ||||||
|  | 	u32 reg_val; | ||||||
|  | 	u8 offset; | ||||||
|  | 
 | ||||||
|  | 	offset = pi->pciecap_lcreg_offset; | ||||||
|  | 	if (!offset) | ||||||
|  | 		return 0; | ||||||
|  | 
 | ||||||
|  | 	pci_read_config_dword(pi->dev, offset, ®_val); | ||||||
|  | 	/* set operation */ | ||||||
|  | 	if (mask) { | ||||||
|  | 		if (val) | ||||||
|  | 			reg_val |= PCIE_CLKREQ_ENAB; | ||||||
|  | 		else | ||||||
|  | 			reg_val &= ~PCIE_CLKREQ_ENAB; | ||||||
|  | 		pci_write_config_dword(pi->dev, offset, reg_val); | ||||||
|  | 		pci_read_config_dword(pi->dev, offset, ®_val); | ||||||
|  | 	} | ||||||
|  | 	if (reg_val & PCIE_CLKREQ_ENAB) | ||||||
|  | 		return 1; | ||||||
|  | 	else | ||||||
|  | 		return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void pcie_extendL1timer(struct pcicore_info *pi, bool extend) | ||||||
|  | { | ||||||
|  | 	u32 w; | ||||||
|  | 	struct si_pub *sih = pi->sih; | ||||||
|  | 	struct sbpcieregs __iomem *pcieregs = pi->regs.pcieregs; | ||||||
|  | 
 | ||||||
|  | 	if (sih->buscoretype != PCIE_CORE_ID || sih->buscorerev < 7) | ||||||
|  | 		return; | ||||||
|  | 
 | ||||||
|  | 	w = pcie_readreg(pcieregs, PCIE_PCIEREGS, PCIE_DLLP_PMTHRESHREG); | ||||||
|  | 	if (extend) | ||||||
|  | 		w |= PCIE_ASPMTIMER_EXTEND; | ||||||
|  | 	else | ||||||
|  | 		w &= ~PCIE_ASPMTIMER_EXTEND; | ||||||
|  | 	pcie_writereg(pcieregs, PCIE_PCIEREGS, PCIE_DLLP_PMTHRESHREG, w); | ||||||
|  | 	w = pcie_readreg(pcieregs, PCIE_PCIEREGS, PCIE_DLLP_PMTHRESHREG); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* centralized clkreq control policy */ | ||||||
|  | static void pcie_clkreq_upd(struct pcicore_info *pi, uint state) | ||||||
|  | { | ||||||
|  | 	struct si_pub *sih = pi->sih; | ||||||
|  | 
 | ||||||
|  | 	switch (state) { | ||||||
|  | 	case SI_DOATTACH: | ||||||
|  | 		if (PCIE_ASPM(sih)) | ||||||
|  | 			pcie_clkreq(pi, 1, 0); | ||||||
|  | 		break; | ||||||
|  | 	case SI_PCIDOWN: | ||||||
|  | 		if (sih->buscorerev == 6) {	/* turn on serdes PLL down */ | ||||||
|  | 			ai_corereg(sih, SI_CC_IDX, | ||||||
|  | 				   offsetof(struct chipcregs, chipcontrol_addr), | ||||||
|  | 				   ~0, 0); | ||||||
|  | 			ai_corereg(sih, SI_CC_IDX, | ||||||
|  | 				   offsetof(struct chipcregs, chipcontrol_data), | ||||||
|  | 				   ~0x40, 0); | ||||||
|  | 		} else if (pi->pcie_pr42767) { | ||||||
|  | 			pcie_clkreq(pi, 1, 1); | ||||||
|  | 		} | ||||||
|  | 		break; | ||||||
|  | 	case SI_PCIUP: | ||||||
|  | 		if (sih->buscorerev == 6) {	/* turn off serdes PLL down */ | ||||||
|  | 			ai_corereg(sih, SI_CC_IDX, | ||||||
|  | 				   offsetof(struct chipcregs, chipcontrol_addr), | ||||||
|  | 				   ~0, 0); | ||||||
|  | 			ai_corereg(sih, SI_CC_IDX, | ||||||
|  | 				   offsetof(struct chipcregs, chipcontrol_data), | ||||||
|  | 				   ~0x40, 0x40); | ||||||
|  | 		} else if (PCIE_ASPM(sih)) {	/* disable clkreq */ | ||||||
|  | 			pcie_clkreq(pi, 1, 0); | ||||||
|  | 		} | ||||||
|  | 		break; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* ***** PCI core WARs ***** */ | ||||||
|  | /* Done only once at attach time */ | ||||||
|  | static void pcie_war_polarity(struct pcicore_info *pi) | ||||||
|  | { | ||||||
|  | 	u32 w; | ||||||
|  | 
 | ||||||
|  | 	if (pi->pcie_polarity != 0) | ||||||
|  | 		return; | ||||||
|  | 
 | ||||||
|  | 	w = pcie_readreg(pi->regs.pcieregs, PCIE_PCIEREGS, PCIE_PLP_STATUSREG); | ||||||
|  | 
 | ||||||
|  | 	/* Detect the current polarity at attach and force that polarity and
 | ||||||
|  | 	 * disable changing the polarity | ||||||
|  | 	 */ | ||||||
|  | 	if ((w & PCIE_PLP_POLARITYINV_STAT) == 0) | ||||||
|  | 		pi->pcie_polarity = SERDES_RX_CTRL_FORCE; | ||||||
|  | 	else | ||||||
|  | 		pi->pcie_polarity = (SERDES_RX_CTRL_FORCE | | ||||||
|  | 				     SERDES_RX_CTRL_POLARITY); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* enable ASPM and CLKREQ if srom doesn't have it */ | ||||||
|  | /* Needs to happen when update to shadow SROM is needed
 | ||||||
|  |  *   : Coming out of 'standby'/'hibernate' | ||||||
|  |  *   : If pcie_war_aspm_ovr state changed | ||||||
|  |  */ | ||||||
|  | static void pcie_war_aspm_clkreq(struct pcicore_info *pi) | ||||||
|  | { | ||||||
|  | 	struct sbpcieregs __iomem *pcieregs = pi->regs.pcieregs; | ||||||
|  | 	struct si_pub *sih = pi->sih; | ||||||
|  | 	u16 val16; | ||||||
|  | 	u16 __iomem *reg16; | ||||||
|  | 	u32 w; | ||||||
|  | 
 | ||||||
|  | 	if (!PCIE_ASPM(sih)) | ||||||
|  | 		return; | ||||||
|  | 
 | ||||||
|  | 	/* bypass this on QT or VSIM */ | ||||||
|  | 	reg16 = &pcieregs->sprom[SRSH_ASPM_OFFSET]; | ||||||
|  | 	val16 = R_REG(reg16); | ||||||
|  | 
 | ||||||
|  | 	val16 &= ~SRSH_ASPM_ENB; | ||||||
|  | 	if (pi->pcie_war_aspm_ovr == PCIE_ASPM_ENAB) | ||||||
|  | 		val16 |= SRSH_ASPM_ENB; | ||||||
|  | 	else if (pi->pcie_war_aspm_ovr == PCIE_ASPM_L1_ENAB) | ||||||
|  | 		val16 |= SRSH_ASPM_L1_ENB; | ||||||
|  | 	else if (pi->pcie_war_aspm_ovr == PCIE_ASPM_L0s_ENAB) | ||||||
|  | 		val16 |= SRSH_ASPM_L0s_ENB; | ||||||
|  | 
 | ||||||
|  | 	W_REG(reg16, val16); | ||||||
|  | 
 | ||||||
|  | 	pci_read_config_dword(pi->dev, pi->pciecap_lcreg_offset, &w); | ||||||
|  | 	w &= ~PCIE_ASPM_ENAB; | ||||||
|  | 	w |= pi->pcie_war_aspm_ovr; | ||||||
|  | 	pci_write_config_dword(pi->dev, pi->pciecap_lcreg_offset, w); | ||||||
|  | 
 | ||||||
|  | 	reg16 = &pcieregs->sprom[SRSH_CLKREQ_OFFSET_REV5]; | ||||||
|  | 	val16 = R_REG(reg16); | ||||||
|  | 
 | ||||||
|  | 	if (pi->pcie_war_aspm_ovr != PCIE_ASPM_DISAB) { | ||||||
|  | 		val16 |= SRSH_CLKREQ_ENB; | ||||||
|  | 		pi->pcie_pr42767 = true; | ||||||
|  | 	} else | ||||||
|  | 		val16 &= ~SRSH_CLKREQ_ENB; | ||||||
|  | 
 | ||||||
|  | 	W_REG(reg16, val16); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Apply the polarity determined at the start */ | ||||||
|  | /* Needs to happen when coming out of 'standby'/'hibernate' */ | ||||||
|  | static void pcie_war_serdes(struct pcicore_info *pi) | ||||||
|  | { | ||||||
|  | 	u32 w = 0; | ||||||
|  | 
 | ||||||
|  | 	if (pi->pcie_polarity != 0) | ||||||
|  | 		pcie_mdiowrite(pi, MDIODATA_DEV_RX, SERDES_RX_CTRL, | ||||||
|  | 			       pi->pcie_polarity); | ||||||
|  | 
 | ||||||
|  | 	pcie_mdioread(pi, MDIODATA_DEV_PLL, SERDES_PLL_CTRL, &w); | ||||||
|  | 	if (w & PLL_CTRL_FREQDET_EN) { | ||||||
|  | 		w &= ~PLL_CTRL_FREQDET_EN; | ||||||
|  | 		pcie_mdiowrite(pi, MDIODATA_DEV_PLL, SERDES_PLL_CTRL, w); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Fix MISC config to allow coming out of L2/L3-Ready state w/o PRST */ | ||||||
|  | /* Needs to happen when coming out of 'standby'/'hibernate' */ | ||||||
|  | static void pcie_misc_config_fixup(struct pcicore_info *pi) | ||||||
|  | { | ||||||
|  | 	struct sbpcieregs __iomem *pcieregs = pi->regs.pcieregs; | ||||||
|  | 	u16 val16; | ||||||
|  | 	u16 __iomem *reg16; | ||||||
|  | 
 | ||||||
|  | 	reg16 = &pcieregs->sprom[SRSH_PCIE_MISC_CONFIG]; | ||||||
|  | 	val16 = R_REG(reg16); | ||||||
|  | 
 | ||||||
|  | 	if ((val16 & SRSH_L23READY_EXIT_NOPERST) == 0) { | ||||||
|  | 		val16 |= SRSH_L23READY_EXIT_NOPERST; | ||||||
|  | 		W_REG(reg16, val16); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* quick hack for testing */ | ||||||
|  | /* Needs to happen when coming out of 'standby'/'hibernate' */ | ||||||
|  | static void pcie_war_noplldown(struct pcicore_info *pi) | ||||||
|  | { | ||||||
|  | 	struct sbpcieregs __iomem *pcieregs = pi->regs.pcieregs; | ||||||
|  | 	u16 __iomem *reg16; | ||||||
|  | 
 | ||||||
|  | 	/* turn off serdes PLL down */ | ||||||
|  | 	ai_corereg(pi->sih, SI_CC_IDX, offsetof(struct chipcregs, chipcontrol), | ||||||
|  | 		   CHIPCTRL_4321_PLL_DOWN, CHIPCTRL_4321_PLL_DOWN); | ||||||
|  | 
 | ||||||
|  | 	/* clear srom shadow backdoor */ | ||||||
|  | 	reg16 = &pcieregs->sprom[SRSH_BD_OFFSET]; | ||||||
|  | 	W_REG(reg16, 0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Needs to happen when coming out of 'standby'/'hibernate' */ | ||||||
|  | static void pcie_war_pci_setup(struct pcicore_info *pi) | ||||||
|  | { | ||||||
|  | 	struct si_pub *sih = pi->sih; | ||||||
|  | 	struct sbpcieregs __iomem *pcieregs = pi->regs.pcieregs; | ||||||
|  | 	u32 w; | ||||||
|  | 
 | ||||||
|  | 	if (sih->buscorerev == 0 || sih->buscorerev == 1) { | ||||||
|  | 		w = pcie_readreg(pcieregs, PCIE_PCIEREGS, | ||||||
|  | 				 PCIE_TLP_WORKAROUNDSREG); | ||||||
|  | 		w |= 0x8; | ||||||
|  | 		pcie_writereg(pcieregs, PCIE_PCIEREGS, | ||||||
|  | 			      PCIE_TLP_WORKAROUNDSREG, w); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (sih->buscorerev == 1) { | ||||||
|  | 		w = pcie_readreg(pcieregs, PCIE_PCIEREGS, PCIE_DLLP_LCREG); | ||||||
|  | 		w |= 0x40; | ||||||
|  | 		pcie_writereg(pcieregs, PCIE_PCIEREGS, PCIE_DLLP_LCREG, w); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (sih->buscorerev == 0) { | ||||||
|  | 		pcie_mdiowrite(pi, MDIODATA_DEV_RX, SERDES_RX_TIMER1, 0x8128); | ||||||
|  | 		pcie_mdiowrite(pi, MDIODATA_DEV_RX, SERDES_RX_CDR, 0x0100); | ||||||
|  | 		pcie_mdiowrite(pi, MDIODATA_DEV_RX, SERDES_RX_CDRBW, 0x1466); | ||||||
|  | 	} else if (PCIE_ASPM(sih)) { | ||||||
|  | 		/* Change the L1 threshold for better performance */ | ||||||
|  | 		w = pcie_readreg(pcieregs, PCIE_PCIEREGS, | ||||||
|  | 				 PCIE_DLLP_PMTHRESHREG); | ||||||
|  | 		w &= ~PCIE_L1THRESHOLDTIME_MASK; | ||||||
|  | 		w |= PCIE_L1THRESHOLD_WARVAL << PCIE_L1THRESHOLDTIME_SHIFT; | ||||||
|  | 		pcie_writereg(pcieregs, PCIE_PCIEREGS, | ||||||
|  | 			      PCIE_DLLP_PMTHRESHREG, w); | ||||||
|  | 
 | ||||||
|  | 		pcie_war_serdes(pi); | ||||||
|  | 
 | ||||||
|  | 		pcie_war_aspm_clkreq(pi); | ||||||
|  | 	} else if (pi->sih->buscorerev == 7) | ||||||
|  | 		pcie_war_noplldown(pi); | ||||||
|  | 
 | ||||||
|  | 	/* Note that the fix is actually in the SROM,
 | ||||||
|  | 	 * that's why this is open-ended | ||||||
|  | 	 */ | ||||||
|  | 	if (pi->sih->buscorerev >= 6) | ||||||
|  | 		pcie_misc_config_fixup(pi); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* ***** Functions called during driver state changes ***** */ | ||||||
|  | void pcicore_attach(struct pcicore_info *pi, int state) | ||||||
|  | { | ||||||
|  | 	struct si_pub *sih = pi->sih; | ||||||
|  | 	u32 bfl2 = (u32)getintvar(sih, BRCMS_SROM_BOARDFLAGS2); | ||||||
|  | 
 | ||||||
|  | 	/* Determine if this board needs override */ | ||||||
|  | 	if (PCIE_ASPM(sih)) { | ||||||
|  | 		if (bfl2 & BFL2_PCIEWAR_OVR) | ||||||
|  | 			pi->pcie_war_aspm_ovr = PCIE_ASPM_DISAB; | ||||||
|  | 		else | ||||||
|  | 			pi->pcie_war_aspm_ovr = PCIE_ASPM_ENAB; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* These need to happen in this order only */ | ||||||
|  | 	pcie_war_polarity(pi); | ||||||
|  | 
 | ||||||
|  | 	pcie_war_serdes(pi); | ||||||
|  | 
 | ||||||
|  | 	pcie_war_aspm_clkreq(pi); | ||||||
|  | 
 | ||||||
|  | 	pcie_clkreq_upd(pi, state); | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void pcicore_hwup(struct pcicore_info *pi) | ||||||
|  | { | ||||||
|  | 	if (!pi || pi->sih->buscoretype != PCIE_CORE_ID) | ||||||
|  | 		return; | ||||||
|  | 
 | ||||||
|  | 	pcie_war_pci_setup(pi); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void pcicore_up(struct pcicore_info *pi, int state) | ||||||
|  | { | ||||||
|  | 	if (!pi || pi->sih->buscoretype != PCIE_CORE_ID) | ||||||
|  | 		return; | ||||||
|  | 
 | ||||||
|  | 	/* Restore L1 timer for better performance */ | ||||||
|  | 	pcie_extendL1timer(pi, true); | ||||||
|  | 
 | ||||||
|  | 	pcie_clkreq_upd(pi, state); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* When the device is going to enter D3 state
 | ||||||
|  |  * (or the system is going to enter S3/S4 states) | ||||||
|  |  */ | ||||||
|  | void pcicore_sleep(struct pcicore_info *pi) | ||||||
|  | { | ||||||
|  | 	u32 w; | ||||||
|  | 
 | ||||||
|  | 	if (!pi || !PCIE_ASPM(pi->sih)) | ||||||
|  | 		return; | ||||||
|  | 
 | ||||||
|  | 	pci_read_config_dword(pi->dev, pi->pciecap_lcreg_offset, &w); | ||||||
|  | 	w &= ~PCIE_CAP_LCREG_ASPML1; | ||||||
|  | 	pci_write_config_dword(pi->dev, pi->pciecap_lcreg_offset, w); | ||||||
|  | 
 | ||||||
|  | 	pi->pcie_pr42767 = false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void pcicore_down(struct pcicore_info *pi, int state) | ||||||
|  | { | ||||||
|  | 	if (!pi || pi->sih->buscoretype != PCIE_CORE_ID) | ||||||
|  | 		return; | ||||||
|  | 
 | ||||||
|  | 	pcie_clkreq_upd(pi, state); | ||||||
|  | 
 | ||||||
|  | 	/* Reduce L1 timer for better power savings */ | ||||||
|  | 	pcie_extendL1timer(pi, false); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* precondition: current core is sii->buscoretype */ | ||||||
|  | static void pcicore_fixcfg(struct pcicore_info *pi, u16 __iomem *reg16) | ||||||
|  | { | ||||||
|  | 	struct si_info *sii = (struct si_info *)(pi->sih); | ||||||
|  | 	u16 val16; | ||||||
|  | 	uint pciidx; | ||||||
|  | 
 | ||||||
|  | 	pciidx = ai_coreidx(&sii->pub); | ||||||
|  | 	val16 = R_REG(reg16); | ||||||
|  | 	if (((val16 & SRSH_PI_MASK) >> SRSH_PI_SHIFT) != (u16)pciidx) { | ||||||
|  | 		val16 = (u16)(pciidx << SRSH_PI_SHIFT) | | ||||||
|  | 			(val16 & ~SRSH_PI_MASK); | ||||||
|  | 		W_REG(reg16, val16); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | pcicore_fixcfg_pci(struct pcicore_info *pi, struct sbpciregs __iomem *pciregs) | ||||||
|  | { | ||||||
|  | 	pcicore_fixcfg(pi, &pciregs->sprom[SRSH_PI_OFFSET]); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void pcicore_fixcfg_pcie(struct pcicore_info *pi, | ||||||
|  | 			 struct sbpcieregs __iomem *pcieregs) | ||||||
|  | { | ||||||
|  | 	pcicore_fixcfg(pi, &pcieregs->sprom[SRSH_PI_OFFSET]); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* precondition: current core is pci core */ | ||||||
|  | void | ||||||
|  | pcicore_pci_setup(struct pcicore_info *pi, struct sbpciregs __iomem *pciregs) | ||||||
|  | { | ||||||
|  | 	u32 w; | ||||||
|  | 
 | ||||||
|  | 	OR_REG(&pciregs->sbtopci2, SBTOPCI_PREF | SBTOPCI_BURST); | ||||||
|  | 
 | ||||||
|  | 	if (((struct si_info *)(pi->sih))->pub.buscorerev >= 11) { | ||||||
|  | 		OR_REG(&pciregs->sbtopci2, SBTOPCI_RC_READMULTI); | ||||||
|  | 		w = R_REG(&pciregs->clkrun); | ||||||
|  | 		W_REG(&pciregs->clkrun, w | PCI_CLKRUN_DSBL); | ||||||
|  | 		w = R_REG(&pciregs->clkrun); | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										82
									
								
								drivers/net/wireless/brcm80211/brcmsmac/nicpci.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								drivers/net/wireless/brcm80211/brcmsmac/nicpci.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,82 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef	_BRCM_NICPCI_H_ | ||||||
|  | #define	_BRCM_NICPCI_H_ | ||||||
|  | 
 | ||||||
|  | #include "types.h" | ||||||
|  | 
 | ||||||
|  | /* PCI configuration address space size */ | ||||||
|  | #define PCI_SZPCR		256 | ||||||
|  | 
 | ||||||
|  | /* Brcm PCI configuration registers */ | ||||||
|  | /* backplane address space accessed by BAR0 */ | ||||||
|  | #define PCI_BAR0_WIN		0x80 | ||||||
|  | /* sprom property control */ | ||||||
|  | #define PCI_SPROM_CONTROL	0x88 | ||||||
|  | /* mask of PCI and other cores interrupts */ | ||||||
|  | #define PCI_INT_MASK		0x94 | ||||||
|  | /* backplane core interrupt mask bits offset */ | ||||||
|  | #define  PCI_SBIM_SHIFT		8 | ||||||
|  | /* backplane address space accessed by second 4KB of BAR0 */ | ||||||
|  | #define PCI_BAR0_WIN2		0xac | ||||||
|  | /* pci config space gpio input (>=rev3) */ | ||||||
|  | #define PCI_GPIO_IN		0xb0 | ||||||
|  | /* pci config space gpio output (>=rev3) */ | ||||||
|  | #define PCI_GPIO_OUT		0xb4 | ||||||
|  | /* pci config space gpio output enable (>=rev3) */ | ||||||
|  | #define PCI_GPIO_OUTEN		0xb8 | ||||||
|  | 
 | ||||||
|  | /* bar0 + 4K accesses external sprom */ | ||||||
|  | #define PCI_BAR0_SPROM_OFFSET	(4 * 1024) | ||||||
|  | /* bar0 + 6K accesses pci core registers */ | ||||||
|  | #define PCI_BAR0_PCIREGS_OFFSET	(6 * 1024) | ||||||
|  | /*
 | ||||||
|  |  * pci core SB registers are at the end of the | ||||||
|  |  * 8KB window, so their address is the "regular" | ||||||
|  |  * address plus 4K | ||||||
|  |  */ | ||||||
|  | #define PCI_BAR0_PCISBR_OFFSET	(4 * 1024) | ||||||
|  | /* bar0 window size Match with corerev 13 */ | ||||||
|  | #define PCI_BAR0_WINSZ		(16 * 1024) | ||||||
|  | /* On pci corerev >= 13 and all pcie, the bar0 is now 16KB and it maps: */ | ||||||
|  | /* bar0 + 8K accesses pci/pcie core registers */ | ||||||
|  | #define PCI_16KB0_PCIREGS_OFFSET (8 * 1024) | ||||||
|  | /* bar0 + 12K accesses chipc core registers */ | ||||||
|  | #define PCI_16KB0_CCREGS_OFFSET	(12 * 1024) | ||||||
|  | 
 | ||||||
|  | struct sbpciregs; | ||||||
|  | struct sbpcieregs; | ||||||
|  | 
 | ||||||
|  | extern struct pcicore_info *pcicore_init(struct si_pub *sih, | ||||||
|  | 					 struct pci_dev *pdev, | ||||||
|  | 					 void __iomem *regs); | ||||||
|  | extern void pcicore_deinit(struct pcicore_info *pch); | ||||||
|  | extern void pcicore_attach(struct pcicore_info *pch, int state); | ||||||
|  | extern void pcicore_hwup(struct pcicore_info *pch); | ||||||
|  | extern void pcicore_up(struct pcicore_info *pch, int state); | ||||||
|  | extern void pcicore_sleep(struct pcicore_info *pch); | ||||||
|  | extern void pcicore_down(struct pcicore_info *pch, int state); | ||||||
|  | extern u8 pcicore_find_pci_capability(struct pci_dev *dev, u8 req_cap_id, | ||||||
|  | 				      unsigned char *buf, u32 *buflen); | ||||||
|  | extern void pcicore_fixcfg_pci(struct pcicore_info *pch, | ||||||
|  | 			       struct sbpciregs __iomem *pciregs); | ||||||
|  | extern void pcicore_fixcfg_pcie(struct pcicore_info *pch, | ||||||
|  | 				struct sbpcieregs __iomem *pciregs); | ||||||
|  | extern void pcicore_pci_setup(struct pcicore_info *pch, | ||||||
|  | 			      struct sbpciregs __iomem *pciregs); | ||||||
|  | 
 | ||||||
|  | #endif /* _BRCM_NICPCI_H_ */ | ||||||
							
								
								
									
										426
									
								
								drivers/net/wireless/brcm80211/brcmsmac/otp.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										426
									
								
								drivers/net/wireless/brcm80211/brcmsmac/otp.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,426 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <linux/io.h> | ||||||
|  | #include <linux/errno.h> | ||||||
|  | #include <linux/string.h> | ||||||
|  | 
 | ||||||
|  | #include <brcm_hw_ids.h> | ||||||
|  | #include <chipcommon.h> | ||||||
|  | #include "aiutils.h" | ||||||
|  | #include "otp.h" | ||||||
|  | 
 | ||||||
|  | #define OTPS_GUP_MASK		0x00000f00 | ||||||
|  | #define OTPS_GUP_SHIFT		8 | ||||||
|  | /* h/w subregion is programmed */ | ||||||
|  | #define OTPS_GUP_HW		0x00000100 | ||||||
|  | /* s/w subregion is programmed */ | ||||||
|  | #define OTPS_GUP_SW		0x00000200 | ||||||
|  | /* chipid/pkgopt subregion is programmed */ | ||||||
|  | #define OTPS_GUP_CI		0x00000400 | ||||||
|  | /* fuse subregion is programmed */ | ||||||
|  | #define OTPS_GUP_FUSE		0x00000800 | ||||||
|  | 
 | ||||||
|  | /* Fields in otpprog in rev >= 21 */ | ||||||
|  | #define OTPP_COL_MASK		0x000000ff | ||||||
|  | #define OTPP_COL_SHIFT		0 | ||||||
|  | #define OTPP_ROW_MASK		0x0000ff00 | ||||||
|  | #define OTPP_ROW_SHIFT		8 | ||||||
|  | #define OTPP_OC_MASK		0x0f000000 | ||||||
|  | #define OTPP_OC_SHIFT		24 | ||||||
|  | #define OTPP_READERR		0x10000000 | ||||||
|  | #define OTPP_VALUE_MASK		0x20000000 | ||||||
|  | #define OTPP_VALUE_SHIFT	29 | ||||||
|  | #define OTPP_START_BUSY		0x80000000 | ||||||
|  | #define	OTPP_READ		0x40000000 | ||||||
|  | 
 | ||||||
|  | /* Opcodes for OTPP_OC field */ | ||||||
|  | #define OTPPOC_READ		0 | ||||||
|  | #define OTPPOC_BIT_PROG		1 | ||||||
|  | #define OTPPOC_VERIFY		3 | ||||||
|  | #define OTPPOC_INIT		4 | ||||||
|  | #define OTPPOC_SET		5 | ||||||
|  | #define OTPPOC_RESET		6 | ||||||
|  | #define OTPPOC_OCST		7 | ||||||
|  | #define OTPPOC_ROW_LOCK		8 | ||||||
|  | #define OTPPOC_PRESCN_TEST	9 | ||||||
|  | 
 | ||||||
|  | #define OTPTYPE_IPX(ccrev)	((ccrev) == 21 || (ccrev) >= 23) | ||||||
|  | 
 | ||||||
|  | #define OTPP_TRIES	10000000	/* # of tries for OTPP */ | ||||||
|  | 
 | ||||||
|  | #define MAXNUMRDES		9	/* Maximum OTP redundancy entries */ | ||||||
|  | 
 | ||||||
|  | /* Fixed size subregions sizes in words */ | ||||||
|  | #define OTPGU_CI_SZ		2 | ||||||
|  | 
 | ||||||
|  | struct otpinfo; | ||||||
|  | 
 | ||||||
|  | /* OTP function struct */ | ||||||
|  | struct otp_fn_s { | ||||||
|  | 	int (*init)(struct si_pub *sih, struct otpinfo *oi); | ||||||
|  | 	int (*read_region)(struct otpinfo *oi, int region, u16 *data, | ||||||
|  | 			   uint *wlen); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct otpinfo { | ||||||
|  | 	uint ccrev;		/* chipc revision */ | ||||||
|  | 	const struct otp_fn_s *fn;	/* OTP functions */ | ||||||
|  | 	struct si_pub *sih;		/* Saved sb handle */ | ||||||
|  | 
 | ||||||
|  | 	/* IPX OTP section */ | ||||||
|  | 	u16 wsize;		/* Size of otp in words */ | ||||||
|  | 	u16 rows;		/* Geometry */ | ||||||
|  | 	u16 cols;		/* Geometry */ | ||||||
|  | 	u32 status;		/* Flag bits (lock/prog/rv).
 | ||||||
|  | 				 * (Reflected only when OTP is power cycled) | ||||||
|  | 				 */ | ||||||
|  | 	u16 hwbase;		/* hardware subregion offset */ | ||||||
|  | 	u16 hwlim;		/* hardware subregion boundary */ | ||||||
|  | 	u16 swbase;		/* software subregion offset */ | ||||||
|  | 	u16 swlim;		/* software subregion boundary */ | ||||||
|  | 	u16 fbase;		/* fuse subregion offset */ | ||||||
|  | 	u16 flim;		/* fuse subregion boundary */ | ||||||
|  | 	int otpgu_base;		/* offset to General Use Region */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* OTP layout */ | ||||||
|  | /* CC revs 21, 24 and 27 OTP General Use Region word offset */ | ||||||
|  | #define REVA4_OTPGU_BASE	12 | ||||||
|  | 
 | ||||||
|  | /* CC revs 23, 25, 26, 28 and above OTP General Use Region word offset */ | ||||||
|  | #define REVB8_OTPGU_BASE	20 | ||||||
|  | 
 | ||||||
|  | /* CC rev 36 OTP General Use Region word offset */ | ||||||
|  | #define REV36_OTPGU_BASE	12 | ||||||
|  | 
 | ||||||
|  | /* Subregion word offsets in General Use region */ | ||||||
|  | #define OTPGU_HSB_OFF		0 | ||||||
|  | #define OTPGU_SFB_OFF		1 | ||||||
|  | #define OTPGU_CI_OFF		2 | ||||||
|  | #define OTPGU_P_OFF		3 | ||||||
|  | #define OTPGU_SROM_OFF		4 | ||||||
|  | 
 | ||||||
|  | /* Flag bit offsets in General Use region  */ | ||||||
|  | #define OTPGU_HWP_OFF		60 | ||||||
|  | #define OTPGU_SWP_OFF		61 | ||||||
|  | #define OTPGU_CIP_OFF		62 | ||||||
|  | #define OTPGU_FUSEP_OFF		63 | ||||||
|  | #define OTPGU_CIP_MSK		0x4000 | ||||||
|  | #define OTPGU_P_MSK		0xf000 | ||||||
|  | #define OTPGU_P_SHIFT		(OTPGU_HWP_OFF % 16) | ||||||
|  | 
 | ||||||
|  | /* OTP Size */ | ||||||
|  | #define OTP_SZ_FU_324		((roundup(324, 8))/8)	/* 324 bits */ | ||||||
|  | #define OTP_SZ_FU_288		(288/8)	/* 288 bits */ | ||||||
|  | #define OTP_SZ_FU_216		(216/8)	/* 216 bits */ | ||||||
|  | #define OTP_SZ_FU_72		(72/8)	/* 72 bits */ | ||||||
|  | #define OTP_SZ_CHECKSUM		(16/8)	/* 16 bits */ | ||||||
|  | #define OTP4315_SWREG_SZ	178	/* 178 bytes */ | ||||||
|  | #define OTP_SZ_FU_144		(144/8)	/* 144 bits */ | ||||||
|  | 
 | ||||||
|  | static u16 | ||||||
|  | ipxotp_otpr(struct otpinfo *oi, struct chipcregs __iomem *cc, uint wn) | ||||||
|  | { | ||||||
|  | 	return R_REG(&cc->sromotp[wn]); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Calculate max HW/SW region byte size by subtracting fuse region | ||||||
|  |  * and checksum size, osizew is oi->wsize (OTP size - GU size) in words | ||||||
|  |  */ | ||||||
|  | static int ipxotp_max_rgnsz(struct si_pub *sih, int osizew) | ||||||
|  | { | ||||||
|  | 	int ret = 0; | ||||||
|  | 
 | ||||||
|  | 	switch (sih->chip) { | ||||||
|  | 	case BCM43224_CHIP_ID: | ||||||
|  | 	case BCM43225_CHIP_ID: | ||||||
|  | 		ret = osizew * 2 - OTP_SZ_FU_72 - OTP_SZ_CHECKSUM; | ||||||
|  | 		break; | ||||||
|  | 	case BCM4313_CHIP_ID: | ||||||
|  | 		ret = osizew * 2 - OTP_SZ_FU_72 - OTP_SZ_CHECKSUM; | ||||||
|  | 		break; | ||||||
|  | 	default: | ||||||
|  | 		break;	/* Don't know about this chip */ | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void _ipxotp_init(struct otpinfo *oi, struct chipcregs __iomem *cc) | ||||||
|  | { | ||||||
|  | 	uint k; | ||||||
|  | 	u32 otpp, st; | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * record word offset of General Use Region | ||||||
|  | 	 * for various chipcommon revs | ||||||
|  | 	 */ | ||||||
|  | 	if (oi->sih->ccrev == 21 || oi->sih->ccrev == 24 | ||||||
|  | 	    || oi->sih->ccrev == 27) { | ||||||
|  | 		oi->otpgu_base = REVA4_OTPGU_BASE; | ||||||
|  | 	} else if (oi->sih->ccrev == 36) { | ||||||
|  | 		/*
 | ||||||
|  | 		 * OTP size greater than equal to 2KB (128 words), | ||||||
|  | 		 * otpgu_base is similar to rev23 | ||||||
|  | 		 */ | ||||||
|  | 		if (oi->wsize >= 128) | ||||||
|  | 			oi->otpgu_base = REVB8_OTPGU_BASE; | ||||||
|  | 		else | ||||||
|  | 			oi->otpgu_base = REV36_OTPGU_BASE; | ||||||
|  | 	} else if (oi->sih->ccrev == 23 || oi->sih->ccrev >= 25) { | ||||||
|  | 		oi->otpgu_base = REVB8_OTPGU_BASE; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* First issue an init command so the status is up to date */ | ||||||
|  | 	otpp = | ||||||
|  | 	    OTPP_START_BUSY | ((OTPPOC_INIT << OTPP_OC_SHIFT) & OTPP_OC_MASK); | ||||||
|  | 
 | ||||||
|  | 	W_REG(&cc->otpprog, otpp); | ||||||
|  | 	for (k = 0; | ||||||
|  | 	     ((st = R_REG(&cc->otpprog)) & OTPP_START_BUSY) | ||||||
|  | 	     && (k < OTPP_TRIES); k++) | ||||||
|  | 		; | ||||||
|  | 	if (k >= OTPP_TRIES) | ||||||
|  | 		return; | ||||||
|  | 
 | ||||||
|  | 	/* Read OTP lock bits and subregion programmed indication bits */ | ||||||
|  | 	oi->status = R_REG(&cc->otpstatus); | ||||||
|  | 
 | ||||||
|  | 	if ((oi->sih->chip == BCM43224_CHIP_ID) | ||||||
|  | 	    || (oi->sih->chip == BCM43225_CHIP_ID)) { | ||||||
|  | 		u32 p_bits; | ||||||
|  | 		p_bits = | ||||||
|  | 		    (ipxotp_otpr(oi, cc, oi->otpgu_base + OTPGU_P_OFF) & | ||||||
|  | 		     OTPGU_P_MSK) | ||||||
|  | 		    >> OTPGU_P_SHIFT; | ||||||
|  | 		oi->status |= (p_bits << OTPS_GUP_SHIFT); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * h/w region base and fuse region limit are fixed to | ||||||
|  | 	 * the top and the bottom of the general use region. | ||||||
|  | 	 * Everything else can be flexible. | ||||||
|  | 	 */ | ||||||
|  | 	oi->hwbase = oi->otpgu_base + OTPGU_SROM_OFF; | ||||||
|  | 	oi->hwlim = oi->wsize; | ||||||
|  | 	if (oi->status & OTPS_GUP_HW) { | ||||||
|  | 		oi->hwlim = | ||||||
|  | 		    ipxotp_otpr(oi, cc, oi->otpgu_base + OTPGU_HSB_OFF) / 16; | ||||||
|  | 		oi->swbase = oi->hwlim; | ||||||
|  | 	} else | ||||||
|  | 		oi->swbase = oi->hwbase; | ||||||
|  | 
 | ||||||
|  | 	/* subtract fuse and checksum from beginning */ | ||||||
|  | 	oi->swlim = ipxotp_max_rgnsz(oi->sih, oi->wsize) / 2; | ||||||
|  | 
 | ||||||
|  | 	if (oi->status & OTPS_GUP_SW) { | ||||||
|  | 		oi->swlim = | ||||||
|  | 		    ipxotp_otpr(oi, cc, oi->otpgu_base + OTPGU_SFB_OFF) / 16; | ||||||
|  | 		oi->fbase = oi->swlim; | ||||||
|  | 	} else | ||||||
|  | 		oi->fbase = oi->swbase; | ||||||
|  | 
 | ||||||
|  | 	oi->flim = oi->wsize; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int ipxotp_init(struct si_pub *sih, struct otpinfo *oi) | ||||||
|  | { | ||||||
|  | 	uint idx; | ||||||
|  | 	struct chipcregs __iomem *cc; | ||||||
|  | 
 | ||||||
|  | 	/* Make sure we're running IPX OTP */ | ||||||
|  | 	if (!OTPTYPE_IPX(sih->ccrev)) | ||||||
|  | 		return -EBADE; | ||||||
|  | 
 | ||||||
|  | 	/* Make sure OTP is not disabled */ | ||||||
|  | 	if (ai_is_otp_disabled(sih)) | ||||||
|  | 		return -EBADE; | ||||||
|  | 
 | ||||||
|  | 	/* Check for otp size */ | ||||||
|  | 	switch ((sih->cccaps & CC_CAP_OTPSIZE) >> CC_CAP_OTPSIZE_SHIFT) { | ||||||
|  | 	case 0: | ||||||
|  | 		/* Nothing there */ | ||||||
|  | 		return -EBADE; | ||||||
|  | 	case 1:		/* 32x64 */ | ||||||
|  | 		oi->rows = 32; | ||||||
|  | 		oi->cols = 64; | ||||||
|  | 		oi->wsize = 128; | ||||||
|  | 		break; | ||||||
|  | 	case 2:		/* 64x64 */ | ||||||
|  | 		oi->rows = 64; | ||||||
|  | 		oi->cols = 64; | ||||||
|  | 		oi->wsize = 256; | ||||||
|  | 		break; | ||||||
|  | 	case 5:		/* 96x64 */ | ||||||
|  | 		oi->rows = 96; | ||||||
|  | 		oi->cols = 64; | ||||||
|  | 		oi->wsize = 384; | ||||||
|  | 		break; | ||||||
|  | 	case 7:		/* 16x64 *//* 1024 bits */ | ||||||
|  | 		oi->rows = 16; | ||||||
|  | 		oi->cols = 64; | ||||||
|  | 		oi->wsize = 64; | ||||||
|  | 		break; | ||||||
|  | 	default: | ||||||
|  | 		/* Don't know the geometry */ | ||||||
|  | 		return -EBADE; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Retrieve OTP region info */ | ||||||
|  | 	idx = ai_coreidx(sih); | ||||||
|  | 	cc = ai_setcoreidx(sih, SI_CC_IDX); | ||||||
|  | 
 | ||||||
|  | 	_ipxotp_init(oi, cc); | ||||||
|  | 
 | ||||||
|  | 	ai_setcoreidx(sih, idx); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | ipxotp_read_region(struct otpinfo *oi, int region, u16 *data, uint *wlen) | ||||||
|  | { | ||||||
|  | 	uint idx; | ||||||
|  | 	struct chipcregs __iomem *cc; | ||||||
|  | 	uint base, i, sz; | ||||||
|  | 
 | ||||||
|  | 	/* Validate region selection */ | ||||||
|  | 	switch (region) { | ||||||
|  | 	case OTP_HW_RGN: | ||||||
|  | 		sz = (uint) oi->hwlim - oi->hwbase; | ||||||
|  | 		if (!(oi->status & OTPS_GUP_HW)) { | ||||||
|  | 			*wlen = sz; | ||||||
|  | 			return -ENODATA; | ||||||
|  | 		} | ||||||
|  | 		if (*wlen < sz) { | ||||||
|  | 			*wlen = sz; | ||||||
|  | 			return -EOVERFLOW; | ||||||
|  | 		} | ||||||
|  | 		base = oi->hwbase; | ||||||
|  | 		break; | ||||||
|  | 	case OTP_SW_RGN: | ||||||
|  | 		sz = ((uint) oi->swlim - oi->swbase); | ||||||
|  | 		if (!(oi->status & OTPS_GUP_SW)) { | ||||||
|  | 			*wlen = sz; | ||||||
|  | 			return -ENODATA; | ||||||
|  | 		} | ||||||
|  | 		if (*wlen < sz) { | ||||||
|  | 			*wlen = sz; | ||||||
|  | 			return -EOVERFLOW; | ||||||
|  | 		} | ||||||
|  | 		base = oi->swbase; | ||||||
|  | 		break; | ||||||
|  | 	case OTP_CI_RGN: | ||||||
|  | 		sz = OTPGU_CI_SZ; | ||||||
|  | 		if (!(oi->status & OTPS_GUP_CI)) { | ||||||
|  | 			*wlen = sz; | ||||||
|  | 			return -ENODATA; | ||||||
|  | 		} | ||||||
|  | 		if (*wlen < sz) { | ||||||
|  | 			*wlen = sz; | ||||||
|  | 			return -EOVERFLOW; | ||||||
|  | 		} | ||||||
|  | 		base = oi->otpgu_base + OTPGU_CI_OFF; | ||||||
|  | 		break; | ||||||
|  | 	case OTP_FUSE_RGN: | ||||||
|  | 		sz = (uint) oi->flim - oi->fbase; | ||||||
|  | 		if (!(oi->status & OTPS_GUP_FUSE)) { | ||||||
|  | 			*wlen = sz; | ||||||
|  | 			return -ENODATA; | ||||||
|  | 		} | ||||||
|  | 		if (*wlen < sz) { | ||||||
|  | 			*wlen = sz; | ||||||
|  | 			return -EOVERFLOW; | ||||||
|  | 		} | ||||||
|  | 		base = oi->fbase; | ||||||
|  | 		break; | ||||||
|  | 	case OTP_ALL_RGN: | ||||||
|  | 		sz = ((uint) oi->flim - oi->hwbase); | ||||||
|  | 		if (!(oi->status & (OTPS_GUP_HW | OTPS_GUP_SW))) { | ||||||
|  | 			*wlen = sz; | ||||||
|  | 			return -ENODATA; | ||||||
|  | 		} | ||||||
|  | 		if (*wlen < sz) { | ||||||
|  | 			*wlen = sz; | ||||||
|  | 			return -EOVERFLOW; | ||||||
|  | 		} | ||||||
|  | 		base = oi->hwbase; | ||||||
|  | 		break; | ||||||
|  | 	default: | ||||||
|  | 		return -EINVAL; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	idx = ai_coreidx(oi->sih); | ||||||
|  | 	cc = ai_setcoreidx(oi->sih, SI_CC_IDX); | ||||||
|  | 
 | ||||||
|  | 	/* Read the data */ | ||||||
|  | 	for (i = 0; i < sz; i++) | ||||||
|  | 		data[i] = ipxotp_otpr(oi, cc, base + i); | ||||||
|  | 
 | ||||||
|  | 	ai_setcoreidx(oi->sih, idx); | ||||||
|  | 	*wlen = sz; | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static const struct otp_fn_s ipxotp_fn = { | ||||||
|  | 	(int (*)(struct si_pub *, struct otpinfo *)) ipxotp_init, | ||||||
|  | 	(int (*)(struct otpinfo *, int, u16 *, uint *)) ipxotp_read_region, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static int otp_init(struct si_pub *sih, struct otpinfo *oi) | ||||||
|  | { | ||||||
|  | 
 | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 	memset(oi, 0, sizeof(struct otpinfo)); | ||||||
|  | 
 | ||||||
|  | 	oi->ccrev = sih->ccrev; | ||||||
|  | 
 | ||||||
|  | 	if (OTPTYPE_IPX(oi->ccrev)) | ||||||
|  | 		oi->fn = &ipxotp_fn; | ||||||
|  | 
 | ||||||
|  | 	if (oi->fn == NULL) | ||||||
|  | 		return -EBADE; | ||||||
|  | 
 | ||||||
|  | 	oi->sih = sih; | ||||||
|  | 
 | ||||||
|  | 	ret = (oi->fn->init) (sih, oi); | ||||||
|  | 
 | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | otp_read_region(struct si_pub *sih, int region, u16 *data, uint *wlen) { | ||||||
|  | 	struct otpinfo otpinfo; | ||||||
|  | 	struct otpinfo *oi = &otpinfo; | ||||||
|  | 	int err = 0; | ||||||
|  | 
 | ||||||
|  | 	if (ai_is_otp_disabled(sih)) { | ||||||
|  | 		err = -EPERM; | ||||||
|  | 		goto out; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = otp_init(sih, oi); | ||||||
|  | 	if (err) | ||||||
|  | 		goto out; | ||||||
|  | 
 | ||||||
|  | 	err = ((oi)->fn->read_region)(oi, region, data, wlen); | ||||||
|  | 
 | ||||||
|  |  out: | ||||||
|  | 	return err; | ||||||
|  | } | ||||||
							
								
								
									
										36
									
								
								drivers/net/wireless/brcm80211/brcmsmac/otp.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								drivers/net/wireless/brcm80211/brcmsmac/otp.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,36 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef	_BRCM_OTP_H_ | ||||||
|  | #define	_BRCM_OTP_H_ | ||||||
|  | 
 | ||||||
|  | #include "types.h" | ||||||
|  | 
 | ||||||
|  | /* OTP regions */ | ||||||
|  | #define OTP_HW_RGN	1 | ||||||
|  | #define OTP_SW_RGN	2 | ||||||
|  | #define OTP_CI_RGN	4 | ||||||
|  | #define OTP_FUSE_RGN	8 | ||||||
|  | /* From h/w region to end of OTP including checksum */ | ||||||
|  | #define OTP_ALL_RGN	0xf | ||||||
|  | 
 | ||||||
|  | /* OTP Size */ | ||||||
|  | #define OTP_SZ_MAX		(6144/8)	/* maximum bytes in one CIS */ | ||||||
|  | 
 | ||||||
|  | extern int otp_read_region(struct si_pub *sih, int region, u16 *data, | ||||||
|  | 			   uint *wlen); | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCM_OTP_H_ */ | ||||||
							
								
								
									
										2988
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_cmn.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2988
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_cmn.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										301
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_hal.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										301
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_hal.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,301 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * phy_hal.h:  functionality exported from the phy to higher layers | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCM_PHY_HAL_H_ | ||||||
|  | #define _BRCM_PHY_HAL_H_ | ||||||
|  | 
 | ||||||
|  | #include <brcmu_utils.h> | ||||||
|  | #include <brcmu_wifi.h> | ||||||
|  | #include <phy_shim.h> | ||||||
|  | 
 | ||||||
|  | #define	IDCODE_VER_MASK		0x0000000f | ||||||
|  | #define	IDCODE_VER_SHIFT	0 | ||||||
|  | #define	IDCODE_MFG_MASK		0x00000fff | ||||||
|  | #define	IDCODE_MFG_SHIFT	0 | ||||||
|  | #define	IDCODE_ID_MASK		0x0ffff000 | ||||||
|  | #define	IDCODE_ID_SHIFT		12 | ||||||
|  | #define	IDCODE_REV_MASK		0xf0000000 | ||||||
|  | #define	IDCODE_REV_SHIFT	28 | ||||||
|  | 
 | ||||||
|  | #define	NORADIO_ID		0xe4f5 | ||||||
|  | #define	NORADIO_IDCODE		0x4e4f5246 | ||||||
|  | 
 | ||||||
|  | #define BCM2055_ID		0x2055 | ||||||
|  | #define BCM2055_IDCODE		0x02055000 | ||||||
|  | #define BCM2055A0_IDCODE	0x1205517f | ||||||
|  | 
 | ||||||
|  | #define BCM2056_ID		0x2056 | ||||||
|  | #define BCM2056_IDCODE		0x02056000 | ||||||
|  | #define BCM2056A0_IDCODE	0x1205617f | ||||||
|  | 
 | ||||||
|  | #define BCM2057_ID		0x2057 | ||||||
|  | #define BCM2057_IDCODE		0x02057000 | ||||||
|  | #define BCM2057A0_IDCODE	0x1205717f | ||||||
|  | 
 | ||||||
|  | #define BCM2064_ID		0x2064 | ||||||
|  | #define BCM2064_IDCODE		0x02064000 | ||||||
|  | #define BCM2064A0_IDCODE	0x0206417f | ||||||
|  | 
 | ||||||
|  | #define PHY_TPC_HW_OFF		false | ||||||
|  | #define PHY_TPC_HW_ON		true | ||||||
|  | 
 | ||||||
|  | #define PHY_PERICAL_DRIVERUP	1 | ||||||
|  | #define PHY_PERICAL_WATCHDOG	2 | ||||||
|  | #define PHY_PERICAL_PHYINIT	3 | ||||||
|  | #define PHY_PERICAL_JOIN_BSS	4 | ||||||
|  | #define PHY_PERICAL_START_IBSS	5 | ||||||
|  | #define PHY_PERICAL_UP_BSS	6 | ||||||
|  | #define PHY_PERICAL_CHAN	7 | ||||||
|  | #define PHY_FULLCAL	8 | ||||||
|  | 
 | ||||||
|  | #define PHY_PERICAL_DISABLE	0 | ||||||
|  | #define PHY_PERICAL_SPHASE	1 | ||||||
|  | #define PHY_PERICAL_MPHASE	2 | ||||||
|  | #define PHY_PERICAL_MANUAL	3 | ||||||
|  | 
 | ||||||
|  | #define PHY_HOLD_FOR_ASSOC	1 | ||||||
|  | #define PHY_HOLD_FOR_SCAN	2 | ||||||
|  | #define PHY_HOLD_FOR_RM		4 | ||||||
|  | #define PHY_HOLD_FOR_PLT	8 | ||||||
|  | #define PHY_HOLD_FOR_MUTE	16 | ||||||
|  | #define PHY_HOLD_FOR_NOT_ASSOC 0x20 | ||||||
|  | 
 | ||||||
|  | #define PHY_MUTE_FOR_PREISM	1 | ||||||
|  | #define PHY_MUTE_ALL		0xffffffff | ||||||
|  | 
 | ||||||
|  | #define PHY_NOISE_FIXED_VAL		(-95) | ||||||
|  | #define PHY_NOISE_FIXED_VAL_NPHY	(-92) | ||||||
|  | #define PHY_NOISE_FIXED_VAL_LCNPHY	(-92) | ||||||
|  | 
 | ||||||
|  | #define PHY_MODE_CAL		0x0002 | ||||||
|  | #define PHY_MODE_NOISEM		0x0004 | ||||||
|  | 
 | ||||||
|  | #define BRCMS_TXPWR_DB_FACTOR	4 | ||||||
|  | 
 | ||||||
|  | /* a large TX Power as an init value to factor out of min() calculations,
 | ||||||
|  |  * keep low enough to fit in an s8, units are .25 dBm | ||||||
|  |  */ | ||||||
|  | #define BRCMS_TXPWR_MAX		(127)	/* ~32 dBm = 1,500 mW */ | ||||||
|  | 
 | ||||||
|  | #define BRCMS_NUM_RATES_CCK           4 | ||||||
|  | #define BRCMS_NUM_RATES_OFDM          8 | ||||||
|  | #define BRCMS_NUM_RATES_MCS_1_STREAM  8 | ||||||
|  | #define BRCMS_NUM_RATES_MCS_2_STREAM  8 | ||||||
|  | #define BRCMS_NUM_RATES_MCS_3_STREAM  8 | ||||||
|  | #define BRCMS_NUM_RATES_MCS_4_STREAM  8 | ||||||
|  | 
 | ||||||
|  | #define	BRCMS_RSSI_INVALID	 0	/* invalid RSSI value */ | ||||||
|  | 
 | ||||||
|  | struct d11regs; | ||||||
|  | struct phy_shim_info; | ||||||
|  | 
 | ||||||
|  | struct txpwr_limits { | ||||||
|  | 	u8 cck[BRCMS_NUM_RATES_CCK]; | ||||||
|  | 	u8 ofdm[BRCMS_NUM_RATES_OFDM]; | ||||||
|  | 
 | ||||||
|  | 	u8 ofdm_cdd[BRCMS_NUM_RATES_OFDM]; | ||||||
|  | 
 | ||||||
|  | 	u8 ofdm_40_siso[BRCMS_NUM_RATES_OFDM]; | ||||||
|  | 	u8 ofdm_40_cdd[BRCMS_NUM_RATES_OFDM]; | ||||||
|  | 
 | ||||||
|  | 	u8 mcs_20_siso[BRCMS_NUM_RATES_MCS_1_STREAM]; | ||||||
|  | 	u8 mcs_20_cdd[BRCMS_NUM_RATES_MCS_1_STREAM]; | ||||||
|  | 	u8 mcs_20_stbc[BRCMS_NUM_RATES_MCS_1_STREAM]; | ||||||
|  | 	u8 mcs_20_mimo[BRCMS_NUM_RATES_MCS_2_STREAM]; | ||||||
|  | 
 | ||||||
|  | 	u8 mcs_40_siso[BRCMS_NUM_RATES_MCS_1_STREAM]; | ||||||
|  | 	u8 mcs_40_cdd[BRCMS_NUM_RATES_MCS_1_STREAM]; | ||||||
|  | 	u8 mcs_40_stbc[BRCMS_NUM_RATES_MCS_1_STREAM]; | ||||||
|  | 	u8 mcs_40_mimo[BRCMS_NUM_RATES_MCS_2_STREAM]; | ||||||
|  | 	u8 mcs32; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct tx_power { | ||||||
|  | 	u32 flags; | ||||||
|  | 	u16 chanspec;   /* txpwr report for this channel */ | ||||||
|  | 	u16 local_chanspec;     /* channel on which we are associated */ | ||||||
|  | 	u8 local_max;   /* local max according to the AP */ | ||||||
|  | 	u8 local_constraint;    /* local constraint according to the AP */ | ||||||
|  | 	s8 antgain[2];  /* Ant gain for each band - from SROM */ | ||||||
|  | 	u8 rf_cores;            /* count of RF Cores being reported */ | ||||||
|  | 	u8 est_Pout[4]; /* Latest tx power out estimate per RF chain */ | ||||||
|  | 	u8 est_Pout_act[4];     /* Latest tx power out estimate per RF chain
 | ||||||
|  | 				 * without adjustment */ | ||||||
|  | 	u8 est_Pout_cck;        /* Latest CCK tx power out estimate */ | ||||||
|  | 	u8 tx_power_max[4];     /* Maximum target power among all rates */ | ||||||
|  | 	/* Index of the rate with the max target power */ | ||||||
|  | 	u8 tx_power_max_rate_ind[4]; | ||||||
|  | 	/* User limit */ | ||||||
|  | 	u8 user_limit[WL_TX_POWER_RATES]; | ||||||
|  | 	/* Regulatory power limit */ | ||||||
|  | 	u8 reg_limit[WL_TX_POWER_RATES]; | ||||||
|  | 	/* Max power board can support (SROM) */ | ||||||
|  | 	u8 board_limit[WL_TX_POWER_RATES]; | ||||||
|  | 	/* Latest target power */ | ||||||
|  | 	u8 target[WL_TX_POWER_RATES]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct tx_inst_power { | ||||||
|  | 	u8 txpwr_est_Pout[2];   /* Latest estimate for 2.4 and 5 Ghz */ | ||||||
|  | 	u8 txpwr_est_Pout_gofdm;        /* Pwr estimate for 2.4 OFDM */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct brcms_chanvec { | ||||||
|  | 	u8 vec[MAXCHANNEL / NBBY]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct shared_phy_params { | ||||||
|  | 	struct si_pub *sih; | ||||||
|  | 	struct phy_shim_info *physhim; | ||||||
|  | 	uint unit; | ||||||
|  | 	uint corerev; | ||||||
|  | 	uint buscorerev; | ||||||
|  | 	u16 vid; | ||||||
|  | 	u16 did; | ||||||
|  | 	uint chip; | ||||||
|  | 	uint chiprev; | ||||||
|  | 	uint chippkg; | ||||||
|  | 	uint sromrev; | ||||||
|  | 	uint boardtype; | ||||||
|  | 	uint boardrev; | ||||||
|  | 	uint boardvendor; | ||||||
|  | 	u32 boardflags; | ||||||
|  | 	u32 boardflags2; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | extern struct shared_phy *wlc_phy_shared_attach(struct shared_phy_params *shp); | ||||||
|  | extern struct brcms_phy_pub *wlc_phy_attach(struct shared_phy *sh, | ||||||
|  | 					    struct d11regs __iomem *regs, | ||||||
|  | 					    int bandtype, struct wiphy *wiphy); | ||||||
|  | extern void wlc_phy_detach(struct brcms_phy_pub *ppi); | ||||||
|  | 
 | ||||||
|  | extern bool wlc_phy_get_phyversion(struct brcms_phy_pub *pih, u16 *phytype, | ||||||
|  | 				   u16 *phyrev, u16 *radioid, | ||||||
|  | 				   u16 *radiover); | ||||||
|  | extern bool wlc_phy_get_encore(struct brcms_phy_pub *pih); | ||||||
|  | extern u32 wlc_phy_get_coreflags(struct brcms_phy_pub *pih); | ||||||
|  | 
 | ||||||
|  | extern void wlc_phy_hw_clk_state_upd(struct brcms_phy_pub *ppi, bool newstate); | ||||||
|  | extern void wlc_phy_hw_state_upd(struct brcms_phy_pub *ppi, bool newstate); | ||||||
|  | extern void wlc_phy_init(struct brcms_phy_pub *ppi, u16 chanspec); | ||||||
|  | extern void wlc_phy_watchdog(struct brcms_phy_pub *ppi); | ||||||
|  | extern int wlc_phy_down(struct brcms_phy_pub *ppi); | ||||||
|  | extern u32 wlc_phy_clk_bwbits(struct brcms_phy_pub *pih); | ||||||
|  | extern void wlc_phy_cal_init(struct brcms_phy_pub *ppi); | ||||||
|  | extern void wlc_phy_antsel_init(struct brcms_phy_pub *ppi, bool lut_init); | ||||||
|  | 
 | ||||||
|  | extern void wlc_phy_chanspec_set(struct brcms_phy_pub *ppi, | ||||||
|  | 				 u16 chanspec); | ||||||
|  | extern u16 wlc_phy_chanspec_get(struct brcms_phy_pub *ppi); | ||||||
|  | extern void wlc_phy_chanspec_radio_set(struct brcms_phy_pub *ppi, | ||||||
|  | 				       u16 newch); | ||||||
|  | extern u16 wlc_phy_bw_state_get(struct brcms_phy_pub *ppi); | ||||||
|  | extern void wlc_phy_bw_state_set(struct brcms_phy_pub *ppi, u16 bw); | ||||||
|  | 
 | ||||||
|  | extern int wlc_phy_rssi_compute(struct brcms_phy_pub *pih, | ||||||
|  | 				struct d11rxhdr *rxh); | ||||||
|  | extern void wlc_phy_por_inform(struct brcms_phy_pub *ppi); | ||||||
|  | extern void wlc_phy_noise_sample_intr(struct brcms_phy_pub *ppi); | ||||||
|  | extern bool wlc_phy_bist_check_phy(struct brcms_phy_pub *ppi); | ||||||
|  | 
 | ||||||
|  | extern void wlc_phy_set_deaf(struct brcms_phy_pub *ppi, bool user_flag); | ||||||
|  | 
 | ||||||
|  | extern void wlc_phy_switch_radio(struct brcms_phy_pub *ppi, bool on); | ||||||
|  | extern void wlc_phy_anacore(struct brcms_phy_pub *ppi, bool on); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | extern void wlc_phy_BSSinit(struct brcms_phy_pub *ppi, bool bonlyap, int rssi); | ||||||
|  | 
 | ||||||
|  | extern void wlc_phy_chanspec_ch14_widefilter_set(struct brcms_phy_pub *ppi, | ||||||
|  | 						 bool wide_filter); | ||||||
|  | extern void wlc_phy_chanspec_band_validch(struct brcms_phy_pub *ppi, uint band, | ||||||
|  | 					  struct brcms_chanvec *channels); | ||||||
|  | extern u16 wlc_phy_chanspec_band_firstch(struct brcms_phy_pub *ppi, | ||||||
|  | 					 uint band); | ||||||
|  | 
 | ||||||
|  | extern void wlc_phy_txpower_sromlimit(struct brcms_phy_pub *ppi, uint chan, | ||||||
|  | 				      u8 *_min_, u8 *_max_, int rate); | ||||||
|  | extern void wlc_phy_txpower_sromlimit_max_get(struct brcms_phy_pub *ppi, | ||||||
|  | 					      uint chan, u8 *_max_, u8 *_min_); | ||||||
|  | extern void wlc_phy_txpower_boardlimit_band(struct brcms_phy_pub *ppi, | ||||||
|  | 					    uint band, s32 *, s32 *, u32 *); | ||||||
|  | extern void wlc_phy_txpower_limit_set(struct brcms_phy_pub *ppi, | ||||||
|  | 				      struct txpwr_limits *, | ||||||
|  | 				      u16 chanspec); | ||||||
|  | extern int wlc_phy_txpower_get(struct brcms_phy_pub *ppi, uint *qdbm, | ||||||
|  | 			       bool *override); | ||||||
|  | extern int wlc_phy_txpower_set(struct brcms_phy_pub *ppi, uint qdbm, | ||||||
|  | 			       bool override); | ||||||
|  | extern void wlc_phy_txpower_target_set(struct brcms_phy_pub *ppi, | ||||||
|  | 				       struct txpwr_limits *); | ||||||
|  | extern bool wlc_phy_txpower_hw_ctrl_get(struct brcms_phy_pub *ppi); | ||||||
|  | extern void wlc_phy_txpower_hw_ctrl_set(struct brcms_phy_pub *ppi, | ||||||
|  | 					bool hwpwrctrl); | ||||||
|  | extern u8 wlc_phy_txpower_get_target_min(struct brcms_phy_pub *ppi); | ||||||
|  | extern u8 wlc_phy_txpower_get_target_max(struct brcms_phy_pub *ppi); | ||||||
|  | extern bool wlc_phy_txpower_ipa_ison(struct brcms_phy_pub *pih); | ||||||
|  | 
 | ||||||
|  | extern void wlc_phy_stf_chain_init(struct brcms_phy_pub *pih, u8 txchain, | ||||||
|  | 				   u8 rxchain); | ||||||
|  | extern void wlc_phy_stf_chain_set(struct brcms_phy_pub *pih, u8 txchain, | ||||||
|  | 				  u8 rxchain); | ||||||
|  | extern void wlc_phy_stf_chain_get(struct brcms_phy_pub *pih, u8 *txchain, | ||||||
|  | 				  u8 *rxchain); | ||||||
|  | extern u8 wlc_phy_stf_chain_active_get(struct brcms_phy_pub *pih); | ||||||
|  | extern s8 wlc_phy_stf_ssmode_get(struct brcms_phy_pub *pih, | ||||||
|  | 				 u16 chanspec); | ||||||
|  | extern void wlc_phy_ldpc_override_set(struct brcms_phy_pub *ppi, bool val); | ||||||
|  | 
 | ||||||
|  | extern void wlc_phy_cal_perical(struct brcms_phy_pub *ppi, u8 reason); | ||||||
|  | extern void wlc_phy_noise_sample_request_external(struct brcms_phy_pub *ppi); | ||||||
|  | extern void wlc_phy_edcrs_lock(struct brcms_phy_pub *pih, bool lock); | ||||||
|  | extern void wlc_phy_cal_papd_recal(struct brcms_phy_pub *ppi); | ||||||
|  | 
 | ||||||
|  | extern void wlc_phy_ant_rxdiv_set(struct brcms_phy_pub *ppi, u8 val); | ||||||
|  | extern void wlc_phy_clear_tssi(struct brcms_phy_pub *ppi); | ||||||
|  | extern void wlc_phy_hold_upd(struct brcms_phy_pub *ppi, u32 id, bool val); | ||||||
|  | extern void wlc_phy_mute_upd(struct brcms_phy_pub *ppi, bool val, u32 flags); | ||||||
|  | 
 | ||||||
|  | extern void wlc_phy_antsel_type_set(struct brcms_phy_pub *ppi, u8 antsel_type); | ||||||
|  | 
 | ||||||
|  | extern void wlc_phy_txpower_get_current(struct brcms_phy_pub *ppi, | ||||||
|  | 					struct tx_power *power, uint channel); | ||||||
|  | 
 | ||||||
|  | extern void wlc_phy_initcal_enable(struct brcms_phy_pub *pih, bool initcal); | ||||||
|  | extern bool wlc_phy_test_ison(struct brcms_phy_pub *ppi); | ||||||
|  | extern void wlc_phy_txpwr_percent_set(struct brcms_phy_pub *ppi, | ||||||
|  | 				      u8 txpwr_percent); | ||||||
|  | extern void wlc_phy_ofdm_rateset_war(struct brcms_phy_pub *pih, bool war); | ||||||
|  | extern void wlc_phy_bf_preempt_enable(struct brcms_phy_pub *pih, | ||||||
|  | 				      bool bf_preempt); | ||||||
|  | extern void wlc_phy_machwcap_set(struct brcms_phy_pub *ppi, u32 machwcap); | ||||||
|  | 
 | ||||||
|  | extern void wlc_phy_runbist_config(struct brcms_phy_pub *ppi, bool start_end); | ||||||
|  | 
 | ||||||
|  | extern void wlc_phy_freqtrack_start(struct brcms_phy_pub *ppi); | ||||||
|  | extern void wlc_phy_freqtrack_end(struct brcms_phy_pub *ppi); | ||||||
|  | 
 | ||||||
|  | extern const u8 *wlc_phy_get_ofdm_rate_lookup(void); | ||||||
|  | 
 | ||||||
|  | extern s8 wlc_phy_get_tx_power_offset_by_mcs(struct brcms_phy_pub *ppi, | ||||||
|  | 					     u8 mcs_offset); | ||||||
|  | extern s8 wlc_phy_get_tx_power_offset(struct brcms_phy_pub *ppi, u8 tbl_offset); | ||||||
|  | #endif                          /* _BRCM_PHY_HAL_H_ */ | ||||||
							
								
								
									
										1169
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_int.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1169
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_int.h
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										5154
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_lcn.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5154
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_lcn.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										121
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_lcn.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										121
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_lcn.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,121 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCM_PHY_LCN_H_ | ||||||
|  | #define _BRCM_PHY_LCN_H_ | ||||||
|  | 
 | ||||||
|  | #include <types.h> | ||||||
|  | 
 | ||||||
|  | struct brcms_phy_lcnphy { | ||||||
|  | 	int lcnphy_txrf_sp_9_override; | ||||||
|  | 	u8 lcnphy_full_cal_channel; | ||||||
|  | 	u8 lcnphy_cal_counter; | ||||||
|  | 	u16 lcnphy_cal_temper; | ||||||
|  | 	bool lcnphy_recal; | ||||||
|  | 
 | ||||||
|  | 	u8 lcnphy_rc_cap; | ||||||
|  | 	u32 lcnphy_mcs20_po; | ||||||
|  | 
 | ||||||
|  | 	u8 lcnphy_tr_isolation_mid; | ||||||
|  | 	u8 lcnphy_tr_isolation_low; | ||||||
|  | 	u8 lcnphy_tr_isolation_hi; | ||||||
|  | 
 | ||||||
|  | 	u8 lcnphy_bx_arch; | ||||||
|  | 	u8 lcnphy_rx_power_offset; | ||||||
|  | 	u8 lcnphy_rssi_vf; | ||||||
|  | 	u8 lcnphy_rssi_vc; | ||||||
|  | 	u8 lcnphy_rssi_gs; | ||||||
|  | 	u8 lcnphy_tssi_val; | ||||||
|  | 	u8 lcnphy_rssi_vf_lowtemp; | ||||||
|  | 	u8 lcnphy_rssi_vc_lowtemp; | ||||||
|  | 	u8 lcnphy_rssi_gs_lowtemp; | ||||||
|  | 
 | ||||||
|  | 	u8 lcnphy_rssi_vf_hightemp; | ||||||
|  | 	u8 lcnphy_rssi_vc_hightemp; | ||||||
|  | 	u8 lcnphy_rssi_gs_hightemp; | ||||||
|  | 
 | ||||||
|  | 	s16 lcnphy_pa0b0; | ||||||
|  | 	s16 lcnphy_pa0b1; | ||||||
|  | 	s16 lcnphy_pa0b2; | ||||||
|  | 
 | ||||||
|  | 	u16 lcnphy_rawtempsense; | ||||||
|  | 	u8 lcnphy_measPower; | ||||||
|  | 	u8 lcnphy_tempsense_slope; | ||||||
|  | 	u8 lcnphy_freqoffset_corr; | ||||||
|  | 	u8 lcnphy_tempsense_option; | ||||||
|  | 	u8 lcnphy_tempcorrx; | ||||||
|  | 	bool lcnphy_iqcal_swp_dis; | ||||||
|  | 	bool lcnphy_hw_iqcal_en; | ||||||
|  | 	uint lcnphy_bandedge_corr; | ||||||
|  | 	bool lcnphy_spurmod; | ||||||
|  | 	u16 lcnphy_tssi_tx_cnt; | ||||||
|  | 	u16 lcnphy_tssi_idx; | ||||||
|  | 	u16 lcnphy_tssi_npt; | ||||||
|  | 
 | ||||||
|  | 	u16 lcnphy_target_tx_freq; | ||||||
|  | 	s8 lcnphy_tx_power_idx_override; | ||||||
|  | 	u16 lcnphy_noise_samples; | ||||||
|  | 
 | ||||||
|  | 	u32 lcnphy_papdRxGnIdx; | ||||||
|  | 	u32 lcnphy_papd_rxGnCtrl_init; | ||||||
|  | 
 | ||||||
|  | 	u32 lcnphy_gain_idx_14_lowword; | ||||||
|  | 	u32 lcnphy_gain_idx_14_hiword; | ||||||
|  | 	u32 lcnphy_gain_idx_27_lowword; | ||||||
|  | 	u32 lcnphy_gain_idx_27_hiword; | ||||||
|  | 	s16 lcnphy_ofdmgainidxtableoffset; | ||||||
|  | 	s16 lcnphy_dsssgainidxtableoffset; | ||||||
|  | 	u32 lcnphy_tr_R_gain_val; | ||||||
|  | 	u32 lcnphy_tr_T_gain_val; | ||||||
|  | 	s8 lcnphy_input_pwr_offset_db; | ||||||
|  | 	u16 lcnphy_Med_Low_Gain_db; | ||||||
|  | 	u16 lcnphy_Very_Low_Gain_db; | ||||||
|  | 	s8 lcnphy_lastsensed_temperature; | ||||||
|  | 	s8 lcnphy_pkteng_rssi_slope; | ||||||
|  | 	u8 lcnphy_saved_tx_user_target[TXP_NUM_RATES]; | ||||||
|  | 	u8 lcnphy_volt_winner; | ||||||
|  | 	u8 lcnphy_volt_low; | ||||||
|  | 	u8 lcnphy_54_48_36_24mbps_backoff; | ||||||
|  | 	u8 lcnphy_11n_backoff; | ||||||
|  | 	u8 lcnphy_lowerofdm; | ||||||
|  | 	u8 lcnphy_cck; | ||||||
|  | 	u8 lcnphy_psat_2pt3_detected; | ||||||
|  | 	s32 lcnphy_lowest_Re_div_Im; | ||||||
|  | 	s8 lcnphy_final_papd_cal_idx; | ||||||
|  | 	u16 lcnphy_extstxctrl4; | ||||||
|  | 	u16 lcnphy_extstxctrl0; | ||||||
|  | 	u16 lcnphy_extstxctrl1; | ||||||
|  | 	s16 lcnphy_cck_dig_filt_type; | ||||||
|  | 	s16 lcnphy_ofdm_dig_filt_type; | ||||||
|  | 	struct lcnphy_cal_results lcnphy_cal_results; | ||||||
|  | 
 | ||||||
|  | 	u8 lcnphy_psat_pwr; | ||||||
|  | 	u8 lcnphy_psat_indx; | ||||||
|  | 	s32 lcnphy_min_phase; | ||||||
|  | 	u8 lcnphy_final_idx; | ||||||
|  | 	u8 lcnphy_start_idx; | ||||||
|  | 	u8 lcnphy_current_index; | ||||||
|  | 	u16 lcnphy_logen_buf_1; | ||||||
|  | 	u16 lcnphy_local_ovr_2; | ||||||
|  | 	u16 lcnphy_local_oval_6; | ||||||
|  | 	u16 lcnphy_local_oval_5; | ||||||
|  | 	u16 lcnphy_logen_mixer_1; | ||||||
|  | 
 | ||||||
|  | 	u8 lcnphy_aci_stat; | ||||||
|  | 	uint lcnphy_aci_start_time; | ||||||
|  | 	s8 lcnphy_tx_power_offset[TXP_NUM_RATES]; | ||||||
|  | }; | ||||||
|  | #endif				/* _BRCM_PHY_LCN_H_ */ | ||||||
							
								
								
									
										28876
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_n.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28876
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_n.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										308
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_qmath.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										308
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_qmath.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,308 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include "phy_qmath.h" | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Description: This function make 16 bit unsigned multiplication. | ||||||
|  |  * To fit the output into 16 bits the 32 bit multiplication result is right | ||||||
|  |  * shifted by 16 bits. | ||||||
|  |  */ | ||||||
|  | u16 qm_mulu16(u16 op1, u16 op2) | ||||||
|  | { | ||||||
|  | 	return (u16) (((u32) op1 * (u32) op2) >> 16); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Description: This function make 16 bit multiplication and return the result | ||||||
|  |  * in 16 bits. To fit the multiplication result into 16 bits the multiplication | ||||||
|  |  * result is right shifted by 15 bits. Right shifting 15 bits instead of 16 bits | ||||||
|  |  * is done to remove the extra sign bit formed due to the multiplication. | ||||||
|  |  * When both the 16bit inputs are 0x8000 then the output is saturated to | ||||||
|  |  * 0x7fffffff. | ||||||
|  |  */ | ||||||
|  | s16 qm_muls16(s16 op1, s16 op2) | ||||||
|  | { | ||||||
|  | 	s32 result; | ||||||
|  | 	if (op1 == (s16) 0x8000 && op2 == (s16) 0x8000) | ||||||
|  | 		result = 0x7fffffff; | ||||||
|  | 	else | ||||||
|  | 		result = ((s32) (op1) * (s32) (op2)); | ||||||
|  | 
 | ||||||
|  | 	return (s16) (result >> 15); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Description: This function add two 32 bit numbers and return the 32bit | ||||||
|  |  * result. If the result overflow 32 bits, the output will be saturated to | ||||||
|  |  * 32bits. | ||||||
|  |  */ | ||||||
|  | s32 qm_add32(s32 op1, s32 op2) | ||||||
|  | { | ||||||
|  | 	s32 result; | ||||||
|  | 	result = op1 + op2; | ||||||
|  | 	if (op1 < 0 && op2 < 0 && result > 0) | ||||||
|  | 		result = 0x80000000; | ||||||
|  | 	else if (op1 > 0 && op2 > 0 && result < 0) | ||||||
|  | 		result = 0x7fffffff; | ||||||
|  | 
 | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Description: This function add two 16 bit numbers and return the 16bit | ||||||
|  |  * result. If the result overflow 16 bits, the output will be saturated to | ||||||
|  |  * 16bits. | ||||||
|  |  */ | ||||||
|  | s16 qm_add16(s16 op1, s16 op2) | ||||||
|  | { | ||||||
|  | 	s16 result; | ||||||
|  | 	s32 temp = (s32) op1 + (s32) op2; | ||||||
|  | 	if (temp > (s32) 0x7fff) | ||||||
|  | 		result = (s16) 0x7fff; | ||||||
|  | 	else if (temp < (s32) 0xffff8000) | ||||||
|  | 		result = (s16) 0xffff8000; | ||||||
|  | 	else | ||||||
|  | 		result = (s16) temp; | ||||||
|  | 
 | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Description: This function make 16 bit subtraction and return the 16bit | ||||||
|  |  * result. If the result overflow 16 bits, the output will be saturated to | ||||||
|  |  * 16bits. | ||||||
|  |  */ | ||||||
|  | s16 qm_sub16(s16 op1, s16 op2) | ||||||
|  | { | ||||||
|  | 	s16 result; | ||||||
|  | 	s32 temp = (s32) op1 - (s32) op2; | ||||||
|  | 	if (temp > (s32) 0x7fff) | ||||||
|  | 		result = (s16) 0x7fff; | ||||||
|  | 	else if (temp < (s32) 0xffff8000) | ||||||
|  | 		result = (s16) 0xffff8000; | ||||||
|  | 	else | ||||||
|  | 		result = (s16) temp; | ||||||
|  | 
 | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Description: This function make a 32 bit saturated left shift when the | ||||||
|  |  * specified shift is +ve. This function will make a 32 bit right shift when | ||||||
|  |  * the specified shift is -ve. This function return the result after shifting | ||||||
|  |  * operation. | ||||||
|  |  */ | ||||||
|  | s32 qm_shl32(s32 op, int shift) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	s32 result; | ||||||
|  | 	result = op; | ||||||
|  | 	if (shift > 31) | ||||||
|  | 		shift = 31; | ||||||
|  | 	else if (shift < -31) | ||||||
|  | 		shift = -31; | ||||||
|  | 	if (shift >= 0) { | ||||||
|  | 		for (i = 0; i < shift; i++) | ||||||
|  | 			result = qm_add32(result, result); | ||||||
|  | 	} else { | ||||||
|  | 		result = result >> (-shift); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Description: This function make a 16 bit saturated left shift when the | ||||||
|  |  * specified shift is +ve. This function will make a 16 bit right shift when | ||||||
|  |  * the specified shift is -ve. This function return the result after shifting | ||||||
|  |  * operation. | ||||||
|  |  */ | ||||||
|  | s16 qm_shl16(s16 op, int shift) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	s16 result; | ||||||
|  | 	result = op; | ||||||
|  | 	if (shift > 15) | ||||||
|  | 		shift = 15; | ||||||
|  | 	else if (shift < -15) | ||||||
|  | 		shift = -15; | ||||||
|  | 	if (shift > 0) { | ||||||
|  | 		for (i = 0; i < shift; i++) | ||||||
|  | 			result = qm_add16(result, result); | ||||||
|  | 	} else { | ||||||
|  | 		result = result >> (-shift); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Description: This function make a 16 bit right shift when shift is +ve. | ||||||
|  |  * This function make a 16 bit saturated left shift when shift is -ve. This | ||||||
|  |  * function return the result of the shift operation. | ||||||
|  |  */ | ||||||
|  | s16 qm_shr16(s16 op, int shift) | ||||||
|  | { | ||||||
|  | 	return qm_shl16(op, -shift); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Description: This function return the number of redundant sign bits in a | ||||||
|  |  * 32 bit number. Example: qm_norm32(0x00000080) = 23 | ||||||
|  |  */ | ||||||
|  | s16 qm_norm32(s32 op) | ||||||
|  | { | ||||||
|  | 	u16 u16extraSignBits; | ||||||
|  | 	if (op == 0) { | ||||||
|  | 		return 31; | ||||||
|  | 	} else { | ||||||
|  | 		u16extraSignBits = 0; | ||||||
|  | 		while ((op >> 31) == (op >> 30)) { | ||||||
|  | 			u16extraSignBits++; | ||||||
|  | 			op = op << 1; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return u16extraSignBits; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* This table is log2(1+(i/32)) where i=[0:1:31], in q.15 format */ | ||||||
|  | static const s16 log_table[] = { | ||||||
|  | 	0, | ||||||
|  | 	1455, | ||||||
|  | 	2866, | ||||||
|  | 	4236, | ||||||
|  | 	5568, | ||||||
|  | 	6863, | ||||||
|  | 	8124, | ||||||
|  | 	9352, | ||||||
|  | 	10549, | ||||||
|  | 	11716, | ||||||
|  | 	12855, | ||||||
|  | 	13968, | ||||||
|  | 	15055, | ||||||
|  | 	16117, | ||||||
|  | 	17156, | ||||||
|  | 	18173, | ||||||
|  | 	19168, | ||||||
|  | 	20143, | ||||||
|  | 	21098, | ||||||
|  | 	22034, | ||||||
|  | 	22952, | ||||||
|  | 	23852, | ||||||
|  | 	24736, | ||||||
|  | 	25604, | ||||||
|  | 	26455, | ||||||
|  | 	27292, | ||||||
|  | 	28114, | ||||||
|  | 	28922, | ||||||
|  | 	29717, | ||||||
|  | 	30498, | ||||||
|  | 	31267, | ||||||
|  | 	32024 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #define LOG_TABLE_SIZE 32       /* log_table size */ | ||||||
|  | #define LOG2_LOG_TABLE_SIZE 5   /* log2(log_table size) */ | ||||||
|  | #define Q_LOG_TABLE 15          /* qformat of log_table */ | ||||||
|  | #define LOG10_2         19728   /* log10(2) in q.16 */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Description: | ||||||
|  |  * This routine takes the input number N and its q format qN and compute | ||||||
|  |  * the log10(N). This routine first normalizes the input no N.	Then N is in | ||||||
|  |  * mag*(2^x) format. mag is any number in the range 2^30-(2^31 - 1). | ||||||
|  |  * Then log2(mag * 2^x) = log2(mag) + x is computed. From that | ||||||
|  |  * log10(mag * 2^x) = log2(mag * 2^x) * log10(2) is computed. | ||||||
|  |  * This routine looks the log2 value in the table considering | ||||||
|  |  * LOG2_LOG_TABLE_SIZE+1 MSBs. As the MSB is always 1, only next | ||||||
|  |  * LOG2_OF_LOG_TABLE_SIZE MSBs are used for table lookup. Next 16 MSBs are used | ||||||
|  |  * for interpolation. | ||||||
|  |  * Inputs: | ||||||
|  |  * N - number to which log10 has to be found. | ||||||
|  |  * qN - q format of N | ||||||
|  |  * log10N - address where log10(N) will be written. | ||||||
|  |  * qLog10N - address where log10N qformat will be written. | ||||||
|  |  * Note/Problem: | ||||||
|  |  * For accurate results input should be in normalized or near normalized form. | ||||||
|  |  */ | ||||||
|  | void qm_log10(s32 N, s16 qN, s16 *log10N, s16 *qLog10N) | ||||||
|  | { | ||||||
|  | 	s16 s16norm, s16tableIndex, s16errorApproximation; | ||||||
|  | 	u16 u16offset; | ||||||
|  | 	s32 s32log; | ||||||
|  | 
 | ||||||
|  | 	/* normalize the N. */ | ||||||
|  | 	s16norm = qm_norm32(N); | ||||||
|  | 	N = N << s16norm; | ||||||
|  | 
 | ||||||
|  | 	/* The qformat of N after normalization.
 | ||||||
|  | 	 * -30 is added to treat the no as between 1.0 to 2.0 | ||||||
|  | 	 * i.e. after adding the -30 to the qformat the decimal point will be | ||||||
|  | 	 * just rigtht of the MSB. (i.e. after sign bit and 1st MSB). i.e. | ||||||
|  | 	 * at the right side of 30th bit. | ||||||
|  | 	 */ | ||||||
|  | 	qN = qN + s16norm - 30; | ||||||
|  | 
 | ||||||
|  | 	/* take the table index as the LOG2_OF_LOG_TABLE_SIZE bits right of the
 | ||||||
|  | 	 * MSB */ | ||||||
|  | 	s16tableIndex = (s16) (N >> (32 - (2 + LOG2_LOG_TABLE_SIZE))); | ||||||
|  | 
 | ||||||
|  | 	/* remove the MSB. the MSB is always 1 after normalization. */ | ||||||
|  | 	s16tableIndex = | ||||||
|  | 		s16tableIndex & (s16) ((1 << LOG2_LOG_TABLE_SIZE) - 1); | ||||||
|  | 
 | ||||||
|  | 	/* remove the (1+LOG2_OF_LOG_TABLE_SIZE) MSBs in the N. */ | ||||||
|  | 	N = N & ((1 << (32 - (2 + LOG2_LOG_TABLE_SIZE))) - 1); | ||||||
|  | 
 | ||||||
|  | 	/* take the offset as the 16 MSBS after table index.
 | ||||||
|  | 	 */ | ||||||
|  | 	u16offset = (u16) (N >> (32 - (2 + LOG2_LOG_TABLE_SIZE + 16))); | ||||||
|  | 
 | ||||||
|  | 	/* look the log value in the table. */ | ||||||
|  | 	s32log = log_table[s16tableIndex];      /* q.15 format */ | ||||||
|  | 
 | ||||||
|  | 	/* interpolate using the offset. q.15 format. */ | ||||||
|  | 	s16errorApproximation = (s16) qm_mulu16(u16offset, | ||||||
|  | 				(u16) (log_table[s16tableIndex + 1] - | ||||||
|  | 				       log_table[s16tableIndex])); | ||||||
|  | 
 | ||||||
|  | 	 /* q.15 format */ | ||||||
|  | 	s32log = qm_add16((s16) s32log, s16errorApproximation); | ||||||
|  | 
 | ||||||
|  | 	/* adjust for the qformat of the N as
 | ||||||
|  | 	 * log2(mag * 2^x) = log2(mag) + x | ||||||
|  | 	 */ | ||||||
|  | 	s32log = qm_add32(s32log, ((s32) -qN) << 15);   /* q.15 format */ | ||||||
|  | 
 | ||||||
|  | 	/* normalize the result. */ | ||||||
|  | 	s16norm = qm_norm32(s32log); | ||||||
|  | 
 | ||||||
|  | 	/* bring all the important bits into lower 16 bits */ | ||||||
|  | 	/* q.15+s16norm-16 format */ | ||||||
|  | 	s32log = qm_shl32(s32log, s16norm - 16); | ||||||
|  | 
 | ||||||
|  | 	/* compute the log10(N) by multiplying log2(N) with log10(2).
 | ||||||
|  | 	 * as log10(mag * 2^x) = log2(mag * 2^x) * log10(2) | ||||||
|  | 	 * log10N in q.15+s16norm-16+1 (LOG10_2 is in q.16) | ||||||
|  | 	 */ | ||||||
|  | 	*log10N = qm_muls16((s16) s32log, (s16) LOG10_2); | ||||||
|  | 
 | ||||||
|  | 	/* write the q format of the result. */ | ||||||
|  | 	*qLog10N = 15 + s16norm - 16 + 1; | ||||||
|  | 
 | ||||||
|  | 	return; | ||||||
|  | } | ||||||
							
								
								
									
										42
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_qmath.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_qmath.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,42 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCM_QMATH_H_ | ||||||
|  | #define _BRCM_QMATH_H_ | ||||||
|  | 
 | ||||||
|  | #include <types.h> | ||||||
|  | 
 | ||||||
|  | u16 qm_mulu16(u16 op1, u16 op2); | ||||||
|  | 
 | ||||||
|  | s16 qm_muls16(s16 op1, s16 op2); | ||||||
|  | 
 | ||||||
|  | s32 qm_add32(s32 op1, s32 op2); | ||||||
|  | 
 | ||||||
|  | s16 qm_add16(s16 op1, s16 op2); | ||||||
|  | 
 | ||||||
|  | s16 qm_sub16(s16 op1, s16 op2); | ||||||
|  | 
 | ||||||
|  | s32 qm_shl32(s32 op, int shift); | ||||||
|  | 
 | ||||||
|  | s16 qm_shl16(s16 op, int shift); | ||||||
|  | 
 | ||||||
|  | s16 qm_shr16(s16 op, int shift); | ||||||
|  | 
 | ||||||
|  | s16 qm_norm32(s32 op); | ||||||
|  | 
 | ||||||
|  | void qm_log10(s32 N, s16 qN, s16 *log10N, s16 *qLog10N); | ||||||
|  | 
 | ||||||
|  | #endif				/* #ifndef _BRCM_QMATH_H_ */ | ||||||
							
								
								
									
										1533
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_radio.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1533
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phy_radio.h
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										167
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phyreg_n.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										167
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phyreg_n.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,167 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #define NPHY_TBL_ID_GAIN1		0 | ||||||
|  | #define NPHY_TBL_ID_GAIN2		1 | ||||||
|  | #define NPHY_TBL_ID_GAINBITS1		2 | ||||||
|  | #define NPHY_TBL_ID_GAINBITS2		3 | ||||||
|  | #define NPHY_TBL_ID_GAINLIMIT		4 | ||||||
|  | #define NPHY_TBL_ID_WRSSIGainLimit	5 | ||||||
|  | #define NPHY_TBL_ID_RFSEQ		7 | ||||||
|  | #define NPHY_TBL_ID_AFECTRL		8 | ||||||
|  | #define NPHY_TBL_ID_ANTSWCTRLLUT	9 | ||||||
|  | #define NPHY_TBL_ID_IQLOCAL		15 | ||||||
|  | #define NPHY_TBL_ID_NOISEVAR		16 | ||||||
|  | #define NPHY_TBL_ID_SAMPLEPLAY		17 | ||||||
|  | #define NPHY_TBL_ID_CORE1TXPWRCTL	26 | ||||||
|  | #define NPHY_TBL_ID_CORE2TXPWRCTL	27 | ||||||
|  | #define NPHY_TBL_ID_CMPMETRICDATAWEIGHTTBL	30 | ||||||
|  | 
 | ||||||
|  | #define NPHY_TBL_ID_EPSILONTBL0   31 | ||||||
|  | #define NPHY_TBL_ID_SCALARTBL0    32 | ||||||
|  | #define NPHY_TBL_ID_EPSILONTBL1   33 | ||||||
|  | #define NPHY_TBL_ID_SCALARTBL1    34 | ||||||
|  | 
 | ||||||
|  | #define	NPHY_TO_BPHY_OFF	0xc00 | ||||||
|  | 
 | ||||||
|  | #define NPHY_BandControl_currentBand			0x0001 | ||||||
|  | #define RFCC_CHIP0_PU			0x0400 | ||||||
|  | #define RFCC_POR_FORCE			0x0040 | ||||||
|  | #define RFCC_OE_POR_FORCE		0x0080 | ||||||
|  | #define NPHY_RfctrlIntc_override_OFF			0 | ||||||
|  | #define NPHY_RfctrlIntc_override_TRSW			1 | ||||||
|  | #define NPHY_RfctrlIntc_override_PA				2 | ||||||
|  | #define NPHY_RfctrlIntc_override_EXT_LNA_PU		3 | ||||||
|  | #define NPHY_RfctrlIntc_override_EXT_LNA_GAIN	4 | ||||||
|  | #define RIFS_ENABLE			0x80 | ||||||
|  | #define BPHY_BAND_SEL_UP20		0x10 | ||||||
|  | #define NPHY_MLenable			0x02 | ||||||
|  | 
 | ||||||
|  | #define NPHY_RfseqMode_CoreActv_override 0x0001 | ||||||
|  | #define NPHY_RfseqMode_Trigger_override	0x0002 | ||||||
|  | #define NPHY_RfseqCoreActv_TxRxChain0	(0x11) | ||||||
|  | #define NPHY_RfseqCoreActv_TxRxChain1	(0x22) | ||||||
|  | 
 | ||||||
|  | #define NPHY_RfseqTrigger_rx2tx		0x0001 | ||||||
|  | #define NPHY_RfseqTrigger_tx2rx		0x0002 | ||||||
|  | #define NPHY_RfseqTrigger_updategainh	0x0004 | ||||||
|  | #define NPHY_RfseqTrigger_updategainl	0x0008 | ||||||
|  | #define NPHY_RfseqTrigger_updategainu	0x0010 | ||||||
|  | #define NPHY_RfseqTrigger_reset2rx	0x0020 | ||||||
|  | #define NPHY_RfseqStatus_rx2tx		0x0001 | ||||||
|  | #define NPHY_RfseqStatus_tx2rx		0x0002 | ||||||
|  | #define NPHY_RfseqStatus_updategainh	0x0004 | ||||||
|  | #define NPHY_RfseqStatus_updategainl	0x0008 | ||||||
|  | #define NPHY_RfseqStatus_updategainu	0x0010 | ||||||
|  | #define NPHY_RfseqStatus_reset2rx	0x0020 | ||||||
|  | #define NPHY_ClassifierCtrl_cck_en	0x1 | ||||||
|  | #define NPHY_ClassifierCtrl_ofdm_en	0x2 | ||||||
|  | #define NPHY_ClassifierCtrl_waited_en	0x4 | ||||||
|  | #define NPHY_IQFlip_ADC1		0x0001 | ||||||
|  | #define NPHY_IQFlip_ADC2		0x0010 | ||||||
|  | #define NPHY_sampleCmd_STOP		0x0002 | ||||||
|  | 
 | ||||||
|  | #define RX_GF_OR_MM			0x0004 | ||||||
|  | #define RX_GF_MM_AUTO			0x0100 | ||||||
|  | 
 | ||||||
|  | #define NPHY_iqloCalCmdGctl_IQLO_CAL_EN	0x8000 | ||||||
|  | 
 | ||||||
|  | #define NPHY_IqestCmd_iqstart		0x1 | ||||||
|  | #define NPHY_IqestCmd_iqMode		0x2 | ||||||
|  | 
 | ||||||
|  | #define NPHY_TxPwrCtrlCmd_pwrIndex_init		0x40 | ||||||
|  | #define NPHY_TxPwrCtrlCmd_pwrIndex_init_rev7	0x19 | ||||||
|  | 
 | ||||||
|  | #define PRIM_SEL_UP20		0x8000 | ||||||
|  | 
 | ||||||
|  | #define NPHY_RFSEQ_RX2TX		0x0 | ||||||
|  | #define NPHY_RFSEQ_TX2RX		0x1 | ||||||
|  | #define NPHY_RFSEQ_RESET2RX		0x2 | ||||||
|  | #define NPHY_RFSEQ_UPDATEGAINH		0x3 | ||||||
|  | #define NPHY_RFSEQ_UPDATEGAINL		0x4 | ||||||
|  | #define NPHY_RFSEQ_UPDATEGAINU		0x5 | ||||||
|  | 
 | ||||||
|  | #define NPHY_RFSEQ_CMD_NOP		0x0 | ||||||
|  | #define NPHY_RFSEQ_CMD_RXG_FBW		0x1 | ||||||
|  | #define NPHY_RFSEQ_CMD_TR_SWITCH	0x2 | ||||||
|  | #define NPHY_RFSEQ_CMD_EXT_PA		0x3 | ||||||
|  | #define NPHY_RFSEQ_CMD_RXPD_TXPD	0x4 | ||||||
|  | #define NPHY_RFSEQ_CMD_TX_GAIN		0x5 | ||||||
|  | #define NPHY_RFSEQ_CMD_RX_GAIN		0x6 | ||||||
|  | #define NPHY_RFSEQ_CMD_SET_HPF_BW	0x7 | ||||||
|  | #define NPHY_RFSEQ_CMD_CLR_HIQ_DIS	0x8 | ||||||
|  | #define NPHY_RFSEQ_CMD_END		0xf | ||||||
|  | 
 | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_NOP		0x0 | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_RXG_FBW	0x1 | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_TR_SWITCH	0x2 | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_INT_PA_PU	0x3 | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_EXT_PA	0x4 | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_RXPD_TXPD	0x5 | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_TX_GAIN	0x6 | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_RX_GAIN	0x7 | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_CLR_HIQ_DIS	0x8 | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_SET_HPF_H_HPC	0x9 | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_SET_LPF_H_HPC	0xa | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_SET_HPF_M_HPC	0xb | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_SET_LPF_M_HPC	0xc | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_SET_HPF_L_HPC	0xd | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_SET_LPF_L_HPC	0xe | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_CLR_RXRX_BIAS	0xf | ||||||
|  | #define NPHY_REV3_RFSEQ_CMD_END		0x1f | ||||||
|  | 
 | ||||||
|  | #define NPHY_RSSI_SEL_W1		0x0 | ||||||
|  | #define NPHY_RSSI_SEL_W2		0x1 | ||||||
|  | #define NPHY_RSSI_SEL_NB		0x2 | ||||||
|  | #define NPHY_RSSI_SEL_IQ		0x3 | ||||||
|  | #define NPHY_RSSI_SEL_TSSI_2G		0x4 | ||||||
|  | #define NPHY_RSSI_SEL_TSSI_5G		0x5 | ||||||
|  | #define NPHY_RSSI_SEL_TBD		0x6 | ||||||
|  | 
 | ||||||
|  | #define NPHY_RAIL_I			0x0 | ||||||
|  | #define NPHY_RAIL_Q			0x1 | ||||||
|  | 
 | ||||||
|  | #define NPHY_FORCESIG_DECODEGATEDCLKS	0x8 | ||||||
|  | 
 | ||||||
|  | #define NPHY_REV7_RfctrlOverride_cmd_rxrf_pu 0x0 | ||||||
|  | #define NPHY_REV7_RfctrlOverride_cmd_rx_pu   0x1 | ||||||
|  | #define NPHY_REV7_RfctrlOverride_cmd_tx_pu   0x2 | ||||||
|  | #define NPHY_REV7_RfctrlOverride_cmd_rxgain  0x3 | ||||||
|  | #define NPHY_REV7_RfctrlOverride_cmd_txgain  0x4 | ||||||
|  | 
 | ||||||
|  | #define NPHY_REV7_RXGAINCODE_RFMXGAIN_MASK 0x000ff | ||||||
|  | #define NPHY_REV7_RXGAINCODE_LPFGAIN_MASK  0x0ff00 | ||||||
|  | #define NPHY_REV7_RXGAINCODE_DVGAGAIN_MASK 0xf0000 | ||||||
|  | 
 | ||||||
|  | #define NPHY_REV7_TXGAINCODE_TGAIN_MASK     0x7fff | ||||||
|  | #define NPHY_REV7_TXGAINCODE_LPFGAIN_MASK   0x8000 | ||||||
|  | #define NPHY_REV7_TXGAINCODE_BIQ0GAIN_SHIFT 14 | ||||||
|  | 
 | ||||||
|  | #define NPHY_REV7_RFCTRLOVERRIDE_ID0 0x0 | ||||||
|  | #define NPHY_REV7_RFCTRLOVERRIDE_ID1 0x1 | ||||||
|  | #define NPHY_REV7_RFCTRLOVERRIDE_ID2 0x2 | ||||||
|  | 
 | ||||||
|  | #define NPHY_IqestIqAccLo(core)  ((core == 0) ? 0x12c : 0x134) | ||||||
|  | 
 | ||||||
|  | #define NPHY_IqestIqAccHi(core)  ((core == 0) ? 0x12d : 0x135) | ||||||
|  | 
 | ||||||
|  | #define NPHY_IqestipwrAccLo(core)  ((core == 0) ? 0x12e : 0x136) | ||||||
|  | 
 | ||||||
|  | #define NPHY_IqestipwrAccHi(core)  ((core == 0) ? 0x12f : 0x137) | ||||||
|  | 
 | ||||||
|  | #define NPHY_IqestqpwrAccLo(core)  ((core == 0) ? 0x130 : 0x138) | ||||||
|  | 
 | ||||||
|  | #define NPHY_IqestqpwrAccHi(core)  ((core == 0) ? 0x131 : 0x139) | ||||||
							
								
								
									
										3250
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phytbl_lcn.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3250
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phytbl_lcn.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										54
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phytbl_lcn.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phytbl_lcn.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,54 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <types.h> | ||||||
|  | #include "phy_int.h" | ||||||
|  | 
 | ||||||
|  | extern const struct phytbl_info dot11lcnphytbl_rx_gain_info_rev0[]; | ||||||
|  | extern const u32 dot11lcnphytbl_rx_gain_info_sz_rev0; | ||||||
|  | extern const struct phytbl_info dot11lcn_sw_ctrl_tbl_info_4313; | ||||||
|  | extern const struct phytbl_info dot11lcn_sw_ctrl_tbl_info_4313_epa; | ||||||
|  | extern const struct phytbl_info dot11lcn_sw_ctrl_tbl_info_4313_epa_combo; | ||||||
|  | extern const struct phytbl_info dot11lcn_sw_ctrl_tbl_info_4313_bt_epa; | ||||||
|  | extern const struct phytbl_info dot11lcn_sw_ctrl_tbl_info_4313_bt_epa_p250; | ||||||
|  | 
 | ||||||
|  | extern const struct phytbl_info dot11lcnphytbl_info_rev0[]; | ||||||
|  | extern const u32 dot11lcnphytbl_info_sz_rev0; | ||||||
|  | 
 | ||||||
|  | extern const struct phytbl_info dot11lcnphytbl_rx_gain_info_2G_rev2[]; | ||||||
|  | extern const u32 dot11lcnphytbl_rx_gain_info_2G_rev2_sz; | ||||||
|  | 
 | ||||||
|  | extern const struct phytbl_info dot11lcnphytbl_rx_gain_info_5G_rev2[]; | ||||||
|  | extern const u32 dot11lcnphytbl_rx_gain_info_5G_rev2_sz; | ||||||
|  | 
 | ||||||
|  | extern const struct phytbl_info dot11lcnphytbl_rx_gain_info_extlna_2G_rev2[]; | ||||||
|  | 
 | ||||||
|  | extern const struct phytbl_info dot11lcnphytbl_rx_gain_info_extlna_5G_rev2[]; | ||||||
|  | 
 | ||||||
|  | struct lcnphy_tx_gain_tbl_entry { | ||||||
|  | 	unsigned char gm; | ||||||
|  | 	unsigned char pga; | ||||||
|  | 	unsigned char pad; | ||||||
|  | 	unsigned char dac; | ||||||
|  | 	unsigned char bb_mult; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | extern const struct lcnphy_tx_gain_tbl_entry dot11lcnphy_2GHz_gaintable_rev0[]; | ||||||
|  | 
 | ||||||
|  | extern const struct | ||||||
|  | lcnphy_tx_gain_tbl_entry dot11lcnphy_2GHz_extPA_gaintable_rev0[]; | ||||||
|  | 
 | ||||||
|  | extern const struct lcnphy_tx_gain_tbl_entry dot11lcnphy_5GHz_gaintable_rev0[]; | ||||||
							
								
								
									
										10630
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phytbl_n.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10630
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phytbl_n.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										50
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phytbl_n.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy/phytbl_n.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,50 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #define ANT_SWCTRL_TBL_REV3_IDX (0) | ||||||
|  | 
 | ||||||
|  | #include <types.h> | ||||||
|  | #include "phy_int.h" | ||||||
|  | 
 | ||||||
|  | extern const struct phytbl_info mimophytbl_info_rev0[], | ||||||
|  | 				mimophytbl_info_rev0_volatile[]; | ||||||
|  | 
 | ||||||
|  | extern const u32 mimophytbl_info_sz_rev0, | ||||||
|  | 		 mimophytbl_info_sz_rev0_volatile; | ||||||
|  | 
 | ||||||
|  | extern const struct phytbl_info mimophytbl_info_rev3[], | ||||||
|  | 				mimophytbl_info_rev3_volatile[], | ||||||
|  | 				mimophytbl_info_rev3_volatile1[], | ||||||
|  | 				mimophytbl_info_rev3_volatile2[], | ||||||
|  | 				mimophytbl_info_rev3_volatile3[]; | ||||||
|  | 
 | ||||||
|  | extern const u32 mimophytbl_info_sz_rev3, | ||||||
|  | 		 mimophytbl_info_sz_rev3_volatile, | ||||||
|  | 		 mimophytbl_info_sz_rev3_volatile1, | ||||||
|  | 		 mimophytbl_info_sz_rev3_volatile2, | ||||||
|  | 		 mimophytbl_info_sz_rev3_volatile3; | ||||||
|  | 
 | ||||||
|  | extern const u32 noise_var_tbl_rev3[]; | ||||||
|  | 
 | ||||||
|  | extern const struct phytbl_info mimophytbl_info_rev7[]; | ||||||
|  | 
 | ||||||
|  | extern const u32 mimophytbl_info_sz_rev7; | ||||||
|  | 
 | ||||||
|  | extern const u32 noise_var_tbl_rev7[]; | ||||||
|  | 
 | ||||||
|  | extern const struct phytbl_info mimophytbl_info_rev16[]; | ||||||
|  | 
 | ||||||
|  | extern const u32 mimophytbl_info_sz_rev16; | ||||||
							
								
								
									
										226
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy_shim.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										226
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy_shim.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,226 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * This is "two-way" interface, acting as the SHIM layer between driver | ||||||
|  |  * and PHY layer. The driver can optionally call this translation layer | ||||||
|  |  * to do some preprocessing, then reach PHY. On the PHY->driver direction, | ||||||
|  |  * all calls go through this layer since PHY doesn't have access to the | ||||||
|  |  * driver's brcms_hardware pointer. | ||||||
|  |  */ | ||||||
|  | #include <linux/slab.h> | ||||||
|  | #include <net/mac80211.h> | ||||||
|  | 
 | ||||||
|  | #include "main.h" | ||||||
|  | #include "mac80211_if.h" | ||||||
|  | #include "phy_shim.h" | ||||||
|  | 
 | ||||||
|  | /* PHY SHIM module specific state */ | ||||||
|  | struct phy_shim_info { | ||||||
|  | 	struct brcms_hardware *wlc_hw;	/* pointer to main wlc_hw structure */ | ||||||
|  | 	struct brcms_c_info *wlc;	/* pointer to main wlc structure */ | ||||||
|  | 	struct brcms_info *wl; /* pointer to os-specific private state */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct phy_shim_info *wlc_phy_shim_attach(struct brcms_hardware *wlc_hw, | ||||||
|  | 					  struct brcms_info *wl, | ||||||
|  | 					  struct brcms_c_info *wlc) { | ||||||
|  | 	struct phy_shim_info *physhim = NULL; | ||||||
|  | 
 | ||||||
|  | 	physhim = kzalloc(sizeof(struct phy_shim_info), GFP_ATOMIC); | ||||||
|  | 	if (!physhim) | ||||||
|  | 		return NULL; | ||||||
|  | 
 | ||||||
|  | 	physhim->wlc_hw = wlc_hw; | ||||||
|  | 	physhim->wlc = wlc; | ||||||
|  | 	physhim->wl = wl; | ||||||
|  | 
 | ||||||
|  | 	return physhim; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlc_phy_shim_detach(struct phy_shim_info *physhim) | ||||||
|  | { | ||||||
|  | 	kfree(physhim); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct wlapi_timer *wlapi_init_timer(struct phy_shim_info *physhim, | ||||||
|  | 				     void (*fn)(struct brcms_phy *pi), | ||||||
|  | 				     void *arg, const char *name) | ||||||
|  | { | ||||||
|  | 	return (struct wlapi_timer *) | ||||||
|  | 			brcms_init_timer(physhim->wl, (void (*)(void *))fn, | ||||||
|  | 					 arg, name); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_free_timer(struct phy_shim_info *physhim, struct wlapi_timer *t) | ||||||
|  | { | ||||||
|  | 	brcms_free_timer(physhim->wl, (struct brcms_timer *)t); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | wlapi_add_timer(struct phy_shim_info *physhim, struct wlapi_timer *t, uint ms, | ||||||
|  | 		int periodic) | ||||||
|  | { | ||||||
|  | 	brcms_add_timer(physhim->wl, (struct brcms_timer *)t, ms, periodic); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool wlapi_del_timer(struct phy_shim_info *physhim, struct wlapi_timer *t) | ||||||
|  | { | ||||||
|  | 	return brcms_del_timer(physhim->wl, (struct brcms_timer *)t); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_intrson(struct phy_shim_info *physhim) | ||||||
|  | { | ||||||
|  | 	brcms_intrson(physhim->wl); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | u32 wlapi_intrsoff(struct phy_shim_info *physhim) | ||||||
|  | { | ||||||
|  | 	return brcms_intrsoff(physhim->wl); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_intrsrestore(struct phy_shim_info *physhim, u32 macintmask) | ||||||
|  | { | ||||||
|  | 	brcms_intrsrestore(physhim->wl, macintmask); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_bmac_write_shm(struct phy_shim_info *physhim, uint offset, u16 v) | ||||||
|  | { | ||||||
|  | 	brcms_b_write_shm(physhim->wlc_hw, offset, v); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | u16 wlapi_bmac_read_shm(struct phy_shim_info *physhim, uint offset) | ||||||
|  | { | ||||||
|  | 	return brcms_b_read_shm(physhim->wlc_hw, offset); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | wlapi_bmac_mhf(struct phy_shim_info *physhim, u8 idx, u16 mask, | ||||||
|  | 	       u16 val, int bands) | ||||||
|  | { | ||||||
|  | 	brcms_b_mhf(physhim->wlc_hw, idx, mask, val, bands); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_bmac_corereset(struct phy_shim_info *physhim, u32 flags) | ||||||
|  | { | ||||||
|  | 	brcms_b_corereset(physhim->wlc_hw, flags); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_suspend_mac_and_wait(struct phy_shim_info *physhim) | ||||||
|  | { | ||||||
|  | 	brcms_c_suspend_mac_and_wait(physhim->wlc); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_switch_macfreq(struct phy_shim_info *physhim, u8 spurmode) | ||||||
|  | { | ||||||
|  | 	brcms_b_switch_macfreq(physhim->wlc_hw, spurmode); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_enable_mac(struct phy_shim_info *physhim) | ||||||
|  | { | ||||||
|  | 	brcms_c_enable_mac(physhim->wlc); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_bmac_mctrl(struct phy_shim_info *physhim, u32 mask, u32 val) | ||||||
|  | { | ||||||
|  | 	brcms_b_mctrl(physhim->wlc_hw, mask, val); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_bmac_phy_reset(struct phy_shim_info *physhim) | ||||||
|  | { | ||||||
|  | 	brcms_b_phy_reset(physhim->wlc_hw); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_bmac_bw_set(struct phy_shim_info *physhim, u16 bw) | ||||||
|  | { | ||||||
|  | 	brcms_b_bw_set(physhim->wlc_hw, bw); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | u16 wlapi_bmac_get_txant(struct phy_shim_info *physhim) | ||||||
|  | { | ||||||
|  | 	return brcms_b_get_txant(physhim->wlc_hw); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_bmac_phyclk_fgc(struct phy_shim_info *physhim, bool clk) | ||||||
|  | { | ||||||
|  | 	brcms_b_phyclk_fgc(physhim->wlc_hw, clk); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_bmac_macphyclk_set(struct phy_shim_info *physhim, bool clk) | ||||||
|  | { | ||||||
|  | 	brcms_b_macphyclk_set(physhim->wlc_hw, clk); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_bmac_core_phypll_ctl(struct phy_shim_info *physhim, bool on) | ||||||
|  | { | ||||||
|  | 	brcms_b_core_phypll_ctl(physhim->wlc_hw, on); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_bmac_core_phypll_reset(struct phy_shim_info *physhim) | ||||||
|  | { | ||||||
|  | 	brcms_b_core_phypll_reset(physhim->wlc_hw); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_bmac_ucode_wake_override_phyreg_set(struct phy_shim_info *physhim) | ||||||
|  | { | ||||||
|  | 	brcms_c_ucode_wake_override_set(physhim->wlc_hw, | ||||||
|  | 					BRCMS_WAKE_OVERRIDE_PHYREG); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_bmac_ucode_wake_override_phyreg_clear(struct phy_shim_info *physhim) | ||||||
|  | { | ||||||
|  | 	brcms_c_ucode_wake_override_clear(physhim->wlc_hw, | ||||||
|  | 					  BRCMS_WAKE_OVERRIDE_PHYREG); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | wlapi_bmac_write_template_ram(struct phy_shim_info *physhim, int offset, | ||||||
|  | 			      int len, void *buf) | ||||||
|  | { | ||||||
|  | 	brcms_b_write_template_ram(physhim->wlc_hw, offset, len, buf); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | u16 wlapi_bmac_rate_shm_offset(struct phy_shim_info *physhim, u8 rate) | ||||||
|  | { | ||||||
|  | 	return brcms_b_rate_shm_offset(physhim->wlc_hw, rate); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlapi_ucode_sample_init(struct phy_shim_info *physhim) | ||||||
|  | { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | wlapi_copyfrom_objmem(struct phy_shim_info *physhim, uint offset, void *buf, | ||||||
|  | 		      int len, u32 sel) | ||||||
|  | { | ||||||
|  | 	brcms_b_copyfrom_objmem(physhim->wlc_hw, offset, buf, len, sel); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | wlapi_copyto_objmem(struct phy_shim_info *physhim, uint offset, const void *buf, | ||||||
|  | 		    int l, u32 sel) | ||||||
|  | { | ||||||
|  | 	brcms_b_copyto_objmem(physhim->wlc_hw, offset, buf, l, sel); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | char *wlapi_getvar(struct phy_shim_info *physhim, enum brcms_srom_id id) | ||||||
|  | { | ||||||
|  | 	return getvar(physhim->wlc_hw->sih, id); | ||||||
|  | } | ||||||
|  | int wlapi_getintvar(struct phy_shim_info *physhim, enum brcms_srom_id id) | ||||||
|  | { | ||||||
|  | 	return getintvar(physhim->wlc_hw->sih, id); | ||||||
|  | } | ||||||
							
								
								
									
										185
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy_shim.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										185
									
								
								drivers/net/wireless/brcm80211/brcmsmac/phy_shim.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,185 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * phy_shim.h: stuff defined in phy_shim.c and included only by the phy | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCM_PHY_SHIM_H_ | ||||||
|  | #define _BRCM_PHY_SHIM_H_ | ||||||
|  | 
 | ||||||
|  | #include "types.h" | ||||||
|  | 
 | ||||||
|  | #define RADAR_TYPE_NONE		0	/* Radar type None */ | ||||||
|  | #define RADAR_TYPE_ETSI_1	1	/* ETSI 1 Radar type */ | ||||||
|  | #define RADAR_TYPE_ETSI_2	2	/* ETSI 2 Radar type */ | ||||||
|  | #define RADAR_TYPE_ETSI_3	3	/* ETSI 3 Radar type */ | ||||||
|  | #define RADAR_TYPE_ITU_E	4	/* ITU E Radar type */ | ||||||
|  | #define RADAR_TYPE_ITU_K	5	/* ITU K Radar type */ | ||||||
|  | #define RADAR_TYPE_UNCLASSIFIED	6	/* Unclassified Radar type  */ | ||||||
|  | #define RADAR_TYPE_BIN5		7	/* long pulse radar type */ | ||||||
|  | #define RADAR_TYPE_STG2		8	/* staggered-2 radar */ | ||||||
|  | #define RADAR_TYPE_STG3		9	/* staggered-3 radar */ | ||||||
|  | #define RADAR_TYPE_FRA		10	/* French radar */ | ||||||
|  | 
 | ||||||
|  | /* French radar pulse widths */ | ||||||
|  | #define FRA_T1_20MHZ	52770 | ||||||
|  | #define FRA_T2_20MHZ	61538 | ||||||
|  | #define FRA_T3_20MHZ	66002 | ||||||
|  | #define FRA_T1_40MHZ	105541 | ||||||
|  | #define FRA_T2_40MHZ	123077 | ||||||
|  | #define FRA_T3_40MHZ	132004 | ||||||
|  | #define FRA_ERR_20MHZ	60 | ||||||
|  | #define FRA_ERR_40MHZ	120 | ||||||
|  | 
 | ||||||
|  | #define ANTSEL_NA		0 /* No boardlevel selection available */ | ||||||
|  | #define ANTSEL_2x4		1 /* 2x4 boardlevel selection available */ | ||||||
|  | #define ANTSEL_2x3		2 /* 2x3 CB2 boardlevel selection available */ | ||||||
|  | 
 | ||||||
|  | /* Rx Antenna diversity control values */ | ||||||
|  | #define	ANT_RX_DIV_FORCE_0	0	/* Use antenna 0 */ | ||||||
|  | #define	ANT_RX_DIV_FORCE_1	1	/* Use antenna 1 */ | ||||||
|  | #define	ANT_RX_DIV_START_1	2	/* Choose starting with 1 */ | ||||||
|  | #define	ANT_RX_DIV_START_0	3	/* Choose starting with 0 */ | ||||||
|  | #define	ANT_RX_DIV_ENABLE	3	/* APHY bbConfig Enable RX Diversity */ | ||||||
|  | #define ANT_RX_DIV_DEF		ANT_RX_DIV_START_0 /* default antdiv setting */ | ||||||
|  | 
 | ||||||
|  | #define WL_ANT_RX_MAX		2	/* max 2 receive antennas */ | ||||||
|  | #define WL_ANT_HT_RX_MAX	3	/* max 3 receive antennas/cores */ | ||||||
|  | #define WL_ANT_IDX_1		0	/* antenna index 1 */ | ||||||
|  | #define WL_ANT_IDX_2		1	/* antenna index 2 */ | ||||||
|  | 
 | ||||||
|  | /* values for n_preamble_type */ | ||||||
|  | #define BRCMS_N_PREAMBLE_MIXEDMODE	0 | ||||||
|  | #define BRCMS_N_PREAMBLE_GF		1 | ||||||
|  | #define BRCMS_N_PREAMBLE_GF_BRCM          2 | ||||||
|  | 
 | ||||||
|  | #define WL_TX_POWER_RATES_LEGACY	45 | ||||||
|  | #define WL_TX_POWER_MCS20_FIRST	        12 | ||||||
|  | #define WL_TX_POWER_MCS20_NUM	        16 | ||||||
|  | #define WL_TX_POWER_MCS40_FIRST	        28 | ||||||
|  | #define WL_TX_POWER_MCS40_NUM	        17 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #define WL_TX_POWER_RATES	       101 | ||||||
|  | #define WL_TX_POWER_CCK_FIRST	       0 | ||||||
|  | #define WL_TX_POWER_CCK_NUM	       4 | ||||||
|  | /* Index for first 20MHz OFDM SISO rate */ | ||||||
|  | #define WL_TX_POWER_OFDM_FIRST	       4 | ||||||
|  | /* Index for first 20MHz OFDM CDD rate */ | ||||||
|  | #define WL_TX_POWER_OFDM20_CDD_FIRST   12 | ||||||
|  | /* Index for first 40MHz OFDM SISO rate */ | ||||||
|  | #define WL_TX_POWER_OFDM40_SISO_FIRST  52 | ||||||
|  | /* Index for first 40MHz OFDM CDD rate */ | ||||||
|  | #define WL_TX_POWER_OFDM40_CDD_FIRST   60 | ||||||
|  | #define WL_TX_POWER_OFDM_NUM	       8 | ||||||
|  | /* Index for first 20MHz MCS SISO rate */ | ||||||
|  | #define WL_TX_POWER_MCS20_SISO_FIRST   20 | ||||||
|  | /* Index for first 20MHz MCS CDD rate */ | ||||||
|  | #define WL_TX_POWER_MCS20_CDD_FIRST    28 | ||||||
|  | /* Index for first 20MHz MCS STBC rate */ | ||||||
|  | #define WL_TX_POWER_MCS20_STBC_FIRST   36 | ||||||
|  | /* Index for first 20MHz MCS SDM rate */ | ||||||
|  | #define WL_TX_POWER_MCS20_SDM_FIRST    44 | ||||||
|  | /* Index for first 40MHz MCS SISO rate */ | ||||||
|  | #define WL_TX_POWER_MCS40_SISO_FIRST   68 | ||||||
|  | /* Index for first 40MHz MCS CDD rate */ | ||||||
|  | #define WL_TX_POWER_MCS40_CDD_FIRST    76 | ||||||
|  | /* Index for first 40MHz MCS STBC rate */ | ||||||
|  | #define WL_TX_POWER_MCS40_STBC_FIRST   84 | ||||||
|  | /* Index for first 40MHz MCS SDM rate */ | ||||||
|  | #define WL_TX_POWER_MCS40_SDM_FIRST    92 | ||||||
|  | #define WL_TX_POWER_MCS_1_STREAM_NUM   8 | ||||||
|  | #define WL_TX_POWER_MCS_2_STREAM_NUM   8 | ||||||
|  | /* Index for 40MHz rate MCS 32 */ | ||||||
|  | #define WL_TX_POWER_MCS_32	       100 | ||||||
|  | #define WL_TX_POWER_MCS_32_NUM	       1 | ||||||
|  | 
 | ||||||
|  | /* sslpnphy specifics */ | ||||||
|  | /* Index for first 20MHz MCS SISO rate */ | ||||||
|  | #define WL_TX_POWER_MCS20_SISO_FIRST_SSN   12 | ||||||
|  | 
 | ||||||
|  | /* struct tx_power::flags bits */ | ||||||
|  | #define WL_TX_POWER_F_ENABLED	1 | ||||||
|  | #define WL_TX_POWER_F_HW	2 | ||||||
|  | #define WL_TX_POWER_F_MIMO	4 | ||||||
|  | #define WL_TX_POWER_F_SISO	8 | ||||||
|  | 
 | ||||||
|  | /* values to force tx/rx chain */ | ||||||
|  | #define BRCMS_N_TXRX_CHAIN0		0 | ||||||
|  | #define BRCMS_N_TXRX_CHAIN1		1 | ||||||
|  | 
 | ||||||
|  | struct brcms_phy; | ||||||
|  | 
 | ||||||
|  | extern struct phy_shim_info *wlc_phy_shim_attach(struct brcms_hardware *wlc_hw, | ||||||
|  | 						 struct brcms_info *wl, | ||||||
|  | 						 struct brcms_c_info *wlc); | ||||||
|  | extern void wlc_phy_shim_detach(struct phy_shim_info *physhim); | ||||||
|  | 
 | ||||||
|  | /* PHY to WL utility functions */ | ||||||
|  | extern struct wlapi_timer *wlapi_init_timer(struct phy_shim_info *physhim, | ||||||
|  | 					    void (*fn) (struct brcms_phy *pi), | ||||||
|  | 					    void *arg, const char *name); | ||||||
|  | extern void wlapi_free_timer(struct phy_shim_info *physhim, | ||||||
|  | 			     struct wlapi_timer *t); | ||||||
|  | extern void wlapi_add_timer(struct phy_shim_info *physhim, | ||||||
|  | 			    struct wlapi_timer *t, uint ms, int periodic); | ||||||
|  | extern bool wlapi_del_timer(struct phy_shim_info *physhim, | ||||||
|  | 			    struct wlapi_timer *t); | ||||||
|  | extern void wlapi_intrson(struct phy_shim_info *physhim); | ||||||
|  | extern u32 wlapi_intrsoff(struct phy_shim_info *physhim); | ||||||
|  | extern void wlapi_intrsrestore(struct phy_shim_info *physhim, | ||||||
|  | 			       u32 macintmask); | ||||||
|  | 
 | ||||||
|  | extern void wlapi_bmac_write_shm(struct phy_shim_info *physhim, uint offset, | ||||||
|  | 				 u16 v); | ||||||
|  | extern u16 wlapi_bmac_read_shm(struct phy_shim_info *physhim, uint offset); | ||||||
|  | extern void wlapi_bmac_mhf(struct phy_shim_info *physhim, u8 idx, | ||||||
|  | 			   u16 mask, u16 val, int bands); | ||||||
|  | extern void wlapi_bmac_corereset(struct phy_shim_info *physhim, u32 flags); | ||||||
|  | extern void wlapi_suspend_mac_and_wait(struct phy_shim_info *physhim); | ||||||
|  | extern void wlapi_switch_macfreq(struct phy_shim_info *physhim, u8 spurmode); | ||||||
|  | extern void wlapi_enable_mac(struct phy_shim_info *physhim); | ||||||
|  | extern void wlapi_bmac_mctrl(struct phy_shim_info *physhim, u32 mask, | ||||||
|  | 			     u32 val); | ||||||
|  | extern void wlapi_bmac_phy_reset(struct phy_shim_info *physhim); | ||||||
|  | extern void wlapi_bmac_bw_set(struct phy_shim_info *physhim, u16 bw); | ||||||
|  | extern void wlapi_bmac_phyclk_fgc(struct phy_shim_info *physhim, bool clk); | ||||||
|  | extern void wlapi_bmac_macphyclk_set(struct phy_shim_info *physhim, bool clk); | ||||||
|  | extern void wlapi_bmac_core_phypll_ctl(struct phy_shim_info *physhim, bool on); | ||||||
|  | extern void wlapi_bmac_core_phypll_reset(struct phy_shim_info *physhim); | ||||||
|  | extern void wlapi_bmac_ucode_wake_override_phyreg_set(struct phy_shim_info * | ||||||
|  | 						      physhim); | ||||||
|  | extern void wlapi_bmac_ucode_wake_override_phyreg_clear(struct phy_shim_info * | ||||||
|  | 							physhim); | ||||||
|  | extern void wlapi_bmac_write_template_ram(struct phy_shim_info *physhim, int o, | ||||||
|  | 					  int len, void *buf); | ||||||
|  | extern u16 wlapi_bmac_rate_shm_offset(struct phy_shim_info *physhim, | ||||||
|  | 					 u8 rate); | ||||||
|  | extern void wlapi_ucode_sample_init(struct phy_shim_info *physhim); | ||||||
|  | extern void wlapi_copyfrom_objmem(struct phy_shim_info *physhim, uint, | ||||||
|  | 				  void *buf, int, u32 sel); | ||||||
|  | extern void wlapi_copyto_objmem(struct phy_shim_info *physhim, uint, | ||||||
|  | 				const void *buf, int, u32); | ||||||
|  | 
 | ||||||
|  | extern void wlapi_high_update_phy_mode(struct phy_shim_info *physhim, | ||||||
|  | 				       u32 phy_mode); | ||||||
|  | extern u16 wlapi_bmac_get_txant(struct phy_shim_info *physhim); | ||||||
|  | extern char *wlapi_getvar(struct phy_shim_info *physhim, enum brcms_srom_id id); | ||||||
|  | extern int wlapi_getintvar(struct phy_shim_info *physhim, | ||||||
|  | 			   enum brcms_srom_id id); | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCM_PHY_SHIM_H_ */ | ||||||
							
								
								
									
										458
									
								
								drivers/net/wireless/brcm80211/brcmsmac/pmu.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										458
									
								
								drivers/net/wireless/brcm80211/brcmsmac/pmu.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,458 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2011 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <linux/delay.h> | ||||||
|  | #include <linux/io.h> | ||||||
|  | 
 | ||||||
|  | #include <brcm_hw_ids.h> | ||||||
|  | #include <chipcommon.h> | ||||||
|  | #include <brcmu_utils.h> | ||||||
|  | #include "pub.h" | ||||||
|  | #include "aiutils.h" | ||||||
|  | #include "pmu.h" | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * external LPO crystal frequency | ||||||
|  |  */ | ||||||
|  | #define EXT_ILP_HZ 32768 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Duration for ILP clock frequency measurment in milliseconds | ||||||
|  |  * | ||||||
|  |  * remark: 1000 must be an integer multiple of this duration | ||||||
|  |  */ | ||||||
|  | #define ILP_CALC_DUR	10 | ||||||
|  | 
 | ||||||
|  | /* Fields in pmucontrol */ | ||||||
|  | #define	PCTL_ILP_DIV_MASK	0xffff0000 | ||||||
|  | #define	PCTL_ILP_DIV_SHIFT	16 | ||||||
|  | #define PCTL_PLL_PLLCTL_UPD	0x00000400	/* rev 2 */ | ||||||
|  | #define PCTL_NOILP_ON_WAIT	0x00000200	/* rev 1 */ | ||||||
|  | #define	PCTL_HT_REQ_EN		0x00000100 | ||||||
|  | #define	PCTL_ALP_REQ_EN		0x00000080 | ||||||
|  | #define	PCTL_XTALFREQ_MASK	0x0000007c | ||||||
|  | #define	PCTL_XTALFREQ_SHIFT	2 | ||||||
|  | #define	PCTL_ILP_DIV_EN		0x00000002 | ||||||
|  | #define	PCTL_LPO_SEL		0x00000001 | ||||||
|  | 
 | ||||||
|  | /* ILP clock */ | ||||||
|  | #define	ILP_CLOCK		32000 | ||||||
|  | 
 | ||||||
|  | /* ALP clock on pre-PMU chips */ | ||||||
|  | #define	ALP_CLOCK		20000000 | ||||||
|  | 
 | ||||||
|  | /* pmustatus */ | ||||||
|  | #define PST_EXTLPOAVAIL	0x0100 | ||||||
|  | #define PST_WDRESET	0x0080 | ||||||
|  | #define	PST_INTPEND	0x0040 | ||||||
|  | #define	PST_SBCLKST	0x0030 | ||||||
|  | #define	PST_SBCLKST_ILP	0x0010 | ||||||
|  | #define	PST_SBCLKST_ALP	0x0020 | ||||||
|  | #define	PST_SBCLKST_HT	0x0030 | ||||||
|  | #define	PST_ALPAVAIL	0x0008 | ||||||
|  | #define	PST_HTAVAIL	0x0004 | ||||||
|  | #define	PST_RESINIT	0x0003 | ||||||
|  | 
 | ||||||
|  | /* PMU resource bit position */ | ||||||
|  | #define PMURES_BIT(bit)	(1 << (bit)) | ||||||
|  | 
 | ||||||
|  | /* PMU corerev and chip specific PLL controls.
 | ||||||
|  |  * PMU<rev>_PLL<num>_XX where <rev> is PMU corerev and <num> is an arbitrary | ||||||
|  |  * number to differentiate different PLLs controlled by the same PMU rev. | ||||||
|  |  */ | ||||||
|  | /* pllcontrol registers:
 | ||||||
|  |  * ndiv_pwrdn, pwrdn_ch<x>, refcomp_pwrdn, dly_ch<x>, | ||||||
|  |  * p1div, p2div, _bypass_sdmod | ||||||
|  |  */ | ||||||
|  | #define PMU1_PLL0_PLLCTL0		0 | ||||||
|  | #define PMU1_PLL0_PLLCTL1		1 | ||||||
|  | #define PMU1_PLL0_PLLCTL2		2 | ||||||
|  | #define PMU1_PLL0_PLLCTL3		3 | ||||||
|  | #define PMU1_PLL0_PLLCTL4		4 | ||||||
|  | #define PMU1_PLL0_PLLCTL5		5 | ||||||
|  | 
 | ||||||
|  | /* pmu XtalFreqRatio */ | ||||||
|  | #define	PMU_XTALFREQ_REG_ILPCTR_MASK	0x00001FFF | ||||||
|  | #define	PMU_XTALFREQ_REG_MEASURE_MASK	0x80000000 | ||||||
|  | #define	PMU_XTALFREQ_REG_MEASURE_SHIFT	31 | ||||||
|  | 
 | ||||||
|  | /* 4313 resources */ | ||||||
|  | #define	RES4313_BB_PU_RSRC		0 | ||||||
|  | #define	RES4313_ILP_REQ_RSRC		1 | ||||||
|  | #define	RES4313_XTAL_PU_RSRC		2 | ||||||
|  | #define	RES4313_ALP_AVAIL_RSRC		3 | ||||||
|  | #define	RES4313_RADIO_PU_RSRC		4 | ||||||
|  | #define	RES4313_BG_PU_RSRC		5 | ||||||
|  | #define	RES4313_VREG1P4_PU_RSRC		6 | ||||||
|  | #define	RES4313_AFE_PWRSW_RSRC		7 | ||||||
|  | #define	RES4313_RX_PWRSW_RSRC		8 | ||||||
|  | #define	RES4313_TX_PWRSW_RSRC		9 | ||||||
|  | #define	RES4313_BB_PWRSW_RSRC		10 | ||||||
|  | #define	RES4313_SYNTH_PWRSW_RSRC	11 | ||||||
|  | #define	RES4313_MISC_PWRSW_RSRC		12 | ||||||
|  | #define	RES4313_BB_PLL_PWRSW_RSRC	13 | ||||||
|  | #define	RES4313_HT_AVAIL_RSRC		14 | ||||||
|  | #define	RES4313_MACPHY_CLK_AVAIL_RSRC	15 | ||||||
|  | 
 | ||||||
|  | /* Determine min/max rsrc masks. Value 0 leaves hardware at default. */ | ||||||
|  | static void si_pmu_res_masks(struct si_pub *sih, u32 * pmin, u32 * pmax) | ||||||
|  | { | ||||||
|  | 	u32 min_mask = 0, max_mask = 0; | ||||||
|  | 	uint rsrcs; | ||||||
|  | 
 | ||||||
|  | 	/* # resources */ | ||||||
|  | 	rsrcs = (sih->pmucaps & PCAP_RC_MASK) >> PCAP_RC_SHIFT; | ||||||
|  | 
 | ||||||
|  | 	/* determine min/max rsrc masks */ | ||||||
|  | 	switch (sih->chip) { | ||||||
|  | 	case BCM43224_CHIP_ID: | ||||||
|  | 	case BCM43225_CHIP_ID: | ||||||
|  | 		/* ??? */ | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case BCM4313_CHIP_ID: | ||||||
|  | 		min_mask = PMURES_BIT(RES4313_BB_PU_RSRC) | | ||||||
|  | 		    PMURES_BIT(RES4313_XTAL_PU_RSRC) | | ||||||
|  | 		    PMURES_BIT(RES4313_ALP_AVAIL_RSRC) | | ||||||
|  | 		    PMURES_BIT(RES4313_BB_PLL_PWRSW_RSRC); | ||||||
|  | 		max_mask = 0xffff; | ||||||
|  | 		break; | ||||||
|  | 	default: | ||||||
|  | 		break; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	*pmin = min_mask; | ||||||
|  | 	*pmax = max_mask; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void | ||||||
|  | si_pmu_spuravoid_pllupdate(struct si_pub *sih, struct chipcregs __iomem *cc, | ||||||
|  | 			   u8 spuravoid) | ||||||
|  | { | ||||||
|  | 	u32 tmp = 0; | ||||||
|  | 
 | ||||||
|  | 	switch (sih->chip) { | ||||||
|  | 	case BCM43224_CHIP_ID: | ||||||
|  | 	case BCM43225_CHIP_ID: | ||||||
|  | 		if (spuravoid == 1) { | ||||||
|  | 			W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL0); | ||||||
|  | 			W_REG(&cc->pllcontrol_data, 0x11500010); | ||||||
|  | 			W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL1); | ||||||
|  | 			W_REG(&cc->pllcontrol_data, 0x000C0C06); | ||||||
|  | 			W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL2); | ||||||
|  | 			W_REG(&cc->pllcontrol_data, 0x0F600a08); | ||||||
|  | 			W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL3); | ||||||
|  | 			W_REG(&cc->pllcontrol_data, 0x00000000); | ||||||
|  | 			W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL4); | ||||||
|  | 			W_REG(&cc->pllcontrol_data, 0x2001E920); | ||||||
|  | 			W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL5); | ||||||
|  | 			W_REG(&cc->pllcontrol_data, 0x88888815); | ||||||
|  | 		} else { | ||||||
|  | 			W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL0); | ||||||
|  | 			W_REG(&cc->pllcontrol_data, 0x11100010); | ||||||
|  | 			W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL1); | ||||||
|  | 			W_REG(&cc->pllcontrol_data, 0x000c0c06); | ||||||
|  | 			W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL2); | ||||||
|  | 			W_REG(&cc->pllcontrol_data, 0x03000a08); | ||||||
|  | 			W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL3); | ||||||
|  | 			W_REG(&cc->pllcontrol_data, 0x00000000); | ||||||
|  | 			W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL4); | ||||||
|  | 			W_REG(&cc->pllcontrol_data, 0x200005c0); | ||||||
|  | 			W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL5); | ||||||
|  | 			W_REG(&cc->pllcontrol_data, 0x88888815); | ||||||
|  | 		} | ||||||
|  | 		tmp = 1 << 10; | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 		W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL0); | ||||||
|  | 		W_REG(&cc->pllcontrol_data, 0x11100008); | ||||||
|  | 		W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL1); | ||||||
|  | 		W_REG(&cc->pllcontrol_data, 0x0c000c06); | ||||||
|  | 		W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL2); | ||||||
|  | 		W_REG(&cc->pllcontrol_data, 0x03000a08); | ||||||
|  | 		W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL3); | ||||||
|  | 		W_REG(&cc->pllcontrol_data, 0x00000000); | ||||||
|  | 		W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL4); | ||||||
|  | 		W_REG(&cc->pllcontrol_data, 0x200005c0); | ||||||
|  | 		W_REG(&cc->pllcontrol_addr, PMU1_PLL0_PLLCTL5); | ||||||
|  | 		W_REG(&cc->pllcontrol_data, 0x88888855); | ||||||
|  | 
 | ||||||
|  | 		tmp = 1 << 10; | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	default: | ||||||
|  | 		/* bail out */ | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	tmp |= R_REG(&cc->pmucontrol); | ||||||
|  | 	W_REG(&cc->pmucontrol, tmp); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | u16 si_pmu_fast_pwrup_delay(struct si_pub *sih) | ||||||
|  | { | ||||||
|  | 	uint delay = PMU_MAX_TRANSITION_DLY; | ||||||
|  | 
 | ||||||
|  | 	switch (sih->chip) { | ||||||
|  | 	case BCM43224_CHIP_ID: | ||||||
|  | 	case BCM43225_CHIP_ID: | ||||||
|  | 	case BCM4313_CHIP_ID: | ||||||
|  | 		delay = 3700; | ||||||
|  | 		break; | ||||||
|  | 	default: | ||||||
|  | 		break; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return (u16) delay; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void si_pmu_sprom_enable(struct si_pub *sih, bool enable) | ||||||
|  | { | ||||||
|  | 	struct chipcregs __iomem *cc; | ||||||
|  | 	uint origidx; | ||||||
|  | 
 | ||||||
|  | 	/* Remember original core before switch to chipc */ | ||||||
|  | 	origidx = ai_coreidx(sih); | ||||||
|  | 	cc = ai_setcoreidx(sih, SI_CC_IDX); | ||||||
|  | 
 | ||||||
|  | 	/* Return to original core */ | ||||||
|  | 	ai_setcoreidx(sih, origidx); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Read/write a chipcontrol reg */ | ||||||
|  | u32 si_pmu_chipcontrol(struct si_pub *sih, uint reg, u32 mask, u32 val) | ||||||
|  | { | ||||||
|  | 	ai_corereg(sih, SI_CC_IDX, offsetof(struct chipcregs, chipcontrol_addr), | ||||||
|  | 		   ~0, reg); | ||||||
|  | 	return ai_corereg(sih, SI_CC_IDX, | ||||||
|  | 			  offsetof(struct chipcregs, chipcontrol_data), mask, | ||||||
|  | 			  val); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Read/write a regcontrol reg */ | ||||||
|  | u32 si_pmu_regcontrol(struct si_pub *sih, uint reg, u32 mask, u32 val) | ||||||
|  | { | ||||||
|  | 	ai_corereg(sih, SI_CC_IDX, offsetof(struct chipcregs, regcontrol_addr), | ||||||
|  | 		   ~0, reg); | ||||||
|  | 	return ai_corereg(sih, SI_CC_IDX, | ||||||
|  | 			  offsetof(struct chipcregs, regcontrol_data), mask, | ||||||
|  | 			  val); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Read/write a pllcontrol reg */ | ||||||
|  | u32 si_pmu_pllcontrol(struct si_pub *sih, uint reg, u32 mask, u32 val) | ||||||
|  | { | ||||||
|  | 	ai_corereg(sih, SI_CC_IDX, offsetof(struct chipcregs, pllcontrol_addr), | ||||||
|  | 		   ~0, reg); | ||||||
|  | 	return ai_corereg(sih, SI_CC_IDX, | ||||||
|  | 			  offsetof(struct chipcregs, pllcontrol_data), mask, | ||||||
|  | 			  val); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* PMU PLL update */ | ||||||
|  | void si_pmu_pllupd(struct si_pub *sih) | ||||||
|  | { | ||||||
|  | 	ai_corereg(sih, SI_CC_IDX, offsetof(struct chipcregs, pmucontrol), | ||||||
|  | 		   PCTL_PLL_PLLCTL_UPD, PCTL_PLL_PLLCTL_UPD); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* query alp/xtal clock frequency */ | ||||||
|  | u32 si_pmu_alp_clock(struct si_pub *sih) | ||||||
|  | { | ||||||
|  | 	u32 clock = ALP_CLOCK; | ||||||
|  | 
 | ||||||
|  | 	/* bail out with default */ | ||||||
|  | 	if (!(sih->cccaps & CC_CAP_PMU)) | ||||||
|  | 		return clock; | ||||||
|  | 
 | ||||||
|  | 	switch (sih->chip) { | ||||||
|  | 	case BCM43224_CHIP_ID: | ||||||
|  | 	case BCM43225_CHIP_ID: | ||||||
|  | 	case BCM4313_CHIP_ID: | ||||||
|  | 		/* always 20Mhz */ | ||||||
|  | 		clock = 20000 * 1000; | ||||||
|  | 		break; | ||||||
|  | 	default: | ||||||
|  | 		break; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return clock; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void si_pmu_spuravoid(struct si_pub *sih, u8 spuravoid) | ||||||
|  | { | ||||||
|  | 	struct chipcregs __iomem *cc; | ||||||
|  | 	uint origidx, intr_val; | ||||||
|  | 
 | ||||||
|  | 	/* Remember original core before switch to chipc */ | ||||||
|  | 	cc = (struct chipcregs __iomem *) | ||||||
|  | 			ai_switch_core(sih, CC_CORE_ID, &origidx, &intr_val); | ||||||
|  | 
 | ||||||
|  | 	/* update the pll changes */ | ||||||
|  | 	si_pmu_spuravoid_pllupdate(sih, cc, spuravoid); | ||||||
|  | 
 | ||||||
|  | 	/* Return to original core */ | ||||||
|  | 	ai_restore_core(sih, origidx, intr_val); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* initialize PMU */ | ||||||
|  | void si_pmu_init(struct si_pub *sih) | ||||||
|  | { | ||||||
|  | 	struct chipcregs __iomem *cc; | ||||||
|  | 	uint origidx; | ||||||
|  | 
 | ||||||
|  | 	/* Remember original core before switch to chipc */ | ||||||
|  | 	origidx = ai_coreidx(sih); | ||||||
|  | 	cc = ai_setcoreidx(sih, SI_CC_IDX); | ||||||
|  | 
 | ||||||
|  | 	if (sih->pmurev == 1) | ||||||
|  | 		AND_REG(&cc->pmucontrol, ~PCTL_NOILP_ON_WAIT); | ||||||
|  | 	else if (sih->pmurev >= 2) | ||||||
|  | 		OR_REG(&cc->pmucontrol, PCTL_NOILP_ON_WAIT); | ||||||
|  | 
 | ||||||
|  | 	/* Return to original core */ | ||||||
|  | 	ai_setcoreidx(sih, origidx); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* initialize PMU chip controls and other chip level stuff */ | ||||||
|  | void si_pmu_chip_init(struct si_pub *sih) | ||||||
|  | { | ||||||
|  | 	uint origidx; | ||||||
|  | 
 | ||||||
|  | 	/* Gate off SPROM clock and chip select signals */ | ||||||
|  | 	si_pmu_sprom_enable(sih, false); | ||||||
|  | 
 | ||||||
|  | 	/* Remember original core */ | ||||||
|  | 	origidx = ai_coreidx(sih); | ||||||
|  | 
 | ||||||
|  | 	/* Return to original core */ | ||||||
|  | 	ai_setcoreidx(sih, origidx); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* initialize PMU switch/regulators */ | ||||||
|  | void si_pmu_swreg_init(struct si_pub *sih) | ||||||
|  | { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* initialize PLL */ | ||||||
|  | void si_pmu_pll_init(struct si_pub *sih, uint xtalfreq) | ||||||
|  | { | ||||||
|  | 	struct chipcregs __iomem *cc; | ||||||
|  | 	uint origidx; | ||||||
|  | 
 | ||||||
|  | 	/* Remember original core before switch to chipc */ | ||||||
|  | 	origidx = ai_coreidx(sih); | ||||||
|  | 	cc = ai_setcoreidx(sih, SI_CC_IDX); | ||||||
|  | 
 | ||||||
|  | 	switch (sih->chip) { | ||||||
|  | 	case BCM4313_CHIP_ID: | ||||||
|  | 	case BCM43224_CHIP_ID: | ||||||
|  | 	case BCM43225_CHIP_ID: | ||||||
|  | 		/* ??? */ | ||||||
|  | 		break; | ||||||
|  | 	default: | ||||||
|  | 		break; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Return to original core */ | ||||||
|  | 	ai_setcoreidx(sih, origidx); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* initialize PMU resources */ | ||||||
|  | void si_pmu_res_init(struct si_pub *sih) | ||||||
|  | { | ||||||
|  | 	struct chipcregs __iomem *cc; | ||||||
|  | 	uint origidx; | ||||||
|  | 	u32 min_mask = 0, max_mask = 0; | ||||||
|  | 
 | ||||||
|  | 	/* Remember original core before switch to chipc */ | ||||||
|  | 	origidx = ai_coreidx(sih); | ||||||
|  | 	cc = ai_setcoreidx(sih, SI_CC_IDX); | ||||||
|  | 
 | ||||||
|  | 	/* Determine min/max rsrc masks */ | ||||||
|  | 	si_pmu_res_masks(sih, &min_mask, &max_mask); | ||||||
|  | 
 | ||||||
|  | 	/* It is required to program max_mask first and then min_mask */ | ||||||
|  | 
 | ||||||
|  | 	/* Program max resource mask */ | ||||||
|  | 
 | ||||||
|  | 	if (max_mask) | ||||||
|  | 		W_REG(&cc->max_res_mask, max_mask); | ||||||
|  | 
 | ||||||
|  | 	/* Program min resource mask */ | ||||||
|  | 
 | ||||||
|  | 	if (min_mask) | ||||||
|  | 		W_REG(&cc->min_res_mask, min_mask); | ||||||
|  | 
 | ||||||
|  | 	/* Add some delay; allow resources to come up and settle. */ | ||||||
|  | 	mdelay(2); | ||||||
|  | 
 | ||||||
|  | 	/* Return to original core */ | ||||||
|  | 	ai_setcoreidx(sih, origidx); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | u32 si_pmu_measure_alpclk(struct si_pub *sih) | ||||||
|  | { | ||||||
|  | 	struct chipcregs __iomem *cc; | ||||||
|  | 	uint origidx; | ||||||
|  | 	u32 alp_khz; | ||||||
|  | 
 | ||||||
|  | 	if (sih->pmurev < 10) | ||||||
|  | 		return 0; | ||||||
|  | 
 | ||||||
|  | 	/* Remember original core before switch to chipc */ | ||||||
|  | 	origidx = ai_coreidx(sih); | ||||||
|  | 	cc = ai_setcoreidx(sih, SI_CC_IDX); | ||||||
|  | 
 | ||||||
|  | 	if (R_REG(&cc->pmustatus) & PST_EXTLPOAVAIL) { | ||||||
|  | 		u32 ilp_ctr, alp_hz; | ||||||
|  | 
 | ||||||
|  | 		/*
 | ||||||
|  | 		 * Enable the reg to measure the freq, | ||||||
|  | 		 * in case it was disabled before | ||||||
|  | 		 */ | ||||||
|  | 		W_REG(&cc->pmu_xtalfreq, | ||||||
|  | 		      1U << PMU_XTALFREQ_REG_MEASURE_SHIFT); | ||||||
|  | 
 | ||||||
|  | 		/* Delay for well over 4 ILP clocks */ | ||||||
|  | 		udelay(1000); | ||||||
|  | 
 | ||||||
|  | 		/* Read the latched number of ALP ticks per 4 ILP ticks */ | ||||||
|  | 		ilp_ctr = | ||||||
|  | 		    R_REG(&cc->pmu_xtalfreq) & PMU_XTALFREQ_REG_ILPCTR_MASK; | ||||||
|  | 
 | ||||||
|  | 		/*
 | ||||||
|  | 		 * Turn off the PMU_XTALFREQ_REG_MEASURE_SHIFT | ||||||
|  | 		 * bit to save power | ||||||
|  | 		 */ | ||||||
|  | 		W_REG(&cc->pmu_xtalfreq, 0); | ||||||
|  | 
 | ||||||
|  | 		/* Calculate ALP frequency */ | ||||||
|  | 		alp_hz = (ilp_ctr * EXT_ILP_HZ) / 4; | ||||||
|  | 
 | ||||||
|  | 		/*
 | ||||||
|  | 		 * Round to nearest 100KHz, and at | ||||||
|  | 		 * the same time convert to KHz | ||||||
|  | 		 */ | ||||||
|  | 		alp_khz = (alp_hz + 50000) / 100000 * 100; | ||||||
|  | 	} else | ||||||
|  | 		alp_khz = 0; | ||||||
|  | 
 | ||||||
|  | 	/* Return to original core */ | ||||||
|  | 	ai_setcoreidx(sih, origidx); | ||||||
|  | 
 | ||||||
|  | 	return alp_khz; | ||||||
|  | } | ||||||
							
								
								
									
										38
									
								
								drivers/net/wireless/brcm80211/brcmsmac/pmu.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								drivers/net/wireless/brcm80211/brcmsmac/pmu.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,38 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCM_PMU_H_ | ||||||
|  | #define _BRCM_PMU_H_ | ||||||
|  | 
 | ||||||
|  | #include "types.h" | ||||||
|  | 
 | ||||||
|  | extern u16 si_pmu_fast_pwrup_delay(struct si_pub *sih); | ||||||
|  | extern void si_pmu_sprom_enable(struct si_pub *sih, bool enable); | ||||||
|  | extern u32 si_pmu_chipcontrol(struct si_pub *sih, uint reg, u32 mask, u32 val); | ||||||
|  | extern u32 si_pmu_regcontrol(struct si_pub *sih, uint reg, u32 mask, u32 val); | ||||||
|  | extern u32 si_pmu_alp_clock(struct si_pub *sih); | ||||||
|  | extern void si_pmu_pllupd(struct si_pub *sih); | ||||||
|  | extern void si_pmu_spuravoid(struct si_pub *sih, u8 spuravoid); | ||||||
|  | extern u32 si_pmu_pllcontrol(struct si_pub *sih, uint reg, u32 mask, u32 val); | ||||||
|  | extern void si_pmu_init(struct si_pub *sih); | ||||||
|  | extern void si_pmu_chip_init(struct si_pub *sih); | ||||||
|  | extern void si_pmu_pll_init(struct si_pub *sih, u32 xtalfreq); | ||||||
|  | extern void si_pmu_res_init(struct si_pub *sih); | ||||||
|  | extern void si_pmu_swreg_init(struct si_pub *sih); | ||||||
|  | extern u32 si_pmu_measure_alpclk(struct si_pub *sih); | ||||||
|  | 
 | ||||||
|  | #endif /* _BRCM_PMU_H_ */ | ||||||
							
								
								
									
										655
									
								
								drivers/net/wireless/brcm80211/brcmsmac/pub.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										655
									
								
								drivers/net/wireless/brcm80211/brcmsmac/pub.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,655 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCM_PUB_H_ | ||||||
|  | #define _BRCM_PUB_H_ | ||||||
|  | 
 | ||||||
|  | #include <brcmu_wifi.h> | ||||||
|  | #include "types.h" | ||||||
|  | #include "defs.h" | ||||||
|  | 
 | ||||||
|  | enum brcms_srom_id { | ||||||
|  | 	BRCMS_SROM_NULL, | ||||||
|  | 	BRCMS_SROM_CONT, | ||||||
|  | 	BRCMS_SROM_AA2G, | ||||||
|  | 	BRCMS_SROM_AA5G, | ||||||
|  | 	BRCMS_SROM_AG0, | ||||||
|  | 	BRCMS_SROM_AG1, | ||||||
|  | 	BRCMS_SROM_AG2, | ||||||
|  | 	BRCMS_SROM_AG3, | ||||||
|  | 	BRCMS_SROM_ANTSWCTL2G, | ||||||
|  | 	BRCMS_SROM_ANTSWCTL5G, | ||||||
|  | 	BRCMS_SROM_ANTSWITCH, | ||||||
|  | 	BRCMS_SROM_BOARDFLAGS2, | ||||||
|  | 	BRCMS_SROM_BOARDFLAGS, | ||||||
|  | 	BRCMS_SROM_BOARDNUM, | ||||||
|  | 	BRCMS_SROM_BOARDREV, | ||||||
|  | 	BRCMS_SROM_BOARDTYPE, | ||||||
|  | 	BRCMS_SROM_BW40PO, | ||||||
|  | 	BRCMS_SROM_BWDUPPO, | ||||||
|  | 	BRCMS_SROM_BXA2G, | ||||||
|  | 	BRCMS_SROM_BXA5G, | ||||||
|  | 	BRCMS_SROM_CC, | ||||||
|  | 	BRCMS_SROM_CCK2GPO, | ||||||
|  | 	BRCMS_SROM_CCKBW202GPO, | ||||||
|  | 	BRCMS_SROM_CCKBW20UL2GPO, | ||||||
|  | 	BRCMS_SROM_CCODE, | ||||||
|  | 	BRCMS_SROM_CDDPO, | ||||||
|  | 	BRCMS_SROM_DEVID, | ||||||
|  | 	BRCMS_SROM_ET1MACADDR, | ||||||
|  | 	BRCMS_SROM_EXTPAGAIN2G, | ||||||
|  | 	BRCMS_SROM_EXTPAGAIN5G, | ||||||
|  | 	BRCMS_SROM_FREQOFFSET_CORR, | ||||||
|  | 	BRCMS_SROM_HW_IQCAL_EN, | ||||||
|  | 	BRCMS_SROM_IL0MACADDR, | ||||||
|  | 	BRCMS_SROM_IQCAL_SWP_DIS, | ||||||
|  | 	BRCMS_SROM_LEDBH0, | ||||||
|  | 	BRCMS_SROM_LEDBH1, | ||||||
|  | 	BRCMS_SROM_LEDBH2, | ||||||
|  | 	BRCMS_SROM_LEDBH3, | ||||||
|  | 	BRCMS_SROM_LEDDC, | ||||||
|  | 	BRCMS_SROM_LEGOFDM40DUPPO, | ||||||
|  | 	BRCMS_SROM_LEGOFDMBW202GPO, | ||||||
|  | 	BRCMS_SROM_LEGOFDMBW205GHPO, | ||||||
|  | 	BRCMS_SROM_LEGOFDMBW205GLPO, | ||||||
|  | 	BRCMS_SROM_LEGOFDMBW205GMPO, | ||||||
|  | 	BRCMS_SROM_LEGOFDMBW20UL2GPO, | ||||||
|  | 	BRCMS_SROM_LEGOFDMBW20UL5GHPO, | ||||||
|  | 	BRCMS_SROM_LEGOFDMBW20UL5GLPO, | ||||||
|  | 	BRCMS_SROM_LEGOFDMBW20UL5GMPO, | ||||||
|  | 	BRCMS_SROM_MACADDR, | ||||||
|  | 	BRCMS_SROM_MCS2GPO0, | ||||||
|  | 	BRCMS_SROM_MCS2GPO1, | ||||||
|  | 	BRCMS_SROM_MCS2GPO2, | ||||||
|  | 	BRCMS_SROM_MCS2GPO3, | ||||||
|  | 	BRCMS_SROM_MCS2GPO4, | ||||||
|  | 	BRCMS_SROM_MCS2GPO5, | ||||||
|  | 	BRCMS_SROM_MCS2GPO6, | ||||||
|  | 	BRCMS_SROM_MCS2GPO7, | ||||||
|  | 	BRCMS_SROM_MCS32PO, | ||||||
|  | 	BRCMS_SROM_MCS5GHPO0, | ||||||
|  | 	BRCMS_SROM_MCS5GHPO1, | ||||||
|  | 	BRCMS_SROM_MCS5GHPO2, | ||||||
|  | 	BRCMS_SROM_MCS5GHPO3, | ||||||
|  | 	BRCMS_SROM_MCS5GHPO4, | ||||||
|  | 	BRCMS_SROM_MCS5GHPO5, | ||||||
|  | 	BRCMS_SROM_MCS5GHPO6, | ||||||
|  | 	BRCMS_SROM_MCS5GHPO7, | ||||||
|  | 	BRCMS_SROM_MCS5GLPO0, | ||||||
|  | 	BRCMS_SROM_MCS5GLPO1, | ||||||
|  | 	BRCMS_SROM_MCS5GLPO2, | ||||||
|  | 	BRCMS_SROM_MCS5GLPO3, | ||||||
|  | 	BRCMS_SROM_MCS5GLPO4, | ||||||
|  | 	BRCMS_SROM_MCS5GLPO5, | ||||||
|  | 	BRCMS_SROM_MCS5GLPO6, | ||||||
|  | 	BRCMS_SROM_MCS5GLPO7, | ||||||
|  | 	BRCMS_SROM_MCS5GPO0, | ||||||
|  | 	BRCMS_SROM_MCS5GPO1, | ||||||
|  | 	BRCMS_SROM_MCS5GPO2, | ||||||
|  | 	BRCMS_SROM_MCS5GPO3, | ||||||
|  | 	BRCMS_SROM_MCS5GPO4, | ||||||
|  | 	BRCMS_SROM_MCS5GPO5, | ||||||
|  | 	BRCMS_SROM_MCS5GPO6, | ||||||
|  | 	BRCMS_SROM_MCS5GPO7, | ||||||
|  | 	BRCMS_SROM_MCSBW202GPO, | ||||||
|  | 	BRCMS_SROM_MCSBW205GHPO, | ||||||
|  | 	BRCMS_SROM_MCSBW205GLPO, | ||||||
|  | 	BRCMS_SROM_MCSBW205GMPO, | ||||||
|  | 	BRCMS_SROM_MCSBW20UL2GPO, | ||||||
|  | 	BRCMS_SROM_MCSBW20UL5GHPO, | ||||||
|  | 	BRCMS_SROM_MCSBW20UL5GLPO, | ||||||
|  | 	BRCMS_SROM_MCSBW20UL5GMPO, | ||||||
|  | 	BRCMS_SROM_MCSBW402GPO, | ||||||
|  | 	BRCMS_SROM_MCSBW405GHPO, | ||||||
|  | 	BRCMS_SROM_MCSBW405GLPO, | ||||||
|  | 	BRCMS_SROM_MCSBW405GMPO, | ||||||
|  | 	BRCMS_SROM_MEASPOWER, | ||||||
|  | 	BRCMS_SROM_OFDM2GPO, | ||||||
|  | 	BRCMS_SROM_OFDM5GHPO, | ||||||
|  | 	BRCMS_SROM_OFDM5GLPO, | ||||||
|  | 	BRCMS_SROM_OFDM5GPO, | ||||||
|  | 	BRCMS_SROM_OPO, | ||||||
|  | 	BRCMS_SROM_PA0B0, | ||||||
|  | 	BRCMS_SROM_PA0B1, | ||||||
|  | 	BRCMS_SROM_PA0B2, | ||||||
|  | 	BRCMS_SROM_PA0ITSSIT, | ||||||
|  | 	BRCMS_SROM_PA0MAXPWR, | ||||||
|  | 	BRCMS_SROM_PA1B0, | ||||||
|  | 	BRCMS_SROM_PA1B1, | ||||||
|  | 	BRCMS_SROM_PA1B2, | ||||||
|  | 	BRCMS_SROM_PA1HIB0, | ||||||
|  | 	BRCMS_SROM_PA1HIB1, | ||||||
|  | 	BRCMS_SROM_PA1HIB2, | ||||||
|  | 	BRCMS_SROM_PA1HIMAXPWR, | ||||||
|  | 	BRCMS_SROM_PA1ITSSIT, | ||||||
|  | 	BRCMS_SROM_PA1LOB0, | ||||||
|  | 	BRCMS_SROM_PA1LOB1, | ||||||
|  | 	BRCMS_SROM_PA1LOB2, | ||||||
|  | 	BRCMS_SROM_PA1LOMAXPWR, | ||||||
|  | 	BRCMS_SROM_PA1MAXPWR, | ||||||
|  | 	BRCMS_SROM_PDETRANGE2G, | ||||||
|  | 	BRCMS_SROM_PDETRANGE5G, | ||||||
|  | 	BRCMS_SROM_PHYCAL_TEMPDELTA, | ||||||
|  | 	BRCMS_SROM_RAWTEMPSENSE, | ||||||
|  | 	BRCMS_SROM_REGREV, | ||||||
|  | 	BRCMS_SROM_REV, | ||||||
|  | 	BRCMS_SROM_RSSISAV2G, | ||||||
|  | 	BRCMS_SROM_RSSISAV5G, | ||||||
|  | 	BRCMS_SROM_RSSISMC2G, | ||||||
|  | 	BRCMS_SROM_RSSISMC5G, | ||||||
|  | 	BRCMS_SROM_RSSISMF2G, | ||||||
|  | 	BRCMS_SROM_RSSISMF5G, | ||||||
|  | 	BRCMS_SROM_RXCHAIN, | ||||||
|  | 	BRCMS_SROM_RXPO2G, | ||||||
|  | 	BRCMS_SROM_RXPO5G, | ||||||
|  | 	BRCMS_SROM_STBCPO, | ||||||
|  | 	BRCMS_SROM_TEMPCORRX, | ||||||
|  | 	BRCMS_SROM_TEMPOFFSET, | ||||||
|  | 	BRCMS_SROM_TEMPSENSE_OPTION, | ||||||
|  | 	BRCMS_SROM_TEMPSENSE_SLOPE, | ||||||
|  | 	BRCMS_SROM_TEMPTHRESH, | ||||||
|  | 	BRCMS_SROM_TRI2G, | ||||||
|  | 	BRCMS_SROM_TRI5GH, | ||||||
|  | 	BRCMS_SROM_TRI5GL, | ||||||
|  | 	BRCMS_SROM_TRI5G, | ||||||
|  | 	BRCMS_SROM_TRISO2G, | ||||||
|  | 	BRCMS_SROM_TRISO5G, | ||||||
|  | 	BRCMS_SROM_TSSIPOS2G, | ||||||
|  | 	BRCMS_SROM_TSSIPOS5G, | ||||||
|  | 	BRCMS_SROM_TXCHAIN, | ||||||
|  | 	BRCMS_SROM_TXPID2GA0, | ||||||
|  | 	BRCMS_SROM_TXPID2GA1, | ||||||
|  | 	BRCMS_SROM_TXPID2GA2, | ||||||
|  | 	BRCMS_SROM_TXPID2GA3, | ||||||
|  | 	BRCMS_SROM_TXPID5GA0, | ||||||
|  | 	BRCMS_SROM_TXPID5GA1, | ||||||
|  | 	BRCMS_SROM_TXPID5GA2, | ||||||
|  | 	BRCMS_SROM_TXPID5GA3, | ||||||
|  | 	BRCMS_SROM_TXPID5GHA0, | ||||||
|  | 	BRCMS_SROM_TXPID5GHA1, | ||||||
|  | 	BRCMS_SROM_TXPID5GHA2, | ||||||
|  | 	BRCMS_SROM_TXPID5GHA3, | ||||||
|  | 	BRCMS_SROM_TXPID5GLA0, | ||||||
|  | 	BRCMS_SROM_TXPID5GLA1, | ||||||
|  | 	BRCMS_SROM_TXPID5GLA2, | ||||||
|  | 	BRCMS_SROM_TXPID5GLA3, | ||||||
|  | 	/*
 | ||||||
|  | 	 * per-path identifiers (see srom.c) | ||||||
|  | 	 */ | ||||||
|  | 	BRCMS_SROM_ITT2GA0, | ||||||
|  | 	BRCMS_SROM_ITT2GA1, | ||||||
|  | 	BRCMS_SROM_ITT2GA2, | ||||||
|  | 	BRCMS_SROM_ITT2GA3, | ||||||
|  | 	BRCMS_SROM_ITT5GA0, | ||||||
|  | 	BRCMS_SROM_ITT5GA1, | ||||||
|  | 	BRCMS_SROM_ITT5GA2, | ||||||
|  | 	BRCMS_SROM_ITT5GA3, | ||||||
|  | 	BRCMS_SROM_MAXP2GA0, | ||||||
|  | 	BRCMS_SROM_MAXP2GA1, | ||||||
|  | 	BRCMS_SROM_MAXP2GA2, | ||||||
|  | 	BRCMS_SROM_MAXP2GA3, | ||||||
|  | 	BRCMS_SROM_MAXP5GA0, | ||||||
|  | 	BRCMS_SROM_MAXP5GA1, | ||||||
|  | 	BRCMS_SROM_MAXP5GA2, | ||||||
|  | 	BRCMS_SROM_MAXP5GA3, | ||||||
|  | 	BRCMS_SROM_MAXP5GHA0, | ||||||
|  | 	BRCMS_SROM_MAXP5GHA1, | ||||||
|  | 	BRCMS_SROM_MAXP5GHA2, | ||||||
|  | 	BRCMS_SROM_MAXP5GHA3, | ||||||
|  | 	BRCMS_SROM_MAXP5GLA0, | ||||||
|  | 	BRCMS_SROM_MAXP5GLA1, | ||||||
|  | 	BRCMS_SROM_MAXP5GLA2, | ||||||
|  | 	BRCMS_SROM_MAXP5GLA3, | ||||||
|  | 	BRCMS_SROM_PA2GW0A0, | ||||||
|  | 	BRCMS_SROM_PA2GW0A1, | ||||||
|  | 	BRCMS_SROM_PA2GW0A2, | ||||||
|  | 	BRCMS_SROM_PA2GW0A3, | ||||||
|  | 	BRCMS_SROM_PA2GW1A0, | ||||||
|  | 	BRCMS_SROM_PA2GW1A1, | ||||||
|  | 	BRCMS_SROM_PA2GW1A2, | ||||||
|  | 	BRCMS_SROM_PA2GW1A3, | ||||||
|  | 	BRCMS_SROM_PA2GW2A0, | ||||||
|  | 	BRCMS_SROM_PA2GW2A1, | ||||||
|  | 	BRCMS_SROM_PA2GW2A2, | ||||||
|  | 	BRCMS_SROM_PA2GW2A3, | ||||||
|  | 	BRCMS_SROM_PA2GW3A0, | ||||||
|  | 	BRCMS_SROM_PA2GW3A1, | ||||||
|  | 	BRCMS_SROM_PA2GW3A2, | ||||||
|  | 	BRCMS_SROM_PA2GW3A3, | ||||||
|  | 	BRCMS_SROM_PA5GHW0A0, | ||||||
|  | 	BRCMS_SROM_PA5GHW0A1, | ||||||
|  | 	BRCMS_SROM_PA5GHW0A2, | ||||||
|  | 	BRCMS_SROM_PA5GHW0A3, | ||||||
|  | 	BRCMS_SROM_PA5GHW1A0, | ||||||
|  | 	BRCMS_SROM_PA5GHW1A1, | ||||||
|  | 	BRCMS_SROM_PA5GHW1A2, | ||||||
|  | 	BRCMS_SROM_PA5GHW1A3, | ||||||
|  | 	BRCMS_SROM_PA5GHW2A0, | ||||||
|  | 	BRCMS_SROM_PA5GHW2A1, | ||||||
|  | 	BRCMS_SROM_PA5GHW2A2, | ||||||
|  | 	BRCMS_SROM_PA5GHW2A3, | ||||||
|  | 	BRCMS_SROM_PA5GHW3A0, | ||||||
|  | 	BRCMS_SROM_PA5GHW3A1, | ||||||
|  | 	BRCMS_SROM_PA5GHW3A2, | ||||||
|  | 	BRCMS_SROM_PA5GHW3A3, | ||||||
|  | 	BRCMS_SROM_PA5GLW0A0, | ||||||
|  | 	BRCMS_SROM_PA5GLW0A1, | ||||||
|  | 	BRCMS_SROM_PA5GLW0A2, | ||||||
|  | 	BRCMS_SROM_PA5GLW0A3, | ||||||
|  | 	BRCMS_SROM_PA5GLW1A0, | ||||||
|  | 	BRCMS_SROM_PA5GLW1A1, | ||||||
|  | 	BRCMS_SROM_PA5GLW1A2, | ||||||
|  | 	BRCMS_SROM_PA5GLW1A3, | ||||||
|  | 	BRCMS_SROM_PA5GLW2A0, | ||||||
|  | 	BRCMS_SROM_PA5GLW2A1, | ||||||
|  | 	BRCMS_SROM_PA5GLW2A2, | ||||||
|  | 	BRCMS_SROM_PA5GLW2A3, | ||||||
|  | 	BRCMS_SROM_PA5GLW3A0, | ||||||
|  | 	BRCMS_SROM_PA5GLW3A1, | ||||||
|  | 	BRCMS_SROM_PA5GLW3A2, | ||||||
|  | 	BRCMS_SROM_PA5GLW3A3, | ||||||
|  | 	BRCMS_SROM_PA5GW0A0, | ||||||
|  | 	BRCMS_SROM_PA5GW0A1, | ||||||
|  | 	BRCMS_SROM_PA5GW0A2, | ||||||
|  | 	BRCMS_SROM_PA5GW0A3, | ||||||
|  | 	BRCMS_SROM_PA5GW1A0, | ||||||
|  | 	BRCMS_SROM_PA5GW1A1, | ||||||
|  | 	BRCMS_SROM_PA5GW1A2, | ||||||
|  | 	BRCMS_SROM_PA5GW1A3, | ||||||
|  | 	BRCMS_SROM_PA5GW2A0, | ||||||
|  | 	BRCMS_SROM_PA5GW2A1, | ||||||
|  | 	BRCMS_SROM_PA5GW2A2, | ||||||
|  | 	BRCMS_SROM_PA5GW2A3, | ||||||
|  | 	BRCMS_SROM_PA5GW3A0, | ||||||
|  | 	BRCMS_SROM_PA5GW3A1, | ||||||
|  | 	BRCMS_SROM_PA5GW3A2, | ||||||
|  | 	BRCMS_SROM_PA5GW3A3, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #define	BRCMS_NUMRATES	16	/* max # of rates in a rateset */ | ||||||
|  | #define	D11_PHY_HDR_LEN	6	/* Phy header length - 6 bytes */ | ||||||
|  | 
 | ||||||
|  | /* phy types */ | ||||||
|  | #define	PHY_TYPE_A	0	/* Phy type A */ | ||||||
|  | #define	PHY_TYPE_G	2	/* Phy type G */ | ||||||
|  | #define	PHY_TYPE_N	4	/* Phy type N */ | ||||||
|  | #define	PHY_TYPE_LP	5	/* Phy type Low Power A/B/G */ | ||||||
|  | #define	PHY_TYPE_SSN	6	/* Phy type Single Stream N */ | ||||||
|  | #define	PHY_TYPE_LCN	8	/* Phy type Single Stream N */ | ||||||
|  | #define	PHY_TYPE_LCNXN	9	/* Phy type 2-stream N */ | ||||||
|  | #define	PHY_TYPE_HT	7	/* Phy type 3-Stream N */ | ||||||
|  | 
 | ||||||
|  | /* bw */ | ||||||
|  | #define BRCMS_10_MHZ	10	/* 10Mhz nphy channel bandwidth */ | ||||||
|  | #define BRCMS_20_MHZ	20	/* 20Mhz nphy channel bandwidth */ | ||||||
|  | #define BRCMS_40_MHZ	40	/* 40Mhz nphy channel bandwidth */ | ||||||
|  | 
 | ||||||
|  | #define	BRCMS_RSSI_MINVAL	-200	/* Low value, e.g. for forcing roam */ | ||||||
|  | #define	BRCMS_RSSI_NO_SIGNAL	-91	/* NDIS RSSI link quality cutoffs */ | ||||||
|  | #define	BRCMS_RSSI_VERY_LOW	-80	/* Very low quality cutoffs */ | ||||||
|  | #define	BRCMS_RSSI_LOW		-70	/* Low quality cutoffs */ | ||||||
|  | #define	BRCMS_RSSI_GOOD		-68	/* Good quality cutoffs */ | ||||||
|  | #define	BRCMS_RSSI_VERY_GOOD	-58	/* Very good quality cutoffs */ | ||||||
|  | #define	BRCMS_RSSI_EXCELLENT	-57	/* Excellent quality cutoffs */ | ||||||
|  | 
 | ||||||
|  | /* a large TX Power as an init value to factor out of min() calculations,
 | ||||||
|  |  * keep low enough to fit in an s8, units are .25 dBm | ||||||
|  |  */ | ||||||
|  | #define BRCMS_TXPWR_MAX		(127)	/* ~32 dBm = 1,500 mW */ | ||||||
|  | 
 | ||||||
|  | /* rate related definitions */ | ||||||
|  | #define	BRCMS_RATE_FLAG	0x80	/* Flag to indicate it is a basic rate */ | ||||||
|  | #define	BRCMS_RATE_MASK	0x7f	/* Rate value mask w/o basic rate flag */ | ||||||
|  | 
 | ||||||
|  | /* legacy rx Antenna diversity for SISO rates */ | ||||||
|  | #define	ANT_RX_DIV_FORCE_0	0	/* Use antenna 0 */ | ||||||
|  | #define	ANT_RX_DIV_FORCE_1	1	/* Use antenna 1 */ | ||||||
|  | #define	ANT_RX_DIV_START_1	2	/* Choose starting with 1 */ | ||||||
|  | #define	ANT_RX_DIV_START_0	3	/* Choose starting with 0 */ | ||||||
|  | #define	ANT_RX_DIV_ENABLE	3	/* APHY bbConfig Enable RX Diversity */ | ||||||
|  | /* default antdiv setting */ | ||||||
|  | #define ANT_RX_DIV_DEF		ANT_RX_DIV_START_0 | ||||||
|  | 
 | ||||||
|  | /* legacy rx Antenna diversity for SISO rates */ | ||||||
|  | /* Tx on antenna 0, "legacy term Main" */ | ||||||
|  | #define ANT_TX_FORCE_0		0 | ||||||
|  | /* Tx on antenna 1, "legacy term Aux" */ | ||||||
|  | #define ANT_TX_FORCE_1		1 | ||||||
|  | /* Tx on phy's last good Rx antenna */ | ||||||
|  | #define ANT_TX_LAST_RX		3 | ||||||
|  | /* driver's default tx antenna setting */ | ||||||
|  | #define ANT_TX_DEF		3 | ||||||
|  | 
 | ||||||
|  | /* Tx Chain values */ | ||||||
|  | /* def bitmap of txchain */ | ||||||
|  | #define TXCHAIN_DEF		0x1 | ||||||
|  | /* default bitmap of tx chains for nphy */ | ||||||
|  | #define TXCHAIN_DEF_NPHY	0x3 | ||||||
|  | /* default bitmap of tx chains for nphy */ | ||||||
|  | #define TXCHAIN_DEF_HTPHY	0x7 | ||||||
|  | /* def bitmap of rxchain */ | ||||||
|  | #define RXCHAIN_DEF		0x1 | ||||||
|  | /* default bitmap of rx chains for nphy */ | ||||||
|  | #define RXCHAIN_DEF_NPHY	0x3 | ||||||
|  | /* default bitmap of rx chains for nphy */ | ||||||
|  | #define RXCHAIN_DEF_HTPHY	0x7 | ||||||
|  | /* no antenna switch */ | ||||||
|  | #define ANTSWITCH_NONE		0 | ||||||
|  | /* antenna switch on 4321CB2, 2of3 */ | ||||||
|  | #define ANTSWITCH_TYPE_1	1 | ||||||
|  | /* antenna switch on 4321MPCI, 2of3 */ | ||||||
|  | #define ANTSWITCH_TYPE_2	2 | ||||||
|  | /* antenna switch on 4322, 2of3 */ | ||||||
|  | #define ANTSWITCH_TYPE_3	3 | ||||||
|  | 
 | ||||||
|  | #define RXBUFSZ		PKTBUFSZ | ||||||
|  | 
 | ||||||
|  | #define MAX_STREAMS_SUPPORTED	4	/* max number of streams supported */ | ||||||
|  | 
 | ||||||
|  | struct brcm_rateset { | ||||||
|  | 	/* # rates in this set */ | ||||||
|  | 	u32 count; | ||||||
|  | 	/* rates in 500kbps units w/hi bit set if basic */ | ||||||
|  | 	u8 rates[WL_NUMRATES]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct brcms_c_rateset { | ||||||
|  | 	uint count;		/* number of rates in rates[] */ | ||||||
|  | 	 /* rates in 500kbps units w/hi bit set if basic */ | ||||||
|  | 	u8 rates[BRCMS_NUMRATES]; | ||||||
|  | 	u8 htphy_membership;	/* HT PHY Membership */ | ||||||
|  | 	u8 mcs[MCSSET_LEN];	/* supported mcs index bit map */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* All the HT-specific default advertised capabilities (including AMPDU)
 | ||||||
|  |  * should be grouped here at one place | ||||||
|  |  */ | ||||||
|  | #define AMPDU_DEF_MPDU_DENSITY	6	/* default mpdu density (110 ==> 4us) */ | ||||||
|  | 
 | ||||||
|  | /* wlc internal bss_info */ | ||||||
|  | struct brcms_bss_info { | ||||||
|  | 	u8 BSSID[ETH_ALEN];	/* network BSSID */ | ||||||
|  | 	u16 flags;		/* flags for internal attributes */ | ||||||
|  | 	u8 SSID_len;		/* the length of SSID */ | ||||||
|  | 	u8 SSID[32];		/* SSID string */ | ||||||
|  | 	s16 RSSI;		/* receive signal strength (in dBm) */ | ||||||
|  | 	s16 SNR;		/* receive signal SNR in dB */ | ||||||
|  | 	u16 beacon_period;	/* units are Kusec */ | ||||||
|  | 	u16 chanspec;	/* Channel num, bw, ctrl_sb and band */ | ||||||
|  | 	struct brcms_c_rateset rateset;	/* supported rates */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #define MAC80211_PROMISC_BCNS	(1 << 0) | ||||||
|  | #define MAC80211_SCAN		(1 << 1) | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Public portion of common driver state structure. | ||||||
|  |  * The wlc handle points at this. | ||||||
|  |  */ | ||||||
|  | struct brcms_pub { | ||||||
|  | 	struct brcms_c_info *wlc; | ||||||
|  | 	struct ieee80211_hw *ieee_hw; | ||||||
|  | 	struct scb_ampdu *global_ampdu; | ||||||
|  | 	uint mac80211_state; | ||||||
|  | 	uint unit;		/* device instance number */ | ||||||
|  | 	uint corerev;		/* core revision */ | ||||||
|  | 	struct si_pub *sih;	/* SI handle (cookie for siutils calls) */ | ||||||
|  | 	bool up;		/* interface up and running */ | ||||||
|  | 	bool hw_off;		/* HW is off */ | ||||||
|  | 	bool hw_up;		/* one time hw up/down */ | ||||||
|  | 	bool _piomode;		/* true if pio mode */ | ||||||
|  | 	uint _nbands;		/* # bands supported */ | ||||||
|  | 	uint now;		/* # elapsed seconds */ | ||||||
|  | 
 | ||||||
|  | 	bool promisc;		/* promiscuous destination address */ | ||||||
|  | 	bool delayed_down;	/* down delayed */ | ||||||
|  | 	bool associated;	/* true:part of [I]BSS, false: not */ | ||||||
|  | 	/* (union of stas_associated, aps_associated) */ | ||||||
|  | 	bool _ampdu;		/* ampdu enabled or not */ | ||||||
|  | 	u8 _n_enab;		/* bitmap of 11N + HT support */ | ||||||
|  | 
 | ||||||
|  | 	u8 cur_etheraddr[ETH_ALEN];	/* our local ethernet address */ | ||||||
|  | 
 | ||||||
|  | 	int bcmerror;		/* last bcm error */ | ||||||
|  | 
 | ||||||
|  | 	u32 radio_disabled;	/* bit vector for radio disabled reasons */ | ||||||
|  | 
 | ||||||
|  | 	u16 boardrev;	/* version # of particular board */ | ||||||
|  | 	u8 sromrev;		/* version # of the srom */ | ||||||
|  | 	char srom_ccode[BRCM_CNTRY_BUF_SZ];	/* Country Code in SROM */ | ||||||
|  | 	u32 boardflags;	/* Board specific flags from srom */ | ||||||
|  | 	u32 boardflags2;	/* More board flags if sromrev >= 4 */ | ||||||
|  | 	bool phy_11ncapable;	/* the PHY/HW is capable of 802.11N */ | ||||||
|  | 
 | ||||||
|  | 	struct wl_cnt *_cnt;	/* low-level counters in driver */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | enum wlc_par_id { | ||||||
|  | 	IOV_MPC = 1, | ||||||
|  | 	IOV_RTSTHRESH, | ||||||
|  | 	IOV_QTXPOWER, | ||||||
|  | 	IOV_BCN_LI_BCN		/* Beacon listen interval in # of beacons */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /***********************************************
 | ||||||
|  |  * Feature-related macros to optimize out code * | ||||||
|  |  * ********************************************* | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #define ENAB_1x1	0x01 | ||||||
|  | #define ENAB_2x2	0x02 | ||||||
|  | #define ENAB_3x3	0x04 | ||||||
|  | #define ENAB_4x4	0x08 | ||||||
|  | #define SUPPORT_11N	(ENAB_1x1|ENAB_2x2) | ||||||
|  | #define SUPPORT_HT	(ENAB_1x1|ENAB_2x2|ENAB_3x3) | ||||||
|  | 
 | ||||||
|  | /* WL11N Support */ | ||||||
|  | #define AMPDU_AGG_HOST	1 | ||||||
|  | 
 | ||||||
|  | /* pri is priority encoded in the packet. This maps the Packet priority to
 | ||||||
|  |  * enqueue precedence as defined in wlc_prec_map | ||||||
|  |  */ | ||||||
|  | extern const u8 wlc_prio2prec_map[]; | ||||||
|  | #define BRCMS_PRIO_TO_PREC(pri)	wlc_prio2prec_map[(pri) & 7] | ||||||
|  | 
 | ||||||
|  | #define	BRCMS_PREC_COUNT	16	/* Max precedence level implemented */ | ||||||
|  | 
 | ||||||
|  | /* Mask to describe all precedence levels */ | ||||||
|  | #define BRCMS_PREC_BMP_ALL		MAXBITVAL(BRCMS_PREC_COUNT) | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * This maps priority to one precedence higher - Used by PS-Poll response | ||||||
|  |  * packets to simulate enqueue-at-head operation, but still maintain the | ||||||
|  |  * order on the queue | ||||||
|  |  */ | ||||||
|  | #define BRCMS_PRIO_TO_HI_PREC(pri)	min(BRCMS_PRIO_TO_PREC(pri) + 1,\ | ||||||
|  | 					    BRCMS_PREC_COUNT - 1) | ||||||
|  | 
 | ||||||
|  | /* Define a bitmap of precedences comprised by each AC */ | ||||||
|  | #define BRCMS_PREC_BMP_AC_BE	(NBITVAL(BRCMS_PRIO_TO_PREC(PRIO_8021D_BE)) | \ | ||||||
|  | 			NBITVAL(BRCMS_PRIO_TO_HI_PREC(PRIO_8021D_BE)) |	\ | ||||||
|  | 			NBITVAL(BRCMS_PRIO_TO_PREC(PRIO_8021D_EE)) |	\ | ||||||
|  | 			NBITVAL(BRCMS_PRIO_TO_HI_PREC(PRIO_8021D_EE))) | ||||||
|  | #define BRCMS_PREC_BMP_AC_BK	(NBITVAL(BRCMS_PRIO_TO_PREC(PRIO_8021D_BK)) | \ | ||||||
|  | 			NBITVAL(BRCMS_PRIO_TO_HI_PREC(PRIO_8021D_BK)) |	\ | ||||||
|  | 			NBITVAL(BRCMS_PRIO_TO_PREC(PRIO_8021D_NONE)) |	\ | ||||||
|  | 			NBITVAL(BRCMS_PRIO_TO_HI_PREC(PRIO_8021D_NONE))) | ||||||
|  | #define BRCMS_PREC_BMP_AC_VI	(NBITVAL(BRCMS_PRIO_TO_PREC(PRIO_8021D_CL)) | \ | ||||||
|  | 			NBITVAL(BRCMS_PRIO_TO_HI_PREC(PRIO_8021D_CL)) |	\ | ||||||
|  | 			NBITVAL(BRCMS_PRIO_TO_PREC(PRIO_8021D_VI)) |	\ | ||||||
|  | 			NBITVAL(BRCMS_PRIO_TO_HI_PREC(PRIO_8021D_VI))) | ||||||
|  | #define BRCMS_PREC_BMP_AC_VO	(NBITVAL(BRCMS_PRIO_TO_PREC(PRIO_8021D_VO)) | \ | ||||||
|  | 			NBITVAL(BRCMS_PRIO_TO_HI_PREC(PRIO_8021D_VO)) |	\ | ||||||
|  | 			NBITVAL(BRCMS_PRIO_TO_PREC(PRIO_8021D_NC)) |	\ | ||||||
|  | 			NBITVAL(BRCMS_PRIO_TO_HI_PREC(PRIO_8021D_NC))) | ||||||
|  | 
 | ||||||
|  | /* network protection config */ | ||||||
|  | #define	BRCMS_PROT_G_SPEC		1	/* SPEC g protection */ | ||||||
|  | #define	BRCMS_PROT_G_OVR		2	/* SPEC g prot override */ | ||||||
|  | #define	BRCMS_PROT_G_USER		3	/* gmode specified by user */ | ||||||
|  | #define	BRCMS_PROT_OVERLAP	4	/* overlap */ | ||||||
|  | #define	BRCMS_PROT_N_USER		10	/* nmode specified by user */ | ||||||
|  | #define	BRCMS_PROT_N_CFG		11	/* n protection */ | ||||||
|  | #define	BRCMS_PROT_N_CFG_OVR	12	/* n protection override */ | ||||||
|  | #define	BRCMS_PROT_N_NONGF	13	/* non-GF protection */ | ||||||
|  | #define	BRCMS_PROT_N_NONGF_OVR	14	/* non-GF protection override */ | ||||||
|  | #define	BRCMS_PROT_N_PAM_OVR	15	/* n preamble override */ | ||||||
|  | #define	BRCMS_PROT_N_OBSS		16	/* non-HT OBSS present */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * 54g modes (basic bits may still be overridden) | ||||||
|  |  * | ||||||
|  |  * GMODE_LEGACY_B | ||||||
|  |  *	Rateset: 1b, 2b, 5.5, 11 | ||||||
|  |  *	Preamble: Long | ||||||
|  |  *	Shortslot: Off | ||||||
|  |  * GMODE_AUTO | ||||||
|  |  *	Rateset: 1b, 2b, 5.5b, 11b, 18, 24, 36, 54 | ||||||
|  |  *	Extended Rateset: 6, 9, 12, 48 | ||||||
|  |  *	Preamble: Long | ||||||
|  |  *	Shortslot: Auto | ||||||
|  |  * GMODE_ONLY | ||||||
|  |  *	Rateset: 1b, 2b, 5.5b, 11b, 18, 24b, 36, 54 | ||||||
|  |  *	Extended Rateset: 6b, 9, 12b, 48 | ||||||
|  |  *	Preamble: Short required | ||||||
|  |  *	Shortslot: Auto | ||||||
|  |  * GMODE_B_DEFERRED | ||||||
|  |  *	Rateset: 1b, 2b, 5.5b, 11b, 18, 24, 36, 54 | ||||||
|  |  *	Extended Rateset: 6, 9, 12, 48 | ||||||
|  |  *	Preamble: Long | ||||||
|  |  *	Shortslot: On | ||||||
|  |  * GMODE_PERFORMANCE | ||||||
|  |  *	Rateset: 1b, 2b, 5.5b, 6b, 9, 11b, 12b, 18, 24b, 36, 48, 54 | ||||||
|  |  *	Preamble: Short required | ||||||
|  |  *	Shortslot: On and required | ||||||
|  |  * GMODE_LRS | ||||||
|  |  *	Rateset: 1b, 2b, 5.5b, 11b | ||||||
|  |  *	Extended Rateset: 6, 9, 12, 18, 24, 36, 48, 54 | ||||||
|  |  *	Preamble: Long | ||||||
|  |  *	Shortslot: Auto | ||||||
|  |  */ | ||||||
|  | #define GMODE_LEGACY_B		0 | ||||||
|  | #define GMODE_AUTO		1 | ||||||
|  | #define GMODE_ONLY		2 | ||||||
|  | #define GMODE_B_DEFERRED	3 | ||||||
|  | #define GMODE_PERFORMANCE	4 | ||||||
|  | #define GMODE_LRS		5 | ||||||
|  | #define GMODE_MAX		6 | ||||||
|  | 
 | ||||||
|  | /* MCS values greater than this enable multiple streams */ | ||||||
|  | #define HIGHEST_SINGLE_STREAM_MCS	7 | ||||||
|  | 
 | ||||||
|  | #define	MAXBANDS		2	/* Maximum #of bands */ | ||||||
|  | 
 | ||||||
|  | /* max number of antenna configurations */ | ||||||
|  | #define ANT_SELCFG_MAX		4 | ||||||
|  | 
 | ||||||
|  | struct brcms_antselcfg { | ||||||
|  | 	u8 ant_config[ANT_SELCFG_MAX];	/* antenna configuration */ | ||||||
|  | 	u8 num_antcfg;	/* number of available antenna configurations */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* common functions for every port */ | ||||||
|  | struct brcms_c_info * | ||||||
|  | brcms_c_attach(struct brcms_info *wl, u16 vendor, u16 device, uint unit, | ||||||
|  | 	       bool piomode, void __iomem *regsva, struct pci_dev *btparam, | ||||||
|  | 	       uint *perr); | ||||||
|  | extern uint brcms_c_detach(struct brcms_c_info *wlc); | ||||||
|  | extern int brcms_c_up(struct brcms_c_info *wlc); | ||||||
|  | extern uint brcms_c_down(struct brcms_c_info *wlc); | ||||||
|  | 
 | ||||||
|  | extern bool brcms_c_chipmatch(u16 vendor, u16 device); | ||||||
|  | extern void brcms_c_init(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_reset(struct brcms_c_info *wlc); | ||||||
|  | 
 | ||||||
|  | extern void brcms_c_intrson(struct brcms_c_info *wlc); | ||||||
|  | extern u32 brcms_c_intrsoff(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_intrsrestore(struct brcms_c_info *wlc, u32 macintmask); | ||||||
|  | extern bool brcms_c_intrsupd(struct brcms_c_info *wlc); | ||||||
|  | extern bool brcms_c_isr(struct brcms_c_info *wlc, bool *wantdpc); | ||||||
|  | extern bool brcms_c_dpc(struct brcms_c_info *wlc, bool bounded); | ||||||
|  | extern void brcms_c_sendpkt_mac80211(struct brcms_c_info *wlc, | ||||||
|  | 				     struct sk_buff *sdu, | ||||||
|  | 				     struct ieee80211_hw *hw); | ||||||
|  | extern bool brcms_c_aggregatable(struct brcms_c_info *wlc, u8 tid); | ||||||
|  | 
 | ||||||
|  | /* helper functions */ | ||||||
|  | extern void brcms_c_statsupd(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_protection_upd(struct brcms_c_info *wlc, uint idx, | ||||||
|  | 				   int val); | ||||||
|  | extern int brcms_c_get_header_len(void); | ||||||
|  | extern void brcms_c_mac_bcn_promisc_change(struct brcms_c_info *wlc, | ||||||
|  | 					   bool promisc); | ||||||
|  | extern void brcms_c_set_addrmatch(struct brcms_c_info *wlc, | ||||||
|  | 				  int match_reg_offset, | ||||||
|  | 				  const u8 *addr); | ||||||
|  | extern void brcms_c_wme_setparams(struct brcms_c_info *wlc, u16 aci, | ||||||
|  | 			      const struct ieee80211_tx_queue_params *arg, | ||||||
|  | 			      bool suspend); | ||||||
|  | extern struct brcms_pub *brcms_c_pub(struct brcms_c_info *wlc); | ||||||
|  | 
 | ||||||
|  | /* common functions for every port */ | ||||||
|  | extern void brcms_c_mhf(struct brcms_c_info *wlc, u8 idx, u16 mask, u16 val, | ||||||
|  | 		    int bands); | ||||||
|  | extern void brcms_c_rate_lookup_init(struct brcms_c_info *wlc, | ||||||
|  | 				     struct brcms_c_rateset *rateset); | ||||||
|  | extern void brcms_default_rateset(struct brcms_c_info *wlc, | ||||||
|  | 				  struct brcms_c_rateset *rs); | ||||||
|  | 
 | ||||||
|  | extern void brcms_c_ampdu_flush(struct brcms_c_info *wlc, | ||||||
|  | 			    struct ieee80211_sta *sta, u16 tid); | ||||||
|  | extern void brcms_c_ampdu_tx_operational(struct brcms_c_info *wlc, u8 tid, | ||||||
|  | 					 u8 ba_wsize, uint max_rx_ampdu_bytes); | ||||||
|  | extern char *getvar(struct si_pub *sih, enum brcms_srom_id id); | ||||||
|  | extern int getintvar(struct si_pub *sih, enum brcms_srom_id id); | ||||||
|  | 
 | ||||||
|  | /* wlc_phy.c helper functions */ | ||||||
|  | extern void brcms_c_set_ps_ctrl(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_mctrl(struct brcms_c_info *wlc, u32 mask, u32 val); | ||||||
|  | 
 | ||||||
|  | extern int brcms_c_module_register(struct brcms_pub *pub, | ||||||
|  | 				   const char *name, struct brcms_info *hdl, | ||||||
|  | 				   int (*down_fn)(void *handle)); | ||||||
|  | extern int brcms_c_module_unregister(struct brcms_pub *pub, const char *name, | ||||||
|  | 				     struct brcms_info *hdl); | ||||||
|  | extern void brcms_c_suspend_mac_and_wait(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_enable_mac(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_associate_upd(struct brcms_c_info *wlc, bool state); | ||||||
|  | extern void brcms_c_scan_start(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_scan_stop(struct brcms_c_info *wlc); | ||||||
|  | extern int brcms_c_get_curband(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_wait_for_tx_completion(struct brcms_c_info *wlc, | ||||||
|  | 					   bool drop); | ||||||
|  | 
 | ||||||
|  | int brcms_c_set_channel(struct brcms_c_info *wlc, u16 channel); | ||||||
|  | int brcms_c_set_rate_limit(struct brcms_c_info *wlc, u16 srl, u16 lrl); | ||||||
|  | void brcms_c_get_current_rateset(struct brcms_c_info *wlc, | ||||||
|  | 				 struct brcm_rateset *currs); | ||||||
|  | int brcms_c_set_rateset(struct brcms_c_info *wlc, struct brcm_rateset *rs); | ||||||
|  | int brcms_c_set_beacon_period(struct brcms_c_info *wlc, u16 period); | ||||||
|  | u16 brcms_c_get_phy_type(struct brcms_c_info *wlc, int phyidx); | ||||||
|  | void brcms_c_set_shortslot_override(struct brcms_c_info *wlc, | ||||||
|  | 				    s8 sslot_override); | ||||||
|  | void brcms_c_set_beacon_listen_interval(struct brcms_c_info *wlc, u8 interval); | ||||||
|  | int brcms_c_set_tx_power(struct brcms_c_info *wlc, int txpwr); | ||||||
|  | int brcms_c_get_tx_power(struct brcms_c_info *wlc); | ||||||
|  | void brcms_c_set_radio_mpc(struct brcms_c_info *wlc, bool mpc); | ||||||
|  | 
 | ||||||
|  | /* helper functions */ | ||||||
|  | extern bool brcms_c_check_radio_disabled(struct brcms_c_info *wlc); | ||||||
|  | extern bool brcms_c_radio_monitor_stop(struct brcms_c_info *wlc); | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCM_PUB_H_ */ | ||||||
							
								
								
									
										514
									
								
								drivers/net/wireless/brcm80211/brcmsmac/rate.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										514
									
								
								drivers/net/wireless/brcm80211/brcmsmac/rate.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,514 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <brcmu_wifi.h> | ||||||
|  | #include <brcmu_utils.h> | ||||||
|  | 
 | ||||||
|  | #include "d11.h" | ||||||
|  | #include "pub.h" | ||||||
|  | #include "rate.h" | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Rate info per rate: It tells whether a rate is ofdm or not and its phy_rate | ||||||
|  |  * value | ||||||
|  |  */ | ||||||
|  | const u8 rate_info[BRCM_MAXRATE + 1] = { | ||||||
|  | 	/*  0     1     2     3     4     5     6     7     8     9 */ | ||||||
|  | /*   0 */ 0x00, 0x00, 0x0a, 0x00, 0x14, 0x00, 0x00, 0x00, 0x00, 0x00, | ||||||
|  | /*  10 */ 0x00, 0x37, 0x8b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x8f, 0x00, | ||||||
|  | /*  20 */ 0x00, 0x00, 0x6e, 0x00, 0x8a, 0x00, 0x00, 0x00, 0x00, 0x00, | ||||||
|  | /*  30 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x8e, 0x00, 0x00, 0x00, | ||||||
|  | /*  40 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x89, 0x00, | ||||||
|  | /*  50 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||||||
|  | /*  60 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||||||
|  | /*  70 */ 0x00, 0x00, 0x8d, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||||||
|  | /*  80 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||||||
|  | /*  90 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x88, 0x00, 0x00, 0x00, | ||||||
|  | /* 100 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x8c | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* rates are in units of Kbps */ | ||||||
|  | const struct brcms_mcs_info mcs_table[MCS_TABLE_SIZE] = { | ||||||
|  | 	/* MCS  0: SS 1, MOD: BPSK,  CR 1/2 */ | ||||||
|  | 	{6500, 13500, CEIL(6500 * 10, 9), CEIL(13500 * 10, 9), 0x00, | ||||||
|  | 	 BRCM_RATE_6M}, | ||||||
|  | 	/* MCS  1: SS 1, MOD: QPSK,  CR 1/2 */ | ||||||
|  | 	{13000, 27000, CEIL(13000 * 10, 9), CEIL(27000 * 10, 9), 0x08, | ||||||
|  | 	 BRCM_RATE_12M}, | ||||||
|  | 	/* MCS  2: SS 1, MOD: QPSK,  CR 3/4 */ | ||||||
|  | 	{19500, 40500, CEIL(19500 * 10, 9), CEIL(40500 * 10, 9), 0x0A, | ||||||
|  | 	 BRCM_RATE_18M}, | ||||||
|  | 	/* MCS  3: SS 1, MOD: 16QAM, CR 1/2 */ | ||||||
|  | 	{26000, 54000, CEIL(26000 * 10, 9), CEIL(54000 * 10, 9), 0x10, | ||||||
|  | 	 BRCM_RATE_24M}, | ||||||
|  | 	/* MCS  4: SS 1, MOD: 16QAM, CR 3/4 */ | ||||||
|  | 	{39000, 81000, CEIL(39000 * 10, 9), CEIL(81000 * 10, 9), 0x12, | ||||||
|  | 	 BRCM_RATE_36M}, | ||||||
|  | 	/* MCS  5: SS 1, MOD: 64QAM, CR 2/3 */ | ||||||
|  | 	{52000, 108000, CEIL(52000 * 10, 9), CEIL(108000 * 10, 9), 0x19, | ||||||
|  | 	 BRCM_RATE_48M}, | ||||||
|  | 	/* MCS  6: SS 1, MOD: 64QAM, CR 3/4 */ | ||||||
|  | 	{58500, 121500, CEIL(58500 * 10, 9), CEIL(121500 * 10, 9), 0x1A, | ||||||
|  | 	 BRCM_RATE_54M}, | ||||||
|  | 	/* MCS  7: SS 1, MOD: 64QAM, CR 5/6 */ | ||||||
|  | 	{65000, 135000, CEIL(65000 * 10, 9), CEIL(135000 * 10, 9), 0x1C, | ||||||
|  | 	 BRCM_RATE_54M}, | ||||||
|  | 	/* MCS  8: SS 2, MOD: BPSK,  CR 1/2 */ | ||||||
|  | 	{13000, 27000, CEIL(13000 * 10, 9), CEIL(27000 * 10, 9), 0x40, | ||||||
|  | 	 BRCM_RATE_6M}, | ||||||
|  | 	/* MCS  9: SS 2, MOD: QPSK,  CR 1/2 */ | ||||||
|  | 	{26000, 54000, CEIL(26000 * 10, 9), CEIL(54000 * 10, 9), 0x48, | ||||||
|  | 	 BRCM_RATE_12M}, | ||||||
|  | 	/* MCS 10: SS 2, MOD: QPSK,  CR 3/4 */ | ||||||
|  | 	{39000, 81000, CEIL(39000 * 10, 9), CEIL(81000 * 10, 9), 0x4A, | ||||||
|  | 	 BRCM_RATE_18M}, | ||||||
|  | 	/* MCS 11: SS 2, MOD: 16QAM, CR 1/2 */ | ||||||
|  | 	{52000, 108000, CEIL(52000 * 10, 9), CEIL(108000 * 10, 9), 0x50, | ||||||
|  | 	 BRCM_RATE_24M}, | ||||||
|  | 	/* MCS 12: SS 2, MOD: 16QAM, CR 3/4 */ | ||||||
|  | 	{78000, 162000, CEIL(78000 * 10, 9), CEIL(162000 * 10, 9), 0x52, | ||||||
|  | 	 BRCM_RATE_36M}, | ||||||
|  | 	/* MCS 13: SS 2, MOD: 64QAM, CR 2/3 */ | ||||||
|  | 	{104000, 216000, CEIL(104000 * 10, 9), CEIL(216000 * 10, 9), 0x59, | ||||||
|  | 	 BRCM_RATE_48M}, | ||||||
|  | 	/* MCS 14: SS 2, MOD: 64QAM, CR 3/4 */ | ||||||
|  | 	{117000, 243000, CEIL(117000 * 10, 9), CEIL(243000 * 10, 9), 0x5A, | ||||||
|  | 	 BRCM_RATE_54M}, | ||||||
|  | 	/* MCS 15: SS 2, MOD: 64QAM, CR 5/6 */ | ||||||
|  | 	{130000, 270000, CEIL(130000 * 10, 9), CEIL(270000 * 10, 9), 0x5C, | ||||||
|  | 	 BRCM_RATE_54M}, | ||||||
|  | 	/* MCS 16: SS 3, MOD: BPSK,  CR 1/2 */ | ||||||
|  | 	{19500, 40500, CEIL(19500 * 10, 9), CEIL(40500 * 10, 9), 0x80, | ||||||
|  | 	 BRCM_RATE_6M}, | ||||||
|  | 	/* MCS 17: SS 3, MOD: QPSK,  CR 1/2 */ | ||||||
|  | 	{39000, 81000, CEIL(39000 * 10, 9), CEIL(81000 * 10, 9), 0x88, | ||||||
|  | 	 BRCM_RATE_12M}, | ||||||
|  | 	/* MCS 18: SS 3, MOD: QPSK,  CR 3/4 */ | ||||||
|  | 	{58500, 121500, CEIL(58500 * 10, 9), CEIL(121500 * 10, 9), 0x8A, | ||||||
|  | 	 BRCM_RATE_18M}, | ||||||
|  | 	/* MCS 19: SS 3, MOD: 16QAM, CR 1/2 */ | ||||||
|  | 	{78000, 162000, CEIL(78000 * 10, 9), CEIL(162000 * 10, 9), 0x90, | ||||||
|  | 	 BRCM_RATE_24M}, | ||||||
|  | 	/* MCS 20: SS 3, MOD: 16QAM, CR 3/4 */ | ||||||
|  | 	{117000, 243000, CEIL(117000 * 10, 9), CEIL(243000 * 10, 9), 0x92, | ||||||
|  | 	 BRCM_RATE_36M}, | ||||||
|  | 	/* MCS 21: SS 3, MOD: 64QAM, CR 2/3 */ | ||||||
|  | 	{156000, 324000, CEIL(156000 * 10, 9), CEIL(324000 * 10, 9), 0x99, | ||||||
|  | 	 BRCM_RATE_48M}, | ||||||
|  | 	/* MCS 22: SS 3, MOD: 64QAM, CR 3/4 */ | ||||||
|  | 	{175500, 364500, CEIL(175500 * 10, 9), CEIL(364500 * 10, 9), 0x9A, | ||||||
|  | 	 BRCM_RATE_54M}, | ||||||
|  | 	/* MCS 23: SS 3, MOD: 64QAM, CR 5/6 */ | ||||||
|  | 	{195000, 405000, CEIL(195000 * 10, 9), CEIL(405000 * 10, 9), 0x9B, | ||||||
|  | 	 BRCM_RATE_54M}, | ||||||
|  | 	/* MCS 24: SS 4, MOD: BPSK,  CR 1/2 */ | ||||||
|  | 	{26000, 54000, CEIL(26000 * 10, 9), CEIL(54000 * 10, 9), 0xC0, | ||||||
|  | 	 BRCM_RATE_6M}, | ||||||
|  | 	/* MCS 25: SS 4, MOD: QPSK,  CR 1/2 */ | ||||||
|  | 	{52000, 108000, CEIL(52000 * 10, 9), CEIL(108000 * 10, 9), 0xC8, | ||||||
|  | 	 BRCM_RATE_12M}, | ||||||
|  | 	/* MCS 26: SS 4, MOD: QPSK,  CR 3/4 */ | ||||||
|  | 	{78000, 162000, CEIL(78000 * 10, 9), CEIL(162000 * 10, 9), 0xCA, | ||||||
|  | 	 BRCM_RATE_18M}, | ||||||
|  | 	/* MCS 27: SS 4, MOD: 16QAM, CR 1/2 */ | ||||||
|  | 	{104000, 216000, CEIL(104000 * 10, 9), CEIL(216000 * 10, 9), 0xD0, | ||||||
|  | 	 BRCM_RATE_24M}, | ||||||
|  | 	/* MCS 28: SS 4, MOD: 16QAM, CR 3/4 */ | ||||||
|  | 	{156000, 324000, CEIL(156000 * 10, 9), CEIL(324000 * 10, 9), 0xD2, | ||||||
|  | 	 BRCM_RATE_36M}, | ||||||
|  | 	/* MCS 29: SS 4, MOD: 64QAM, CR 2/3 */ | ||||||
|  | 	{208000, 432000, CEIL(208000 * 10, 9), CEIL(432000 * 10, 9), 0xD9, | ||||||
|  | 	 BRCM_RATE_48M}, | ||||||
|  | 	/* MCS 30: SS 4, MOD: 64QAM, CR 3/4 */ | ||||||
|  | 	{234000, 486000, CEIL(234000 * 10, 9), CEIL(486000 * 10, 9), 0xDA, | ||||||
|  | 	 BRCM_RATE_54M}, | ||||||
|  | 	/* MCS 31: SS 4, MOD: 64QAM, CR 5/6 */ | ||||||
|  | 	{260000, 540000, CEIL(260000 * 10, 9), CEIL(540000 * 10, 9), 0xDB, | ||||||
|  | 	 BRCM_RATE_54M}, | ||||||
|  | 	/* MCS 32: SS 1, MOD: BPSK,  CR 1/2 */ | ||||||
|  | 	{0, 6000, 0, CEIL(6000 * 10, 9), 0x00, BRCM_RATE_6M}, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * phycfg for legacy OFDM frames: code rate, modulation scheme, spatial streams | ||||||
|  |  * Number of spatial streams: always 1 other fields: refer to table 78 of | ||||||
|  |  * section 17.3.2.2 of the original .11a standard | ||||||
|  |  */ | ||||||
|  | struct legacy_phycfg { | ||||||
|  | 	u32 rate_ofdm;	/* ofdm mac rate */ | ||||||
|  | 	/* phy ctl byte 3, code rate, modulation type, # of streams */ | ||||||
|  | 	u8 tx_phy_ctl3; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* Number of legacy_rate_cfg entries in the table */ | ||||||
|  | #define LEGACY_PHYCFG_TABLE_SIZE	12 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * In CCK mode LPPHY overloads OFDM Modulation bits with CCK Data Rate | ||||||
|  |  * Eventually MIMOPHY would also be converted to this format | ||||||
|  |  * 0 = 1Mbps; 1 = 2Mbps; 2 = 5.5Mbps; 3 = 11Mbps | ||||||
|  |  */ | ||||||
|  | static const struct | ||||||
|  | legacy_phycfg legacy_phycfg_table[LEGACY_PHYCFG_TABLE_SIZE] = { | ||||||
|  | 	{BRCM_RATE_1M, 0x00},	/* CCK  1Mbps,  data rate  0 */ | ||||||
|  | 	{BRCM_RATE_2M, 0x08},	/* CCK  2Mbps,  data rate  1 */ | ||||||
|  | 	{BRCM_RATE_5M5, 0x10},	/* CCK  5.5Mbps,  data rate  2 */ | ||||||
|  | 	{BRCM_RATE_11M, 0x18},	/* CCK  11Mbps,  data rate   3 */ | ||||||
|  | 	/* OFDM  6Mbps,  code rate 1/2, BPSK,   1 spatial stream */ | ||||||
|  | 	{BRCM_RATE_6M, 0x00}, | ||||||
|  | 	/* OFDM  9Mbps,  code rate 3/4, BPSK,   1 spatial stream */ | ||||||
|  | 	{BRCM_RATE_9M, 0x02}, | ||||||
|  | 	/* OFDM  12Mbps, code rate 1/2, QPSK,   1 spatial stream */ | ||||||
|  | 	{BRCM_RATE_12M, 0x08}, | ||||||
|  | 	/* OFDM  18Mbps, code rate 3/4, QPSK,   1 spatial stream */ | ||||||
|  | 	{BRCM_RATE_18M, 0x0A}, | ||||||
|  | 	/* OFDM  24Mbps, code rate 1/2, 16-QAM, 1 spatial stream */ | ||||||
|  | 	{BRCM_RATE_24M, 0x10}, | ||||||
|  | 	/* OFDM  36Mbps, code rate 3/4, 16-QAM, 1 spatial stream */ | ||||||
|  | 	{BRCM_RATE_36M, 0x12}, | ||||||
|  | 	/* OFDM  48Mbps, code rate 2/3, 64-QAM, 1 spatial stream */ | ||||||
|  | 	{BRCM_RATE_48M, 0x19}, | ||||||
|  | 	/* OFDM  54Mbps, code rate 3/4, 64-QAM, 1 spatial stream */ | ||||||
|  | 	{BRCM_RATE_54M, 0x1A}, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* Hardware rates (also encodes default basic rates) */ | ||||||
|  | 
 | ||||||
|  | const struct brcms_c_rateset cck_ofdm_mimo_rates = { | ||||||
|  | 	12, | ||||||
|  | 	/*  1b,   2b,   5.5b, 6,    9,    11b,  12,   18,   24,   36,   48, */ | ||||||
|  | 	{ 0x82, 0x84, 0x8b, 0x0c, 0x12, 0x96, 0x18, 0x24, 0x30, 0x48, 0x60, | ||||||
|  | 	/* 54 Mbps */ | ||||||
|  | 	  0x6c}, | ||||||
|  | 	0x00, | ||||||
|  | 	{ 0xff, 0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||||||
|  | 	  0x00, 0x00, 0x00, 0x00, 0x00} | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const struct brcms_c_rateset ofdm_mimo_rates = { | ||||||
|  | 	8, | ||||||
|  | 	/*  6b,   9,    12b,  18,   24b,  36,   48,   54 Mbps */ | ||||||
|  | 	{ 0x8c, 0x12, 0x98, 0x24, 0xb0, 0x48, 0x60, 0x6c}, | ||||||
|  | 	0x00, | ||||||
|  | 	{ 0xff, 0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||||||
|  | 	  0x00, 0x00, 0x00, 0x00, 0x00} | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* Default ratesets that include MCS32 for 40BW channels */ | ||||||
|  | static const struct brcms_c_rateset cck_ofdm_40bw_mimo_rates = { | ||||||
|  | 	12, | ||||||
|  | 	/*  1b,   2b,   5.5b, 6,    9,    11b,  12,   18,   24,   36,   48 */ | ||||||
|  | 	{ 0x82, 0x84, 0x8b, 0x0c, 0x12, 0x96, 0x18, 0x24, 0x30, 0x48, 0x60, | ||||||
|  | 	/* 54 Mbps */ | ||||||
|  | 	  0x6c}, | ||||||
|  | 	0x00, | ||||||
|  | 	{ 0xff, 0xff, 0xff, 0xff, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||||||
|  | 	  0x00, 0x00, 0x00, 0x00, 0x00} | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static const struct brcms_c_rateset ofdm_40bw_mimo_rates = { | ||||||
|  | 	8, | ||||||
|  | 	/*  6b,   9,    12b,  18,   24b,  36,   48,   54 Mbps */ | ||||||
|  | 	{ 0x8c, 0x12, 0x98, 0x24, 0xb0, 0x48, 0x60, 0x6c}, | ||||||
|  | 	0x00, | ||||||
|  | 	{ 0xff, 0xff, 0xff, 0xff, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||||||
|  | 	  0x00, 0x00, 0x00, 0x00, 0x00} | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const struct brcms_c_rateset cck_ofdm_rates = { | ||||||
|  | 	12, | ||||||
|  | 	/*  1b,   2b, 5.5b, 6,    9,    11b,  12,   18,   24,   36,   48,*/ | ||||||
|  | 	{ 0x82, 0x84, 0x8b, 0x0c, 0x12, 0x96, 0x18, 0x24, 0x30, 0x48, 0x60, | ||||||
|  | 	/*54 Mbps */ | ||||||
|  | 	  0x6c}, | ||||||
|  | 	0x00, | ||||||
|  | 	{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||||||
|  | 	  0x00, 0x00, 0x00, 0x00, 0x00} | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const struct brcms_c_rateset gphy_legacy_rates = { | ||||||
|  | 	4, | ||||||
|  | 	/*  1b,   2b,   5.5b, 11b Mbps */ | ||||||
|  | 	{ 0x82, 0x84, 0x8b, 0x96}, | ||||||
|  | 	0x00, | ||||||
|  | 	{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||||||
|  | 	  0x00, 0x00, 0x00, 0x00, 0x00} | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const struct brcms_c_rateset ofdm_rates = { | ||||||
|  | 	8, | ||||||
|  | 	/*  6b,   9,    12b,  18,   24b,  36,   48,   54 Mbps */ | ||||||
|  | 	{ 0x8c, 0x12, 0x98, 0x24, 0xb0, 0x48, 0x60, 0x6c}, | ||||||
|  | 	0x00, | ||||||
|  | 	{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||||||
|  | 	  0x00, 0x00, 0x00, 0x00, 0x00} | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const struct brcms_c_rateset cck_rates = { | ||||||
|  | 	4, | ||||||
|  | 	/*  1b,   2b,   5.5,  11 Mbps */ | ||||||
|  | 	{ 0x82, 0x84, 0x0b, 0x16}, | ||||||
|  | 	0x00, | ||||||
|  | 	{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||||||
|  | 	  0x00, 0x00, 0x00, 0x00, 0x00} | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* check if rateset is valid.
 | ||||||
|  |  * if check_brate is true, rateset without a basic rate is considered NOT valid. | ||||||
|  |  */ | ||||||
|  | static bool brcms_c_rateset_valid(struct brcms_c_rateset *rs, bool check_brate) | ||||||
|  | { | ||||||
|  | 	uint idx; | ||||||
|  | 
 | ||||||
|  | 	if (!rs->count) | ||||||
|  | 		return false; | ||||||
|  | 
 | ||||||
|  | 	if (!check_brate) | ||||||
|  | 		return true; | ||||||
|  | 
 | ||||||
|  | 	/* error if no basic rates */ | ||||||
|  | 	for (idx = 0; idx < rs->count; idx++) { | ||||||
|  | 		if (rs->rates[idx] & BRCMS_RATE_FLAG) | ||||||
|  | 			return true; | ||||||
|  | 	} | ||||||
|  | 	return false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void brcms_c_rateset_mcs_upd(struct brcms_c_rateset *rs, u8 txstreams) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	for (i = txstreams; i < MAX_STREAMS_SUPPORTED; i++) | ||||||
|  | 		rs->mcs[i] = 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * filter based on hardware rateset, and sort filtered rateset with basic | ||||||
|  |  * bit(s) preserved, and check if resulting rateset is valid. | ||||||
|  | */ | ||||||
|  | bool | ||||||
|  | brcms_c_rate_hwrs_filter_sort_validate(struct brcms_c_rateset *rs, | ||||||
|  | 				   const struct brcms_c_rateset *hw_rs, | ||||||
|  | 				   bool check_brate, u8 txstreams) | ||||||
|  | { | ||||||
|  | 	u8 rateset[BRCM_MAXRATE + 1]; | ||||||
|  | 	u8 r; | ||||||
|  | 	uint count; | ||||||
|  | 	uint i; | ||||||
|  | 
 | ||||||
|  | 	memset(rateset, 0, sizeof(rateset)); | ||||||
|  | 	count = rs->count; | ||||||
|  | 
 | ||||||
|  | 	for (i = 0; i < count; i++) { | ||||||
|  | 		/* mask off "basic rate" bit, BRCMS_RATE_FLAG */ | ||||||
|  | 		r = (int)rs->rates[i] & BRCMS_RATE_MASK; | ||||||
|  | 		if ((r > BRCM_MAXRATE) || (rate_info[r] == 0)) | ||||||
|  | 			continue; | ||||||
|  | 		rateset[r] = rs->rates[i];	/* preserve basic bit! */ | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* fill out the rates in order, looking at only supported rates */ | ||||||
|  | 	count = 0; | ||||||
|  | 	for (i = 0; i < hw_rs->count; i++) { | ||||||
|  | 		r = hw_rs->rates[i] & BRCMS_RATE_MASK; | ||||||
|  | 		if (rateset[r]) | ||||||
|  | 			rs->rates[count++] = rateset[r]; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rs->count = count; | ||||||
|  | 
 | ||||||
|  | 	/* only set the mcs rate bit if the equivalent hw mcs bit is set */ | ||||||
|  | 	for (i = 0; i < MCSSET_LEN; i++) | ||||||
|  | 		rs->mcs[i] = (rs->mcs[i] & hw_rs->mcs[i]); | ||||||
|  | 
 | ||||||
|  | 	if (brcms_c_rateset_valid(rs, check_brate)) | ||||||
|  | 		return true; | ||||||
|  | 	else | ||||||
|  | 		return false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* calculate the rate of a rx'd frame and return it as a ratespec */ | ||||||
|  | u32 brcms_c_compute_rspec(struct d11rxhdr *rxh, u8 *plcp) | ||||||
|  | { | ||||||
|  | 	int phy_type; | ||||||
|  | 	u32 rspec = PHY_TXC1_BW_20MHZ << RSPEC_BW_SHIFT; | ||||||
|  | 
 | ||||||
|  | 	phy_type = | ||||||
|  | 	    ((rxh->RxChan & RXS_CHAN_PHYTYPE_MASK) >> RXS_CHAN_PHYTYPE_SHIFT); | ||||||
|  | 
 | ||||||
|  | 	if ((phy_type == PHY_TYPE_N) || (phy_type == PHY_TYPE_SSN) || | ||||||
|  | 	    (phy_type == PHY_TYPE_LCN) || (phy_type == PHY_TYPE_HT)) { | ||||||
|  | 		switch (rxh->PhyRxStatus_0 & PRXS0_FT_MASK) { | ||||||
|  | 		case PRXS0_CCK: | ||||||
|  | 			rspec = | ||||||
|  | 				cck_phy2mac_rate( | ||||||
|  | 				((struct cck_phy_hdr *) plcp)->signal); | ||||||
|  | 			break; | ||||||
|  | 		case PRXS0_OFDM: | ||||||
|  | 			rspec = | ||||||
|  | 			    ofdm_phy2mac_rate( | ||||||
|  | 				((struct ofdm_phy_hdr *) plcp)->rlpt[0]); | ||||||
|  | 			break; | ||||||
|  | 		case PRXS0_PREN: | ||||||
|  | 			rspec = (plcp[0] & MIMO_PLCP_MCS_MASK) | RSPEC_MIMORATE; | ||||||
|  | 			if (plcp[0] & MIMO_PLCP_40MHZ) { | ||||||
|  | 				/* indicate rspec is for 40 MHz mode */ | ||||||
|  | 				rspec &= ~RSPEC_BW_MASK; | ||||||
|  | 				rspec |= (PHY_TXC1_BW_40MHZ << RSPEC_BW_SHIFT); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		case PRXS0_STDN: | ||||||
|  | 			/* fallthru */ | ||||||
|  | 		default: | ||||||
|  | 			/* not supported, error condition */ | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 		if (plcp3_issgi(plcp[3])) | ||||||
|  | 			rspec |= RSPEC_SHORT_GI; | ||||||
|  | 	} else | ||||||
|  | 	    if ((phy_type == PHY_TYPE_A) || (rxh->PhyRxStatus_0 & PRXS0_OFDM)) | ||||||
|  | 		rspec = ofdm_phy2mac_rate( | ||||||
|  | 				((struct ofdm_phy_hdr *) plcp)->rlpt[0]); | ||||||
|  | 	else | ||||||
|  | 		rspec = cck_phy2mac_rate( | ||||||
|  | 				((struct cck_phy_hdr *) plcp)->signal); | ||||||
|  | 
 | ||||||
|  | 	return rspec; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* copy rateset src to dst as-is (no masking or sorting) */ | ||||||
|  | void brcms_c_rateset_copy(const struct brcms_c_rateset *src, | ||||||
|  | 			  struct brcms_c_rateset *dst) | ||||||
|  | { | ||||||
|  | 	memcpy(dst, src, sizeof(struct brcms_c_rateset)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Copy and selectively filter one rateset to another. | ||||||
|  |  * 'basic_only' means only copy basic rates. | ||||||
|  |  * 'rates' indicates cck (11b) and ofdm rates combinations. | ||||||
|  |  *    - 0: cck and ofdm | ||||||
|  |  *    - 1: cck only | ||||||
|  |  *    - 2: ofdm only | ||||||
|  |  * 'xmask' is the copy mask (typically 0x7f or 0xff). | ||||||
|  |  */ | ||||||
|  | void | ||||||
|  | brcms_c_rateset_filter(struct brcms_c_rateset *src, struct brcms_c_rateset *dst, | ||||||
|  | 		       bool basic_only, u8 rates, uint xmask, bool mcsallow) | ||||||
|  | { | ||||||
|  | 	uint i; | ||||||
|  | 	uint r; | ||||||
|  | 	uint count; | ||||||
|  | 
 | ||||||
|  | 	count = 0; | ||||||
|  | 	for (i = 0; i < src->count; i++) { | ||||||
|  | 		r = src->rates[i]; | ||||||
|  | 		if (basic_only && !(r & BRCMS_RATE_FLAG)) | ||||||
|  | 			continue; | ||||||
|  | 		if (rates == BRCMS_RATES_CCK && | ||||||
|  | 		    is_ofdm_rate((r & BRCMS_RATE_MASK))) | ||||||
|  | 			continue; | ||||||
|  | 		if (rates == BRCMS_RATES_OFDM && | ||||||
|  | 		    is_cck_rate((r & BRCMS_RATE_MASK))) | ||||||
|  | 			continue; | ||||||
|  | 		dst->rates[count++] = r & xmask; | ||||||
|  | 	} | ||||||
|  | 	dst->count = count; | ||||||
|  | 	dst->htphy_membership = src->htphy_membership; | ||||||
|  | 
 | ||||||
|  | 	if (mcsallow && rates != BRCMS_RATES_CCK) | ||||||
|  | 		memcpy(&dst->mcs[0], &src->mcs[0], MCSSET_LEN); | ||||||
|  | 	else | ||||||
|  | 		brcms_c_rateset_mcs_clear(dst); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* select rateset for a given phy_type and bandtype and filter it, sort it
 | ||||||
|  |  * and fill rs_tgt with result | ||||||
|  |  */ | ||||||
|  | void | ||||||
|  | brcms_c_rateset_default(struct brcms_c_rateset *rs_tgt, | ||||||
|  | 			const struct brcms_c_rateset *rs_hw, | ||||||
|  | 			uint phy_type, int bandtype, bool cck_only, | ||||||
|  | 			uint rate_mask, bool mcsallow, u8 bw, u8 txstreams) | ||||||
|  | { | ||||||
|  | 	const struct brcms_c_rateset *rs_dflt; | ||||||
|  | 	struct brcms_c_rateset rs_sel; | ||||||
|  | 	if ((PHYTYPE_IS(phy_type, PHY_TYPE_HT)) || | ||||||
|  | 	    (PHYTYPE_IS(phy_type, PHY_TYPE_N)) || | ||||||
|  | 	    (PHYTYPE_IS(phy_type, PHY_TYPE_LCN)) || | ||||||
|  | 	    (PHYTYPE_IS(phy_type, PHY_TYPE_SSN))) { | ||||||
|  | 		if (bandtype == BRCM_BAND_5G) | ||||||
|  | 			rs_dflt = (bw == BRCMS_20_MHZ ? | ||||||
|  | 				   &ofdm_mimo_rates : &ofdm_40bw_mimo_rates); | ||||||
|  | 		else | ||||||
|  | 			rs_dflt = (bw == BRCMS_20_MHZ ? | ||||||
|  | 				   &cck_ofdm_mimo_rates : | ||||||
|  | 				   &cck_ofdm_40bw_mimo_rates); | ||||||
|  | 	} else if (PHYTYPE_IS(phy_type, PHY_TYPE_LP)) { | ||||||
|  | 		rs_dflt = (bandtype == BRCM_BAND_5G) ? | ||||||
|  | 			  &ofdm_rates : &cck_ofdm_rates; | ||||||
|  | 	} else if (PHYTYPE_IS(phy_type, PHY_TYPE_A)) { | ||||||
|  | 		rs_dflt = &ofdm_rates; | ||||||
|  | 	} else if (PHYTYPE_IS(phy_type, PHY_TYPE_G)) { | ||||||
|  | 		rs_dflt = &cck_ofdm_rates; | ||||||
|  | 	} else { | ||||||
|  | 		/* should not happen, error condition */ | ||||||
|  | 		rs_dflt = &cck_rates;	/* force cck */ | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* if hw rateset is not supplied, assign selected rateset to it */ | ||||||
|  | 	if (!rs_hw) | ||||||
|  | 		rs_hw = rs_dflt; | ||||||
|  | 
 | ||||||
|  | 	brcms_c_rateset_copy(rs_dflt, &rs_sel); | ||||||
|  | 	brcms_c_rateset_mcs_upd(&rs_sel, txstreams); | ||||||
|  | 	brcms_c_rateset_filter(&rs_sel, rs_tgt, false, | ||||||
|  | 			   cck_only ? BRCMS_RATES_CCK : BRCMS_RATES_CCK_OFDM, | ||||||
|  | 			   rate_mask, mcsallow); | ||||||
|  | 	brcms_c_rate_hwrs_filter_sort_validate(rs_tgt, rs_hw, false, | ||||||
|  | 					   mcsallow ? txstreams : 1); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | s16 brcms_c_rate_legacy_phyctl(uint rate) | ||||||
|  | { | ||||||
|  | 	uint i; | ||||||
|  | 	for (i = 0; i < LEGACY_PHYCFG_TABLE_SIZE; i++) | ||||||
|  | 		if (rate == legacy_phycfg_table[i].rate_ofdm) | ||||||
|  | 			return legacy_phycfg_table[i].tx_phy_ctl3; | ||||||
|  | 
 | ||||||
|  | 	return -1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void brcms_c_rateset_mcs_clear(struct brcms_c_rateset *rateset) | ||||||
|  | { | ||||||
|  | 	uint i; | ||||||
|  | 	for (i = 0; i < MCSSET_LEN; i++) | ||||||
|  | 		rateset->mcs[i] = 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void brcms_c_rateset_mcs_build(struct brcms_c_rateset *rateset, u8 txstreams) | ||||||
|  | { | ||||||
|  | 	memcpy(&rateset->mcs[0], &cck_ofdm_mimo_rates.mcs[0], MCSSET_LEN); | ||||||
|  | 	brcms_c_rateset_mcs_upd(rateset, txstreams); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Based on bandwidth passed, allow/disallow MCS 32 in the rateset */ | ||||||
|  | void brcms_c_rateset_bw_mcs_filter(struct brcms_c_rateset *rateset, u8 bw) | ||||||
|  | { | ||||||
|  | 	if (bw == BRCMS_40_MHZ) | ||||||
|  | 		setbit(rateset->mcs, 32); | ||||||
|  | 	else | ||||||
|  | 		clrbit(rateset->mcs, 32); | ||||||
|  | } | ||||||
							
								
								
									
										250
									
								
								drivers/net/wireless/brcm80211/brcmsmac/rate.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										250
									
								
								drivers/net/wireless/brcm80211/brcmsmac/rate.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,250 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCM_RATE_H_ | ||||||
|  | #define _BRCM_RATE_H_ | ||||||
|  | 
 | ||||||
|  | #include "types.h" | ||||||
|  | #include "d11.h" | ||||||
|  | 
 | ||||||
|  | extern const u8 rate_info[]; | ||||||
|  | extern const struct brcms_c_rateset cck_ofdm_mimo_rates; | ||||||
|  | extern const struct brcms_c_rateset ofdm_mimo_rates; | ||||||
|  | extern const struct brcms_c_rateset cck_ofdm_rates; | ||||||
|  | extern const struct brcms_c_rateset ofdm_rates; | ||||||
|  | extern const struct brcms_c_rateset cck_rates; | ||||||
|  | extern const struct brcms_c_rateset gphy_legacy_rates; | ||||||
|  | extern const struct brcms_c_rateset rate_limit_1_2; | ||||||
|  | 
 | ||||||
|  | struct brcms_mcs_info { | ||||||
|  | 	/* phy rate in kbps [20Mhz] */ | ||||||
|  | 	u32 phy_rate_20; | ||||||
|  | 	/* phy rate in kbps [40Mhz] */ | ||||||
|  | 	u32 phy_rate_40; | ||||||
|  | 	/* phy rate in kbps [20Mhz] with SGI */ | ||||||
|  | 	u32 phy_rate_20_sgi; | ||||||
|  | 	/* phy rate in kbps [40Mhz] with SGI */ | ||||||
|  | 	u32 phy_rate_40_sgi; | ||||||
|  | 	/* phy ctl byte 3, code rate, modulation type, # of streams */ | ||||||
|  | 	u8 tx_phy_ctl3; | ||||||
|  | 	/* matching legacy ofdm rate in 500bkps */ | ||||||
|  | 	u8 leg_ofdm; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #define BRCMS_MAXMCS	32	/* max valid mcs index */ | ||||||
|  | #define MCS_TABLE_SIZE	33	/* Number of mcs entries in the table */ | ||||||
|  | extern const struct brcms_mcs_info mcs_table[]; | ||||||
|  | 
 | ||||||
|  | #define MCS_TXS_MASK	0xc0	/* num tx streams - 1 bit mask */ | ||||||
|  | #define MCS_TXS_SHIFT	6	/* num tx streams - 1 bit shift */ | ||||||
|  | 
 | ||||||
|  | /* returns num tx streams - 1 */ | ||||||
|  | static inline u8 mcs_2_txstreams(u8 mcs) | ||||||
|  | { | ||||||
|  | 	return (mcs_table[mcs].tx_phy_ctl3 & MCS_TXS_MASK) >> MCS_TXS_SHIFT; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline uint mcs_2_rate(u8 mcs, bool is40, bool sgi) | ||||||
|  | { | ||||||
|  | 	if (sgi) { | ||||||
|  | 		if (is40) | ||||||
|  | 			return mcs_table[mcs].phy_rate_40_sgi; | ||||||
|  | 		return mcs_table[mcs].phy_rate_20_sgi; | ||||||
|  | 	} | ||||||
|  | 	if (is40) | ||||||
|  | 		return mcs_table[mcs].phy_rate_40; | ||||||
|  | 
 | ||||||
|  | 	return mcs_table[mcs].phy_rate_20; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Macro to use the rate_info table */ | ||||||
|  | #define	BRCMS_RATE_MASK_FULL 0xff /* Rate value mask with basic rate flag */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * rate spec : holds rate and mode specific information required to generate a | ||||||
|  |  * tx frame. Legacy CCK and OFDM information is held in the same manner as was | ||||||
|  |  * done in the past (in the lower byte) the upper 3 bytes primarily hold MIMO | ||||||
|  |  * specific information | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* rate spec bit fields */ | ||||||
|  | 
 | ||||||
|  | /* Either 500Kbps units or MIMO MCS idx */ | ||||||
|  | #define RSPEC_RATE_MASK		0x0000007F | ||||||
|  | /* mimo MCS is stored in RSPEC_RATE_MASK */ | ||||||
|  | #define RSPEC_MIMORATE		0x08000000 | ||||||
|  | /* mimo bw mask */ | ||||||
|  | #define RSPEC_BW_MASK		0x00000700 | ||||||
|  | /* mimo bw shift */ | ||||||
|  | #define RSPEC_BW_SHIFT		8 | ||||||
|  | /* mimo Space/Time/Frequency mode mask */ | ||||||
|  | #define RSPEC_STF_MASK		0x00003800 | ||||||
|  | /* mimo Space/Time/Frequency mode shift */ | ||||||
|  | #define RSPEC_STF_SHIFT		11 | ||||||
|  | /* mimo coding type mask */ | ||||||
|  | #define RSPEC_CT_MASK		0x0000C000 | ||||||
|  | /* mimo coding type shift */ | ||||||
|  | #define RSPEC_CT_SHIFT		14 | ||||||
|  | /* mimo num STC streams per PLCP defn. */ | ||||||
|  | #define RSPEC_STC_MASK		0x00300000 | ||||||
|  | /* mimo num STC streams per PLCP defn. */ | ||||||
|  | #define RSPEC_STC_SHIFT		20 | ||||||
|  | /* mimo bit indicates adv coding in use */ | ||||||
|  | #define RSPEC_LDPC_CODING	0x00400000 | ||||||
|  | /* mimo bit indicates short GI in use */ | ||||||
|  | #define RSPEC_SHORT_GI		0x00800000 | ||||||
|  | /* bit indicates override both rate & mode */ | ||||||
|  | #define RSPEC_OVERRIDE		0x80000000 | ||||||
|  | /* bit indicates override rate only */ | ||||||
|  | #define RSPEC_OVERRIDE_MCS_ONLY 0x40000000 | ||||||
|  | 
 | ||||||
|  | static inline bool rspec_active(u32 rspec) | ||||||
|  | { | ||||||
|  | 	return rspec & (RSPEC_RATE_MASK | RSPEC_MIMORATE); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline u8 rspec_phytxbyte2(u32 rspec) | ||||||
|  | { | ||||||
|  | 	return (rspec & 0xff00) >> 8; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline u32 rspec_get_bw(u32 rspec) | ||||||
|  | { | ||||||
|  | 	return (rspec & RSPEC_BW_MASK) >> RSPEC_BW_SHIFT; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline bool rspec_issgi(u32 rspec) | ||||||
|  | { | ||||||
|  | 	return (rspec & RSPEC_SHORT_GI) == RSPEC_SHORT_GI; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline bool rspec_is40mhz(u32 rspec) | ||||||
|  | { | ||||||
|  | 	u32 bw = rspec_get_bw(rspec); | ||||||
|  | 
 | ||||||
|  | 	return bw == PHY_TXC1_BW_40MHZ || bw == PHY_TXC1_BW_40MHZ_DUP; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline uint rspec2rate(u32 rspec) | ||||||
|  | { | ||||||
|  | 	if (rspec & RSPEC_MIMORATE) | ||||||
|  | 		return mcs_2_rate(rspec & RSPEC_RATE_MASK, rspec_is40mhz(rspec), | ||||||
|  | 				  rspec_issgi(rspec)); | ||||||
|  | 	return rspec & RSPEC_RATE_MASK; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline u8 rspec_mimoplcp3(u32 rspec) | ||||||
|  | { | ||||||
|  | 	return (rspec & 0xf00000) >> 16; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline bool plcp3_issgi(u8 plcp) | ||||||
|  | { | ||||||
|  | 	return (plcp & (RSPEC_SHORT_GI >> 16)) != 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline uint rspec_stc(u32 rspec) | ||||||
|  | { | ||||||
|  | 	return (rspec & RSPEC_STC_MASK) >> RSPEC_STC_SHIFT; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline uint rspec_stf(u32 rspec) | ||||||
|  | { | ||||||
|  | 	return (rspec & RSPEC_STF_MASK) >> RSPEC_STF_SHIFT; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline bool is_mcs_rate(u32 ratespec) | ||||||
|  | { | ||||||
|  | 	return (ratespec & RSPEC_MIMORATE) != 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline bool is_ofdm_rate(u32 ratespec) | ||||||
|  | { | ||||||
|  | 	return !is_mcs_rate(ratespec) && | ||||||
|  | 	       (rate_info[ratespec & RSPEC_RATE_MASK] & BRCMS_RATE_FLAG); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline bool is_cck_rate(u32 ratespec) | ||||||
|  | { | ||||||
|  | 	u32 rate = (ratespec & BRCMS_RATE_MASK); | ||||||
|  | 
 | ||||||
|  | 	return !is_mcs_rate(ratespec) && ( | ||||||
|  | 			rate == BRCM_RATE_1M || rate == BRCM_RATE_2M || | ||||||
|  | 			rate == BRCM_RATE_5M5 || rate == BRCM_RATE_11M); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline bool is_single_stream(u8 mcs) | ||||||
|  | { | ||||||
|  | 	return mcs <= HIGHEST_SINGLE_STREAM_MCS || mcs == 32; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline u8 cck_rspec(u8 cck) | ||||||
|  | { | ||||||
|  | 	return cck & RSPEC_RATE_MASK; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Convert encoded rate value in plcp header to numerical rates in 500 KHz
 | ||||||
|  |  * increments */ | ||||||
|  | extern const u8 ofdm_rate_lookup[]; | ||||||
|  | 
 | ||||||
|  | static inline u8 ofdm_phy2mac_rate(u8 rlpt) | ||||||
|  | { | ||||||
|  | 	return ofdm_rate_lookup[rlpt & 0x7]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline u8 cck_phy2mac_rate(u8 signal) | ||||||
|  | { | ||||||
|  | 	return signal/5; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Rates specified in brcms_c_rateset_filter() */ | ||||||
|  | #define BRCMS_RATES_CCK_OFDM	0 | ||||||
|  | #define BRCMS_RATES_CCK		1 | ||||||
|  | #define BRCMS_RATES_OFDM		2 | ||||||
|  | 
 | ||||||
|  | /* sanitize, and sort a rateset with the basic bit(s) preserved, validate
 | ||||||
|  |  * rateset */ | ||||||
|  | extern bool | ||||||
|  | brcms_c_rate_hwrs_filter_sort_validate(struct brcms_c_rateset *rs, | ||||||
|  | 				       const struct brcms_c_rateset *hw_rs, | ||||||
|  | 				       bool check_brate, u8 txstreams); | ||||||
|  | /* copy rateset src to dst as-is (no masking or sorting) */ | ||||||
|  | extern void brcms_c_rateset_copy(const struct brcms_c_rateset *src, | ||||||
|  | 			     struct brcms_c_rateset *dst); | ||||||
|  | 
 | ||||||
|  | /* would be nice to have these documented ... */ | ||||||
|  | extern u32 brcms_c_compute_rspec(struct d11rxhdr *rxh, u8 *plcp); | ||||||
|  | 
 | ||||||
|  | extern void brcms_c_rateset_filter(struct brcms_c_rateset *src, | ||||||
|  | 	struct brcms_c_rateset *dst, bool basic_only, u8 rates, uint xmask, | ||||||
|  | 	bool mcsallow); | ||||||
|  | 
 | ||||||
|  | extern void | ||||||
|  | brcms_c_rateset_default(struct brcms_c_rateset *rs_tgt, | ||||||
|  | 			const struct brcms_c_rateset *rs_hw, uint phy_type, | ||||||
|  | 			int bandtype, bool cck_only, uint rate_mask, | ||||||
|  | 			bool mcsallow, u8 bw, u8 txstreams); | ||||||
|  | 
 | ||||||
|  | extern s16 brcms_c_rate_legacy_phyctl(uint rate); | ||||||
|  | 
 | ||||||
|  | extern void brcms_c_rateset_mcs_upd(struct brcms_c_rateset *rs, u8 txstreams); | ||||||
|  | extern void brcms_c_rateset_mcs_clear(struct brcms_c_rateset *rateset); | ||||||
|  | extern void brcms_c_rateset_mcs_build(struct brcms_c_rateset *rateset, | ||||||
|  | 				      u8 txstreams); | ||||||
|  | extern void brcms_c_rateset_bw_mcs_filter(struct brcms_c_rateset *rateset, | ||||||
|  | 					  u8 bw); | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCM_RATE_H_ */ | ||||||
							
								
								
									
										82
									
								
								drivers/net/wireless/brcm80211/brcmsmac/scb.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								drivers/net/wireless/brcm80211/brcmsmac/scb.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,82 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCM_SCB_H_ | ||||||
|  | #define _BRCM_SCB_H_ | ||||||
|  | 
 | ||||||
|  | #include <linux/if_ether.h> | ||||||
|  | #include <brcmu_utils.h> | ||||||
|  | #include <defs.h> | ||||||
|  | #include "types.h" | ||||||
|  | 
 | ||||||
|  | #define AMPDU_TX_BA_MAX_WSIZE	64	/* max Tx ba window size (in pdu) */ | ||||||
|  | 
 | ||||||
|  | #define AMPDU_MAX_SCB_TID	NUMPRIO | ||||||
|  | 
 | ||||||
|  | /* scb flags */ | ||||||
|  | #define SCB_WMECAP		0x0040 | ||||||
|  | #define SCB_HTCAP		0x10000	/* HT (MIMO) capable device */ | ||||||
|  | #define SCB_IS40		0x80000	/* 40MHz capable */ | ||||||
|  | #define SCB_STBCCAP		0x40000000	/* STBC Capable */ | ||||||
|  | 
 | ||||||
|  | #define SCB_MAGIC	0xbeefcafe | ||||||
|  | 
 | ||||||
|  | /* structure to store per-tid state for the ampdu initiator */ | ||||||
|  | struct scb_ampdu_tid_ini { | ||||||
|  | 	u8 tx_in_transit; /* number of pending mpdus in transit in driver */ | ||||||
|  | 	u8 tid;		  /* initiator tid for easy lookup */ | ||||||
|  | 	/* tx retry count; indexed by seq modulo */ | ||||||
|  | 	u8 txretry[AMPDU_TX_BA_MAX_WSIZE]; | ||||||
|  | 	struct scb *scb;  /* backptr for easy lookup */ | ||||||
|  | 	u8 ba_wsize;	  /* negotiated ba window size (in pdu) */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct scb_ampdu { | ||||||
|  | 	struct scb *scb;	/* back pointer for easy reference */ | ||||||
|  | 	u8 mpdu_density;	/* mpdu density */ | ||||||
|  | 	u8 max_pdu;		/* max pdus allowed in ampdu */ | ||||||
|  | 	u8 release;		/* # of mpdus released at a time */ | ||||||
|  | 	u16 min_len;		/* min mpdu len to support the density */ | ||||||
|  | 	u32 max_rx_ampdu_bytes;	/* max ampdu rcv length; 8k, 16k, 32k, 64k */ | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * This could easily be a ini[] pointer and we keep this info in wl | ||||||
|  | 	 * itself instead of having mac80211 hold it for us. Also could be made | ||||||
|  | 	 * dynamic per tid instead of static. | ||||||
|  | 	 */ | ||||||
|  | 	/* initiator info - per tid (NUMPRIO): */ | ||||||
|  | 	struct scb_ampdu_tid_ini ini[AMPDU_MAX_SCB_TID]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* station control block - one per remote MAC address */ | ||||||
|  | struct scb { | ||||||
|  | 	u32 magic; | ||||||
|  | 	u32 flags;	/* various bit flags as defined below */ | ||||||
|  | 	u32 flags2;	/* various bit flags2 as defined below */ | ||||||
|  | 	u8 state;	/* current state bitfield of auth/assoc process */ | ||||||
|  | 	u8 ea[ETH_ALEN];	/* station address */ | ||||||
|  | 	uint fragresid[NUMPRIO];/* #bytes unused in frag buffer per prio */ | ||||||
|  | 
 | ||||||
|  | 	u16 seqctl[NUMPRIO];	/* seqctl of last received frame (for dups) */ | ||||||
|  | 	/* seqctl of last received frame (for dups) for non-QoS data and
 | ||||||
|  | 	 * management */ | ||||||
|  | 	u16 seqctl_nonqos; | ||||||
|  | 	u16 seqnum[NUMPRIO];/* WME: driver maintained sw seqnum per priority */ | ||||||
|  | 
 | ||||||
|  | 	struct scb_ampdu scb_ampdu;	/* AMPDU state including per tid info */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCM_SCB_H_ */ | ||||||
							
								
								
									
										1298
									
								
								drivers/net/wireless/brcm80211/brcmsmac/srom.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1298
									
								
								drivers/net/wireless/brcm80211/brcmsmac/srom.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										34
									
								
								drivers/net/wireless/brcm80211/brcmsmac/srom.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								drivers/net/wireless/brcm80211/brcmsmac/srom.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,34 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef	_BRCM_SROM_H_ | ||||||
|  | #define	_BRCM_SROM_H_ | ||||||
|  | 
 | ||||||
|  | #include "types.h" | ||||||
|  | 
 | ||||||
|  | /* Prototypes */ | ||||||
|  | extern int srom_var_init(struct si_pub *sih, void __iomem *curmap); | ||||||
|  | extern void srom_free_vars(struct si_pub *sih); | ||||||
|  | 
 | ||||||
|  | extern int srom_read(struct si_pub *sih, uint bus, void *curmap, | ||||||
|  | 		     uint byteoff, uint nbytes, u16 *buf, bool check_crc); | ||||||
|  | 
 | ||||||
|  | /* parse standard PCMCIA cis, normally used by SB/PCMCIA/SDIO/SPI/OTP
 | ||||||
|  |  *   and extract from it into name=value pairs | ||||||
|  |  */ | ||||||
|  | extern int srom_parsecis(u8 **pcis, uint ciscnt, | ||||||
|  | 			 char **vars, uint *count); | ||||||
|  | #endif				/* _BRCM_SROM_H_ */ | ||||||
							
								
								
									
										438
									
								
								drivers/net/wireless/brcm80211/brcmsmac/stf.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										438
									
								
								drivers/net/wireless/brcm80211/brcmsmac/stf.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,438 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <net/mac80211.h> | ||||||
|  | 
 | ||||||
|  | #include "types.h" | ||||||
|  | #include "d11.h" | ||||||
|  | #include "rate.h" | ||||||
|  | #include "phy/phy_hal.h" | ||||||
|  | #include "channel.h" | ||||||
|  | #include "main.h" | ||||||
|  | #include "stf.h" | ||||||
|  | 
 | ||||||
|  | #define MIN_SPATIAL_EXPANSION	0 | ||||||
|  | #define MAX_SPATIAL_EXPANSION	1 | ||||||
|  | 
 | ||||||
|  | #define BRCMS_STF_SS_STBC_RX(wlc) (BRCMS_ISNPHY(wlc->band) && \ | ||||||
|  | 	NREV_GT(wlc->band->phyrev, 3) && NREV_LE(wlc->band->phyrev, 6)) | ||||||
|  | 
 | ||||||
|  | #define BRCMS_BITSCNT(x)	brcmu_bitcount((u8 *)&(x), sizeof(u8)) | ||||||
|  | 
 | ||||||
|  | #define NSTS_1	1 | ||||||
|  | #define NSTS_2	2 | ||||||
|  | #define NSTS_3	3 | ||||||
|  | #define NSTS_4	4 | ||||||
|  | 
 | ||||||
|  | static const u8 txcore_default[5] = { | ||||||
|  | 	(0),			/* bitmap of the core enabled */ | ||||||
|  | 	(0x01),			/* For Nsts = 1, enable core 1 */ | ||||||
|  | 	(0x03),			/* For Nsts = 2, enable core 1 & 2 */ | ||||||
|  | 	(0x07),			/* For Nsts = 3, enable core 1, 2 & 3 */ | ||||||
|  | 	(0x0f)			/* For Nsts = 4, enable all cores */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static void brcms_c_stf_stbc_rx_ht_update(struct brcms_c_info *wlc, int val) | ||||||
|  | { | ||||||
|  | 	/* MIMOPHYs rev3-6 cannot receive STBC with only one rx core active */ | ||||||
|  | 	if (BRCMS_STF_SS_STBC_RX(wlc)) { | ||||||
|  | 		if ((wlc->stf->rxstreams == 1) && (val != HT_CAP_RX_STBC_NO)) | ||||||
|  | 			return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (wlc->pub->up) { | ||||||
|  | 		brcms_c_update_beacon(wlc); | ||||||
|  | 		brcms_c_update_probe_resp(wlc, true); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * every WLC_TEMPSENSE_PERIOD seconds temperature check to decide whether to | ||||||
|  |  * turn on/off txchain. | ||||||
|  |  */ | ||||||
|  | void brcms_c_tempsense_upd(struct brcms_c_info *wlc) | ||||||
|  | { | ||||||
|  | 	struct brcms_phy_pub *pi = wlc->band->pi; | ||||||
|  | 	uint active_chains, txchain; | ||||||
|  | 
 | ||||||
|  | 	/* Check if the chip is too hot. Disable one Tx chain, if it is */ | ||||||
|  | 	/* high 4 bits are for Rx chain, low 4 bits are  for Tx chain */ | ||||||
|  | 	active_chains = wlc_phy_stf_chain_active_get(pi); | ||||||
|  | 	txchain = active_chains & 0xf; | ||||||
|  | 
 | ||||||
|  | 	if (wlc->stf->txchain == wlc->stf->hw_txchain) { | ||||||
|  | 		if (txchain && (txchain < wlc->stf->hw_txchain)) | ||||||
|  | 			/* turn off 1 tx chain */ | ||||||
|  | 			brcms_c_stf_txchain_set(wlc, txchain, true); | ||||||
|  | 	} else if (wlc->stf->txchain < wlc->stf->hw_txchain) { | ||||||
|  | 		if (txchain == wlc->stf->hw_txchain) | ||||||
|  | 			/* turn back on txchain */ | ||||||
|  | 			brcms_c_stf_txchain_set(wlc, txchain, true); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | brcms_c_stf_ss_algo_channel_get(struct brcms_c_info *wlc, u16 *ss_algo_channel, | ||||||
|  | 			    u16 chanspec) | ||||||
|  | { | ||||||
|  | 	struct tx_power power; | ||||||
|  | 	u8 siso_mcs_id, cdd_mcs_id, stbc_mcs_id; | ||||||
|  | 
 | ||||||
|  | 	/* Clear previous settings */ | ||||||
|  | 	*ss_algo_channel = 0; | ||||||
|  | 
 | ||||||
|  | 	if (!wlc->pub->up) { | ||||||
|  | 		*ss_algo_channel = (u16) -1; | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wlc_phy_txpower_get_current(wlc->band->pi, &power, | ||||||
|  | 				    CHSPEC_CHANNEL(chanspec)); | ||||||
|  | 
 | ||||||
|  | 	siso_mcs_id = (CHSPEC_IS40(chanspec)) ? | ||||||
|  | 	    WL_TX_POWER_MCS40_SISO_FIRST : WL_TX_POWER_MCS20_SISO_FIRST; | ||||||
|  | 	cdd_mcs_id = (CHSPEC_IS40(chanspec)) ? | ||||||
|  | 	    WL_TX_POWER_MCS40_CDD_FIRST : WL_TX_POWER_MCS20_CDD_FIRST; | ||||||
|  | 	stbc_mcs_id = (CHSPEC_IS40(chanspec)) ? | ||||||
|  | 	    WL_TX_POWER_MCS40_STBC_FIRST : WL_TX_POWER_MCS20_STBC_FIRST; | ||||||
|  | 
 | ||||||
|  | 	/* criteria to choose stf mode */ | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * the "+3dbm (12 0.25db units)" is to account for the fact that with | ||||||
|  | 	 * CDD, tx occurs on both chains | ||||||
|  | 	 */ | ||||||
|  | 	if (power.target[siso_mcs_id] > (power.target[cdd_mcs_id] + 12)) | ||||||
|  | 		setbit(ss_algo_channel, PHY_TXC1_MODE_SISO); | ||||||
|  | 	else | ||||||
|  | 		setbit(ss_algo_channel, PHY_TXC1_MODE_CDD); | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * STBC is ORed into to algo channel as STBC requires per-packet SCB | ||||||
|  | 	 * capability check so cannot be default mode of operation. One of | ||||||
|  | 	 * SISO, CDD have to be set | ||||||
|  | 	 */ | ||||||
|  | 	if (power.target[siso_mcs_id] <= (power.target[stbc_mcs_id] + 12)) | ||||||
|  | 		setbit(ss_algo_channel, PHY_TXC1_MODE_STBC); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static bool brcms_c_stf_stbc_tx_set(struct brcms_c_info *wlc, s32 int_val) | ||||||
|  | { | ||||||
|  | 	if ((int_val != AUTO) && (int_val != OFF) && (int_val != ON)) | ||||||
|  | 		return false; | ||||||
|  | 
 | ||||||
|  | 	if ((int_val == ON) && (wlc->stf->txstreams == 1)) | ||||||
|  | 		return false; | ||||||
|  | 
 | ||||||
|  | 	wlc->bandstate[BAND_2G_INDEX]->band_stf_stbc_tx = (s8) int_val; | ||||||
|  | 	wlc->bandstate[BAND_5G_INDEX]->band_stf_stbc_tx = (s8) int_val; | ||||||
|  | 
 | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool brcms_c_stf_stbc_rx_set(struct brcms_c_info *wlc, s32 int_val) | ||||||
|  | { | ||||||
|  | 	if ((int_val != HT_CAP_RX_STBC_NO) | ||||||
|  | 	    && (int_val != HT_CAP_RX_STBC_ONE_STREAM)) | ||||||
|  | 		return false; | ||||||
|  | 
 | ||||||
|  | 	if (BRCMS_STF_SS_STBC_RX(wlc)) { | ||||||
|  | 		if ((int_val != HT_CAP_RX_STBC_NO) | ||||||
|  | 		    && (wlc->stf->rxstreams == 1)) | ||||||
|  | 			return false; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	brcms_c_stf_stbc_rx_ht_update(wlc, int_val); | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int brcms_c_stf_txcore_set(struct brcms_c_info *wlc, u8 Nsts, | ||||||
|  | 				  u8 core_mask) | ||||||
|  | { | ||||||
|  | 	BCMMSG(wlc->wiphy, "wl%d: Nsts %d core_mask %x\n", | ||||||
|  | 		 wlc->pub->unit, Nsts, core_mask); | ||||||
|  | 
 | ||||||
|  | 	if (hweight8(core_mask) > wlc->stf->txstreams) | ||||||
|  | 		core_mask = 0; | ||||||
|  | 
 | ||||||
|  | 	if ((hweight8(core_mask) == wlc->stf->txstreams) && | ||||||
|  | 	    ((core_mask & ~wlc->stf->txchain) | ||||||
|  | 	     || !(core_mask & wlc->stf->txchain))) | ||||||
|  | 		core_mask = wlc->stf->txchain; | ||||||
|  | 
 | ||||||
|  | 	wlc->stf->txcore[Nsts] = core_mask; | ||||||
|  | 	/* Nsts = 1..4, txcore index = 1..4 */ | ||||||
|  | 	if (Nsts == 1) { | ||||||
|  | 		/* Needs to update beacon and ucode generated response
 | ||||||
|  | 		 * frames when 1 stream core map changed | ||||||
|  | 		 */ | ||||||
|  | 		wlc->stf->phytxant = core_mask << PHY_TXC_ANT_SHIFT; | ||||||
|  | 		brcms_b_txant_set(wlc->hw, wlc->stf->phytxant); | ||||||
|  | 		if (wlc->clk) { | ||||||
|  | 			brcms_c_suspend_mac_and_wait(wlc); | ||||||
|  | 			brcms_c_beacon_phytxctl_txant_upd(wlc, wlc->bcn_rspec); | ||||||
|  | 			brcms_c_enable_mac(wlc); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int brcms_c_stf_spatial_policy_set(struct brcms_c_info *wlc, int val) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	u8 core_mask = 0; | ||||||
|  | 
 | ||||||
|  | 	BCMMSG(wlc->wiphy, "wl%d: val %x\n", wlc->pub->unit, val); | ||||||
|  | 
 | ||||||
|  | 	wlc->stf->spatial_policy = (s8) val; | ||||||
|  | 	for (i = 1; i <= MAX_STREAMS_SUPPORTED; i++) { | ||||||
|  | 		core_mask = (val == MAX_SPATIAL_EXPANSION) ? | ||||||
|  | 		    wlc->stf->txchain : txcore_default[i]; | ||||||
|  | 		brcms_c_stf_txcore_set(wlc, (u8) i, core_mask); | ||||||
|  | 	} | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Centralized txant update function. call it whenever wlc->stf->txant and/or | ||||||
|  |  * wlc->stf->txchain change. | ||||||
|  |  * | ||||||
|  |  * Antennas are controlled by ucode indirectly, which drives PHY or GPIO to | ||||||
|  |  * achieve various tx/rx antenna selection schemes | ||||||
|  |  * | ||||||
|  |  * legacy phy, bit 6 and bit 7 means antenna 0 and 1 respectively, bit6+bit7 | ||||||
|  |  * means auto(last rx). | ||||||
|  |  * for NREV<3, bit 6 and bit 7 means antenna 0 and 1 respectively, bit6+bit7 | ||||||
|  |  * means last rx and do tx-antenna selection for SISO transmissions | ||||||
|  |  * for NREV=3, bit 6 and bit _8_ means antenna 0 and 1 respectively, bit6+bit7 | ||||||
|  |  * means last rx and do tx-antenna selection for SISO transmissions | ||||||
|  |  * for NREV>=7, bit 6 and bit 7 mean antenna 0 and 1 respectively, nit6+bit7 | ||||||
|  |  * means both cores active | ||||||
|  | */ | ||||||
|  | static void _brcms_c_stf_phy_txant_upd(struct brcms_c_info *wlc) | ||||||
|  | { | ||||||
|  | 	s8 txant; | ||||||
|  | 
 | ||||||
|  | 	txant = (s8) wlc->stf->txant; | ||||||
|  | 	if (BRCMS_PHY_11N_CAP(wlc->band)) { | ||||||
|  | 		if (txant == ANT_TX_FORCE_0) { | ||||||
|  | 			wlc->stf->phytxant = PHY_TXC_ANT_0; | ||||||
|  | 		} else if (txant == ANT_TX_FORCE_1) { | ||||||
|  | 			wlc->stf->phytxant = PHY_TXC_ANT_1; | ||||||
|  | 
 | ||||||
|  | 			if (BRCMS_ISNPHY(wlc->band) && | ||||||
|  | 			    NREV_GE(wlc->band->phyrev, 3) | ||||||
|  | 			    && NREV_LT(wlc->band->phyrev, 7)) | ||||||
|  | 				wlc->stf->phytxant = PHY_TXC_ANT_2; | ||||||
|  | 		} else { | ||||||
|  | 			if (BRCMS_ISLCNPHY(wlc->band) || | ||||||
|  | 			    BRCMS_ISSSLPNPHY(wlc->band)) | ||||||
|  | 				wlc->stf->phytxant = PHY_TXC_LCNPHY_ANT_LAST; | ||||||
|  | 			else { | ||||||
|  | 				/* catch out of sync wlc->stf->txcore */ | ||||||
|  | 				WARN_ON(wlc->stf->txchain <= 0); | ||||||
|  | 				wlc->stf->phytxant = | ||||||
|  | 				    wlc->stf->txchain << PHY_TXC_ANT_SHIFT; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		if (txant == ANT_TX_FORCE_0) | ||||||
|  | 			wlc->stf->phytxant = PHY_TXC_OLD_ANT_0; | ||||||
|  | 		else if (txant == ANT_TX_FORCE_1) | ||||||
|  | 			wlc->stf->phytxant = PHY_TXC_OLD_ANT_1; | ||||||
|  | 		else | ||||||
|  | 			wlc->stf->phytxant = PHY_TXC_OLD_ANT_LAST; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	brcms_b_txant_set(wlc->hw, wlc->stf->phytxant); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int brcms_c_stf_txchain_set(struct brcms_c_info *wlc, s32 int_val, bool force) | ||||||
|  | { | ||||||
|  | 	u8 txchain = (u8) int_val; | ||||||
|  | 	u8 txstreams; | ||||||
|  | 	uint i; | ||||||
|  | 
 | ||||||
|  | 	if (wlc->stf->txchain == txchain) | ||||||
|  | 		return 0; | ||||||
|  | 
 | ||||||
|  | 	if ((txchain & ~wlc->stf->hw_txchain) | ||||||
|  | 	    || !(txchain & wlc->stf->hw_txchain)) | ||||||
|  | 		return -EINVAL; | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * if nrate override is configured to be non-SISO STF mode, reject | ||||||
|  | 	 * reducing txchain to 1 | ||||||
|  | 	 */ | ||||||
|  | 	txstreams = (u8) hweight8(txchain); | ||||||
|  | 	if (txstreams > MAX_STREAMS_SUPPORTED) | ||||||
|  | 		return -EINVAL; | ||||||
|  | 
 | ||||||
|  | 	wlc->stf->txchain = txchain; | ||||||
|  | 	wlc->stf->txstreams = txstreams; | ||||||
|  | 	brcms_c_stf_stbc_tx_set(wlc, wlc->band->band_stf_stbc_tx); | ||||||
|  | 	brcms_c_stf_ss_update(wlc, wlc->bandstate[BAND_2G_INDEX]); | ||||||
|  | 	brcms_c_stf_ss_update(wlc, wlc->bandstate[BAND_5G_INDEX]); | ||||||
|  | 	wlc->stf->txant = | ||||||
|  | 	    (wlc->stf->txstreams == 1) ? ANT_TX_FORCE_0 : ANT_TX_DEF; | ||||||
|  | 	_brcms_c_stf_phy_txant_upd(wlc); | ||||||
|  | 
 | ||||||
|  | 	wlc_phy_stf_chain_set(wlc->band->pi, wlc->stf->txchain, | ||||||
|  | 			      wlc->stf->rxchain); | ||||||
|  | 
 | ||||||
|  | 	for (i = 1; i <= MAX_STREAMS_SUPPORTED; i++) | ||||||
|  | 		brcms_c_stf_txcore_set(wlc, (u8) i, txcore_default[i]); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * update wlc->stf->ss_opmode which represents the operational stf_ss mode | ||||||
|  |  * we're using | ||||||
|  |  */ | ||||||
|  | int brcms_c_stf_ss_update(struct brcms_c_info *wlc, struct brcms_band *band) | ||||||
|  | { | ||||||
|  | 	int ret_code = 0; | ||||||
|  | 	u8 prev_stf_ss; | ||||||
|  | 	u8 upd_stf_ss; | ||||||
|  | 
 | ||||||
|  | 	prev_stf_ss = wlc->stf->ss_opmode; | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * NOTE: opmode can only be SISO or CDD as STBC is decided on a | ||||||
|  | 	 * per-packet basis | ||||||
|  | 	 */ | ||||||
|  | 	if (BRCMS_STBC_CAP_PHY(wlc) && | ||||||
|  | 	    wlc->stf->ss_algosel_auto | ||||||
|  | 	    && (wlc->stf->ss_algo_channel != (u16) -1)) { | ||||||
|  | 		upd_stf_ss = (wlc->stf->txstreams == 1 || | ||||||
|  | 			      isset(&wlc->stf->ss_algo_channel, | ||||||
|  | 				    PHY_TXC1_MODE_SISO)) ? | ||||||
|  | 				    PHY_TXC1_MODE_SISO : PHY_TXC1_MODE_CDD; | ||||||
|  | 	} else { | ||||||
|  | 		if (wlc->band != band) | ||||||
|  | 			return ret_code; | ||||||
|  | 		upd_stf_ss = (wlc->stf->txstreams == 1) ? | ||||||
|  | 				PHY_TXC1_MODE_SISO : band->band_stf_ss_mode; | ||||||
|  | 	} | ||||||
|  | 	if (prev_stf_ss != upd_stf_ss) { | ||||||
|  | 		wlc->stf->ss_opmode = upd_stf_ss; | ||||||
|  | 		brcms_b_band_stf_ss_set(wlc->hw, upd_stf_ss); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return ret_code; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int brcms_c_stf_attach(struct brcms_c_info *wlc) | ||||||
|  | { | ||||||
|  | 	wlc->bandstate[BAND_2G_INDEX]->band_stf_ss_mode = PHY_TXC1_MODE_SISO; | ||||||
|  | 	wlc->bandstate[BAND_5G_INDEX]->band_stf_ss_mode = PHY_TXC1_MODE_CDD; | ||||||
|  | 
 | ||||||
|  | 	if (BRCMS_ISNPHY(wlc->band) && | ||||||
|  | 	    (wlc_phy_txpower_hw_ctrl_get(wlc->band->pi) != PHY_TPC_HW_ON)) | ||||||
|  | 		wlc->bandstate[BAND_2G_INDEX]->band_stf_ss_mode = | ||||||
|  | 		    PHY_TXC1_MODE_CDD; | ||||||
|  | 	brcms_c_stf_ss_update(wlc, wlc->bandstate[BAND_2G_INDEX]); | ||||||
|  | 	brcms_c_stf_ss_update(wlc, wlc->bandstate[BAND_5G_INDEX]); | ||||||
|  | 
 | ||||||
|  | 	brcms_c_stf_stbc_rx_ht_update(wlc, HT_CAP_RX_STBC_NO); | ||||||
|  | 	wlc->bandstate[BAND_2G_INDEX]->band_stf_stbc_tx = OFF; | ||||||
|  | 	wlc->bandstate[BAND_5G_INDEX]->band_stf_stbc_tx = OFF; | ||||||
|  | 
 | ||||||
|  | 	if (BRCMS_STBC_CAP_PHY(wlc)) { | ||||||
|  | 		wlc->stf->ss_algosel_auto = true; | ||||||
|  | 		/* Init the default value */ | ||||||
|  | 		wlc->stf->ss_algo_channel = (u16) -1; | ||||||
|  | 	} | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void brcms_c_stf_detach(struct brcms_c_info *wlc) | ||||||
|  | { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void brcms_c_stf_phy_txant_upd(struct brcms_c_info *wlc) | ||||||
|  | { | ||||||
|  | 	_brcms_c_stf_phy_txant_upd(wlc); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void brcms_c_stf_phy_chain_calc(struct brcms_c_info *wlc) | ||||||
|  | { | ||||||
|  | 	/* get available rx/tx chains */ | ||||||
|  | 	wlc->stf->hw_txchain = (u8) getintvar(wlc->hw->sih, BRCMS_SROM_TXCHAIN); | ||||||
|  | 	wlc->stf->hw_rxchain = (u8) getintvar(wlc->hw->sih, BRCMS_SROM_RXCHAIN); | ||||||
|  | 
 | ||||||
|  | 	/* these parameter are intended to be used for all PHY types */ | ||||||
|  | 	if (wlc->stf->hw_txchain == 0 || wlc->stf->hw_txchain == 0xf) { | ||||||
|  | 		if (BRCMS_ISNPHY(wlc->band)) | ||||||
|  | 			wlc->stf->hw_txchain = TXCHAIN_DEF_NPHY; | ||||||
|  | 		else | ||||||
|  | 			wlc->stf->hw_txchain = TXCHAIN_DEF; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wlc->stf->txchain = wlc->stf->hw_txchain; | ||||||
|  | 	wlc->stf->txstreams = (u8) hweight8(wlc->stf->hw_txchain); | ||||||
|  | 
 | ||||||
|  | 	if (wlc->stf->hw_rxchain == 0 || wlc->stf->hw_rxchain == 0xf) { | ||||||
|  | 		if (BRCMS_ISNPHY(wlc->band)) | ||||||
|  | 			wlc->stf->hw_rxchain = RXCHAIN_DEF_NPHY; | ||||||
|  | 		else | ||||||
|  | 			wlc->stf->hw_rxchain = RXCHAIN_DEF; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wlc->stf->rxchain = wlc->stf->hw_rxchain; | ||||||
|  | 	wlc->stf->rxstreams = (u8) hweight8(wlc->stf->hw_rxchain); | ||||||
|  | 
 | ||||||
|  | 	/* initialize the txcore table */ | ||||||
|  | 	memcpy(wlc->stf->txcore, txcore_default, sizeof(wlc->stf->txcore)); | ||||||
|  | 
 | ||||||
|  | 	/* default spatial_policy */ | ||||||
|  | 	wlc->stf->spatial_policy = MIN_SPATIAL_EXPANSION; | ||||||
|  | 	brcms_c_stf_spatial_policy_set(wlc, MIN_SPATIAL_EXPANSION); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static u16 _brcms_c_stf_phytxchain_sel(struct brcms_c_info *wlc, | ||||||
|  | 				       u32 rspec) | ||||||
|  | { | ||||||
|  | 	u16 phytxant = wlc->stf->phytxant; | ||||||
|  | 
 | ||||||
|  | 	if (rspec_stf(rspec) != PHY_TXC1_MODE_SISO) | ||||||
|  | 		phytxant = wlc->stf->txchain << PHY_TXC_ANT_SHIFT; | ||||||
|  | 	else if (wlc->stf->txant == ANT_TX_DEF) | ||||||
|  | 		phytxant = wlc->stf->txchain << PHY_TXC_ANT_SHIFT; | ||||||
|  | 	phytxant &= PHY_TXC_ANT_MASK; | ||||||
|  | 	return phytxant; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | u16 brcms_c_stf_phytxchain_sel(struct brcms_c_info *wlc, u32 rspec) | ||||||
|  | { | ||||||
|  | 	return _brcms_c_stf_phytxchain_sel(wlc, rspec); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | u16 brcms_c_stf_d11hdrs_phyctl_txant(struct brcms_c_info *wlc, u32 rspec) | ||||||
|  | { | ||||||
|  | 	u16 phytxant = wlc->stf->phytxant; | ||||||
|  | 	u16 mask = PHY_TXC_ANT_MASK; | ||||||
|  | 
 | ||||||
|  | 	/* for non-siso rates or default setting, use the available chains */ | ||||||
|  | 	if (BRCMS_ISNPHY(wlc->band)) { | ||||||
|  | 		phytxant = _brcms_c_stf_phytxchain_sel(wlc, rspec); | ||||||
|  | 		mask = PHY_TXC_HTANT_MASK; | ||||||
|  | 	} | ||||||
|  | 	phytxant |= phytxant & mask; | ||||||
|  | 	return phytxant; | ||||||
|  | } | ||||||
							
								
								
									
										42
									
								
								drivers/net/wireless/brcm80211/brcmsmac/stf.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								drivers/net/wireless/brcm80211/brcmsmac/stf.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,42 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCM_STF_H_ | ||||||
|  | #define _BRCM_STF_H_ | ||||||
|  | 
 | ||||||
|  | #include "types.h" | ||||||
|  | 
 | ||||||
|  | extern int brcms_c_stf_attach(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_stf_detach(struct brcms_c_info *wlc); | ||||||
|  | 
 | ||||||
|  | extern void brcms_c_tempsense_upd(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_stf_ss_algo_channel_get(struct brcms_c_info *wlc, | ||||||
|  | 					u16 *ss_algo_channel, | ||||||
|  | 					u16 chanspec); | ||||||
|  | extern int brcms_c_stf_ss_update(struct brcms_c_info *wlc, | ||||||
|  | 			     struct brcms_band *band); | ||||||
|  | extern void brcms_c_stf_phy_txant_upd(struct brcms_c_info *wlc); | ||||||
|  | extern int brcms_c_stf_txchain_set(struct brcms_c_info *wlc, s32 int_val, | ||||||
|  | 			       bool force); | ||||||
|  | extern bool brcms_c_stf_stbc_rx_set(struct brcms_c_info *wlc, s32 int_val); | ||||||
|  | extern void brcms_c_stf_phy_txant_upd(struct brcms_c_info *wlc); | ||||||
|  | extern void brcms_c_stf_phy_chain_calc(struct brcms_c_info *wlc); | ||||||
|  | extern u16 brcms_c_stf_phytxchain_sel(struct brcms_c_info *wlc, | ||||||
|  | 				      u32 rspec); | ||||||
|  | extern u16 brcms_c_stf_d11hdrs_phyctl_txant(struct brcms_c_info *wlc, | ||||||
|  | 					u32 rspec); | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCM_STF_H_ */ | ||||||
							
								
								
									
										352
									
								
								drivers/net/wireless/brcm80211/brcmsmac/types.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										352
									
								
								drivers/net/wireless/brcm80211/brcmsmac/types.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,352 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _BRCM_TYPES_H_ | ||||||
|  | #define _BRCM_TYPES_H_ | ||||||
|  | 
 | ||||||
|  | #include <linux/types.h> | ||||||
|  | #include <linux/io.h> | ||||||
|  | 
 | ||||||
|  | #define WL_CHAN_FREQ_RANGE_2G      0 | ||||||
|  | #define WL_CHAN_FREQ_RANGE_5GL     1 | ||||||
|  | #define WL_CHAN_FREQ_RANGE_5GM     2 | ||||||
|  | #define WL_CHAN_FREQ_RANGE_5GH     3 | ||||||
|  | 
 | ||||||
|  | /* boardflags */ | ||||||
|  | 
 | ||||||
|  | /* Board has gpio 9 controlling the PA */ | ||||||
|  | #define	BFL_PACTRL		0x00000002 | ||||||
|  | /* Not ok to power down the chip pll and oscillator */ | ||||||
|  | #define	BFL_NOPLLDOWN		0x00000020 | ||||||
|  | /* Board supports the Front End Module */ | ||||||
|  | #define BFL_FEM			0x00000800 | ||||||
|  | /* Board has an external LNA in 2.4GHz band */ | ||||||
|  | #define BFL_EXTLNA		0x00001000 | ||||||
|  | /* Board has no PA */ | ||||||
|  | #define BFL_NOPA		0x00010000 | ||||||
|  | /* Power topology uses BUCKBOOST */ | ||||||
|  | #define BFL_BUCKBOOST		0x00200000 | ||||||
|  | /* Board has FEM and switch to share antenna w/ BT */ | ||||||
|  | #define BFL_FEM_BT		0x00400000 | ||||||
|  | /* Power topology doesn't use CBUCK */ | ||||||
|  | #define BFL_NOCBUCK		0x00800000 | ||||||
|  | /* Power topology uses PALDO */ | ||||||
|  | #define BFL_PALDO		0x02000000 | ||||||
|  | /* Board has an external LNA in 5GHz band */ | ||||||
|  | #define BFL_EXTLNA_5GHz		0x10000000 | ||||||
|  | 
 | ||||||
|  | /* boardflags2 */ | ||||||
|  | 
 | ||||||
|  | /* Board has an external rxbb regulator */ | ||||||
|  | #define BFL2_RXBB_INT_REG_DIS	0x00000001 | ||||||
|  | /* Flag to implement alternative A-band PLL settings */ | ||||||
|  | #define BFL2_APLL_WAR		0x00000002 | ||||||
|  | /* Board permits enabling TX Power Control */ | ||||||
|  | #define BFL2_TXPWRCTRL_EN	0x00000004 | ||||||
|  | /* Board supports the 2X4 diversity switch */ | ||||||
|  | #define BFL2_2X4_DIV		0x00000008 | ||||||
|  | /* Board supports 5G band power gain */ | ||||||
|  | #define BFL2_5G_PWRGAIN		0x00000010 | ||||||
|  | /* Board overrides ASPM and Clkreq settings */ | ||||||
|  | #define BFL2_PCIEWAR_OVR	0x00000020 | ||||||
|  | #define BFL2_LEGACY		0x00000080 | ||||||
|  | /* 4321mcm93 board uses Skyworks FEM */ | ||||||
|  | #define BFL2_SKWRKFEM_BRD	0x00000100 | ||||||
|  | /* Board has a WAR for clock-harmonic spurs */ | ||||||
|  | #define BFL2_SPUR_WAR		0x00000200 | ||||||
|  | /* Flag to narrow G-band PLL loop b/w */ | ||||||
|  | #define BFL2_GPLL_WAR		0x00000400 | ||||||
|  | /* Tx CCK pkts on Ant 0 only */ | ||||||
|  | #define BFL2_SINGLEANT_CCK	0x00001000 | ||||||
|  | /* WAR to reduce and avoid clock-harmonic spurs in 2G */ | ||||||
|  | #define BFL2_2G_SPUR_WAR	0x00002000 | ||||||
|  | /* Flag to widen G-band PLL loop b/w */ | ||||||
|  | #define BFL2_GPLL_WAR2	        0x00010000 | ||||||
|  | #define BFL2_IPALVLSHIFT_3P3    0x00020000 | ||||||
|  | /* Use internal envelope detector for TX IQCAL */ | ||||||
|  | #define BFL2_INTERNDET_TXIQCAL  0x00040000 | ||||||
|  | /* Keep the buffered Xtal output from radio "ON". Most drivers will turn it
 | ||||||
|  |  * off without this flag to save power. */ | ||||||
|  | #define BFL2_XTALBUFOUTEN       0x00080000 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * board specific GPIO assignment, gpio 0-3 are also customer-configurable | ||||||
|  |  * led | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* bit 9 controls the PA on new 4306 boards */ | ||||||
|  | #define	BOARD_GPIO_PACTRL	0x200 | ||||||
|  | #define BOARD_GPIO_12		0x1000 | ||||||
|  | #define BOARD_GPIO_13		0x2000 | ||||||
|  | 
 | ||||||
|  | /* **** Core type/rev defaults **** */ | ||||||
|  | #define D11CONF		0x0fffffb0	/* Supported  D11 revs: 4, 5, 7-27 | ||||||
|  | 					 * also need to update wlc.h MAXCOREREV | ||||||
|  | 					 */ | ||||||
|  | 
 | ||||||
|  | #define NCONF		0x000001ff	/* Supported nphy revs: | ||||||
|  | 					 *      0       4321a0 | ||||||
|  | 					 *      1       4321a1 | ||||||
|  | 					 *      2       4321b0/b1/c0/c1 | ||||||
|  | 					 *      3       4322a0 | ||||||
|  | 					 *      4       4322a1 | ||||||
|  | 					 *      5       4716a0 | ||||||
|  | 					 *      6       43222a0, 43224a0 | ||||||
|  | 					 *      7       43226a0 | ||||||
|  | 					 *      8       5357a0, 43236a0 | ||||||
|  | 					 */ | ||||||
|  | 
 | ||||||
|  | #define LCNCONF		0x00000007	/* Supported lcnphy revs: | ||||||
|  | 					 *      0       4313a0, 4336a0, 4330a0 | ||||||
|  | 					 *      1 | ||||||
|  | 					 *      2       4330a0 | ||||||
|  | 					 */ | ||||||
|  | 
 | ||||||
|  | #define SSLPNCONF	0x0000000f	/* Supported sslpnphy revs: | ||||||
|  | 					 *      0       4329a0/k0 | ||||||
|  | 					 *      1       4329b0/4329C0 | ||||||
|  | 					 *      2       4319a0 | ||||||
|  | 					 *      3       5356a0 | ||||||
|  | 					 */ | ||||||
|  | 
 | ||||||
|  | /********************************************************************
 | ||||||
|  |  * Phy/Core Configuration.  Defines macros to to check core phy/rev * | ||||||
|  |  * compile-time configuration.  Defines default core support.       * | ||||||
|  |  * ****************************************************************** | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* Basic macros to check a configuration bitmask */ | ||||||
|  | 
 | ||||||
|  | #define CONF_HAS(config, val)	((config) & (1 << (val))) | ||||||
|  | #define CONF_MSK(config, mask)	((config) & (mask)) | ||||||
|  | #define MSK_RANGE(low, hi)	((1 << ((hi)+1)) - (1 << (low))) | ||||||
|  | #define CONF_RANGE(config, low, hi) (CONF_MSK(config, MSK_RANGE(low, high))) | ||||||
|  | 
 | ||||||
|  | #define CONF_IS(config, val)	((config) == (1 << (val))) | ||||||
|  | #define CONF_GE(config, val)	((config) & (0-(1 << (val)))) | ||||||
|  | #define CONF_GT(config, val)	((config) & (0-2*(1 << (val)))) | ||||||
|  | #define CONF_LT(config, val)	((config) & ((1 << (val))-1)) | ||||||
|  | #define CONF_LE(config, val)	((config) & (2*(1 << (val))-1)) | ||||||
|  | 
 | ||||||
|  | /* Wrappers for some of the above, specific to config constants */ | ||||||
|  | 
 | ||||||
|  | #define NCONF_HAS(val)	CONF_HAS(NCONF, val) | ||||||
|  | #define NCONF_MSK(mask)	CONF_MSK(NCONF, mask) | ||||||
|  | #define NCONF_IS(val)	CONF_IS(NCONF, val) | ||||||
|  | #define NCONF_GE(val)	CONF_GE(NCONF, val) | ||||||
|  | #define NCONF_GT(val)	CONF_GT(NCONF, val) | ||||||
|  | #define NCONF_LT(val)	CONF_LT(NCONF, val) | ||||||
|  | #define NCONF_LE(val)	CONF_LE(NCONF, val) | ||||||
|  | 
 | ||||||
|  | #define LCNCONF_HAS(val)	CONF_HAS(LCNCONF, val) | ||||||
|  | #define LCNCONF_MSK(mask)	CONF_MSK(LCNCONF, mask) | ||||||
|  | #define LCNCONF_IS(val)		CONF_IS(LCNCONF, val) | ||||||
|  | #define LCNCONF_GE(val)		CONF_GE(LCNCONF, val) | ||||||
|  | #define LCNCONF_GT(val)		CONF_GT(LCNCONF, val) | ||||||
|  | #define LCNCONF_LT(val)		CONF_LT(LCNCONF, val) | ||||||
|  | #define LCNCONF_LE(val)		CONF_LE(LCNCONF, val) | ||||||
|  | 
 | ||||||
|  | #define D11CONF_HAS(val) CONF_HAS(D11CONF, val) | ||||||
|  | #define D11CONF_MSK(mask) CONF_MSK(D11CONF, mask) | ||||||
|  | #define D11CONF_IS(val)	CONF_IS(D11CONF, val) | ||||||
|  | #define D11CONF_GE(val)	CONF_GE(D11CONF, val) | ||||||
|  | #define D11CONF_GT(val)	CONF_GT(D11CONF, val) | ||||||
|  | #define D11CONF_LT(val)	CONF_LT(D11CONF, val) | ||||||
|  | #define D11CONF_LE(val)	CONF_LE(D11CONF, val) | ||||||
|  | 
 | ||||||
|  | #define PHYCONF_HAS(val) CONF_HAS(PHYTYPE, val) | ||||||
|  | #define PHYCONF_IS(val)	CONF_IS(PHYTYPE, val) | ||||||
|  | 
 | ||||||
|  | #define NREV_IS(var, val) \ | ||||||
|  | 	(NCONF_HAS(val) && (NCONF_IS(val) || ((var) == (val)))) | ||||||
|  | 
 | ||||||
|  | #define NREV_GE(var, val) \ | ||||||
|  | 	(NCONF_GE(val) && (!NCONF_LT(val) || ((var) >= (val)))) | ||||||
|  | 
 | ||||||
|  | #define NREV_GT(var, val) \ | ||||||
|  | 	(NCONF_GT(val) && (!NCONF_LE(val) || ((var) > (val)))) | ||||||
|  | 
 | ||||||
|  | #define NREV_LT(var, val) \ | ||||||
|  | 	(NCONF_LT(val) && (!NCONF_GE(val) || ((var) < (val)))) | ||||||
|  | 
 | ||||||
|  | #define NREV_LE(var, val) \ | ||||||
|  | 	(NCONF_LE(val) && (!NCONF_GT(val) || ((var) <= (val)))) | ||||||
|  | 
 | ||||||
|  | #define LCNREV_IS(var, val) \ | ||||||
|  | 	(LCNCONF_HAS(val) && (LCNCONF_IS(val) || ((var) == (val)))) | ||||||
|  | 
 | ||||||
|  | #define LCNREV_GE(var, val) \ | ||||||
|  | 	(LCNCONF_GE(val) && (!LCNCONF_LT(val) || ((var) >= (val)))) | ||||||
|  | 
 | ||||||
|  | #define LCNREV_GT(var, val) \ | ||||||
|  | 	(LCNCONF_GT(val) && (!LCNCONF_LE(val) || ((var) > (val)))) | ||||||
|  | 
 | ||||||
|  | #define LCNREV_LT(var, val) \ | ||||||
|  | 	(LCNCONF_LT(val) && (!LCNCONF_GE(val) || ((var) < (val)))) | ||||||
|  | 
 | ||||||
|  | #define LCNREV_LE(var, val) \ | ||||||
|  | 	(LCNCONF_LE(val) && (!LCNCONF_GT(val) || ((var) <= (val)))) | ||||||
|  | 
 | ||||||
|  | #define D11REV_IS(var, val) \ | ||||||
|  | 	(D11CONF_HAS(val) && (D11CONF_IS(val) || ((var) == (val)))) | ||||||
|  | 
 | ||||||
|  | #define D11REV_GE(var, val) \ | ||||||
|  | 	(D11CONF_GE(val) && (!D11CONF_LT(val) || ((var) >= (val)))) | ||||||
|  | 
 | ||||||
|  | #define D11REV_GT(var, val) \ | ||||||
|  | 	(D11CONF_GT(val) && (!D11CONF_LE(val) || ((var) > (val)))) | ||||||
|  | 
 | ||||||
|  | #define D11REV_LT(var, val) \ | ||||||
|  | 	(D11CONF_LT(val) && (!D11CONF_GE(val) || ((var) < (val)))) | ||||||
|  | 
 | ||||||
|  | #define D11REV_LE(var, val) \ | ||||||
|  | 	(D11CONF_LE(val) && (!D11CONF_GT(val) || ((var) <= (val)))) | ||||||
|  | 
 | ||||||
|  | #define PHYTYPE_IS(var, val)\ | ||||||
|  | 	(PHYCONF_HAS(val) && (PHYCONF_IS(val) || ((var) == (val)))) | ||||||
|  | 
 | ||||||
|  | /* Set up PHYTYPE automatically: (depends on PHY_TYPE_X, from d11.h) */ | ||||||
|  | 
 | ||||||
|  | #define _PHYCONF_N (1 << PHY_TYPE_N) | ||||||
|  | #define _PHYCONF_LCN (1 << PHY_TYPE_LCN) | ||||||
|  | #define _PHYCONF_SSLPN (1 << PHY_TYPE_SSN) | ||||||
|  | 
 | ||||||
|  | #define PHYTYPE (_PHYCONF_N | _PHYCONF_LCN | _PHYCONF_SSLPN) | ||||||
|  | 
 | ||||||
|  | /* Utility macro to identify 802.11n (HT) capable PHYs */ | ||||||
|  | #define PHYTYPE_11N_CAP(phytype) \ | ||||||
|  | 	(PHYTYPE_IS(phytype, PHY_TYPE_N) ||	\ | ||||||
|  | 	 PHYTYPE_IS(phytype, PHY_TYPE_LCN) || \ | ||||||
|  | 	 PHYTYPE_IS(phytype, PHY_TYPE_SSN)) | ||||||
|  | 
 | ||||||
|  | /* Last but not least: shorter wlc-specific var checks */ | ||||||
|  | #define BRCMS_ISNPHY(band)		PHYTYPE_IS((band)->phytype, PHY_TYPE_N) | ||||||
|  | #define BRCMS_ISLCNPHY(band)	PHYTYPE_IS((band)->phytype, PHY_TYPE_LCN) | ||||||
|  | #define BRCMS_ISSSLPNPHY(band)	PHYTYPE_IS((band)->phytype, PHY_TYPE_SSN) | ||||||
|  | 
 | ||||||
|  | #define BRCMS_PHY_11N_CAP(band)	PHYTYPE_11N_CAP((band)->phytype) | ||||||
|  | 
 | ||||||
|  | /**********************************************************************
 | ||||||
|  |  * ------------- End of Core phy/rev configuration. ----------------- * | ||||||
|  |  * ******************************************************************** | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #define BCMMSG(dev, fmt, args...)		\ | ||||||
|  | do {						\ | ||||||
|  | 	if (brcm_msg_level & LOG_TRACE_VAL)	\ | ||||||
|  | 		wiphy_err(dev, "%s: " fmt, __func__, ##args);	\ | ||||||
|  | } while (0) | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Register access macros. | ||||||
|  |  * | ||||||
|  |  * These macro's take a pointer to the address to read as one of their | ||||||
|  |  * arguments. The macro itself deduces the size of the IO transaction (u8, u16 | ||||||
|  |  * or u32). Advantage of this approach in combination with using a struct to | ||||||
|  |  * define the registers in a register block, is that access size and access | ||||||
|  |  * location are defined in only one spot. This reduces the risk of the | ||||||
|  |  * programmer trying to use an unsupported transaction size on a register. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #define R_REG(r) \ | ||||||
|  | 	({ \ | ||||||
|  | 		__typeof(*(r)) __osl_v; \ | ||||||
|  | 		switch (sizeof(*(r))) { \ | ||||||
|  | 		case sizeof(u8): \ | ||||||
|  | 			__osl_v = readb((u8 __iomem *)(r)); \ | ||||||
|  | 			break; \ | ||||||
|  | 		case sizeof(u16): \ | ||||||
|  | 			__osl_v = readw((u16 __iomem *)(r)); \ | ||||||
|  | 			break; \ | ||||||
|  | 		case sizeof(u32): \ | ||||||
|  | 			__osl_v = readl((u32 __iomem *)(r)); \ | ||||||
|  | 			break; \ | ||||||
|  | 		} \ | ||||||
|  | 		__osl_v; \ | ||||||
|  | 	}) | ||||||
|  | 
 | ||||||
|  | #define W_REG(r, v) do { \ | ||||||
|  | 		switch (sizeof(*(r))) { \ | ||||||
|  | 		case sizeof(u8):	\ | ||||||
|  | 			writeb((u8)((v) & 0xFF), (u8 __iomem *)(r)); \ | ||||||
|  | 			break; \ | ||||||
|  | 		case sizeof(u16):	\ | ||||||
|  | 			writew((u16)((v) & 0xFFFF), (u16 __iomem *)(r)); \ | ||||||
|  | 			break; \ | ||||||
|  | 		case sizeof(u32):	\ | ||||||
|  | 			writel((u32)(v), (u32 __iomem *)(r)); \ | ||||||
|  | 			break; \ | ||||||
|  | 		} \ | ||||||
|  | 	} while (0) | ||||||
|  | 
 | ||||||
|  | #ifdef CONFIG_BCM47XX | ||||||
|  | /*
 | ||||||
|  |  * bcm4716 (which includes 4717 & 4718), plus 4706 on PCIe can reorder | ||||||
|  |  * transactions. As a fix, a read after write is performed on certain places | ||||||
|  |  * in the code. Older chips and the newer 5357 family don't require this fix. | ||||||
|  |  */ | ||||||
|  | #define W_REG_FLUSH(r, v)	({ W_REG((r), (v)); (void)R_REG(r); }) | ||||||
|  | #else | ||||||
|  | #define W_REG_FLUSH(r, v)	W_REG((r), (v)) | ||||||
|  | #endif				/* CONFIG_BCM47XX */ | ||||||
|  | 
 | ||||||
|  | #define AND_REG(r, v)	W_REG((r), R_REG(r) & (v)) | ||||||
|  | #define OR_REG(r, v)	W_REG((r), R_REG(r) | (v)) | ||||||
|  | 
 | ||||||
|  | #define SET_REG(r, mask, val) \ | ||||||
|  | 		W_REG((r), ((R_REG(r) & ~(mask)) | (val))) | ||||||
|  | 
 | ||||||
|  | /* multi-bool data type: set of bools, mbool is true if any is set */ | ||||||
|  | 
 | ||||||
|  | /* set one bool */ | ||||||
|  | #define mboolset(mb, bit)		((mb) |= (bit)) | ||||||
|  | /* clear one bool */ | ||||||
|  | #define mboolclr(mb, bit)		((mb) &= ~(bit)) | ||||||
|  | /* true if one bool is set */ | ||||||
|  | #define mboolisset(mb, bit)		(((mb) & (bit)) != 0) | ||||||
|  | #define	mboolmaskset(mb, mask, val)	((mb) = (((mb) & ~(mask)) | (val))) | ||||||
|  | 
 | ||||||
|  | #define CEIL(x, y)		(((x) + ((y)-1)) / (y)) | ||||||
|  | 
 | ||||||
|  | /* forward declarations */ | ||||||
|  | struct wiphy; | ||||||
|  | struct ieee80211_sta; | ||||||
|  | struct ieee80211_tx_queue_params; | ||||||
|  | struct brcms_info; | ||||||
|  | struct brcms_c_info; | ||||||
|  | struct brcms_hardware; | ||||||
|  | struct brcms_txq_info; | ||||||
|  | struct brcms_band; | ||||||
|  | struct dma_pub; | ||||||
|  | struct si_pub; | ||||||
|  | struct tx_status; | ||||||
|  | struct d11rxhdr; | ||||||
|  | struct txpwr_limits; | ||||||
|  | 
 | ||||||
|  | /* iovar structure */ | ||||||
|  | struct brcmu_iovar { | ||||||
|  | 	const char *name;	/* name for lookup and display */ | ||||||
|  | 	u16 varid;	/* id for switch */ | ||||||
|  | 	u16 flags;	/* driver-specific flag bits */ | ||||||
|  | 	u16 type;	/* base type of argument */ | ||||||
|  | 	u16 minlen;	/* min length for buffer vars */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* brcm_msg_level is a bit vector with defs in defs.h */ | ||||||
|  | extern u32 brcm_msg_level; | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCM_TYPES_H_ */ | ||||||
							
								
								
									
										109
									
								
								drivers/net/wireless/brcm80211/brcmsmac/ucode_loader.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										109
									
								
								drivers/net/wireless/brcm80211/brcmsmac/ucode_loader.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,109 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <defs.h> | ||||||
|  | #include "types.h" | ||||||
|  | #include <ucode_loader.h> | ||||||
|  | 
 | ||||||
|  | enum { | ||||||
|  | 	D11UCODE_NAMETAG_START = 0, | ||||||
|  | 	D11LCN0BSINITVALS24, | ||||||
|  | 	D11LCN0INITVALS24, | ||||||
|  | 	D11LCN1BSINITVALS24, | ||||||
|  | 	D11LCN1INITVALS24, | ||||||
|  | 	D11LCN2BSINITVALS24, | ||||||
|  | 	D11LCN2INITVALS24, | ||||||
|  | 	D11N0ABSINITVALS16, | ||||||
|  | 	D11N0BSINITVALS16, | ||||||
|  | 	D11N0INITVALS16, | ||||||
|  | 	D11UCODE_OVERSIGHT16_MIMO, | ||||||
|  | 	D11UCODE_OVERSIGHT16_MIMOSZ, | ||||||
|  | 	D11UCODE_OVERSIGHT24_LCN, | ||||||
|  | 	D11UCODE_OVERSIGHT24_LCNSZ, | ||||||
|  | 	D11UCODE_OVERSIGHT_BOMMAJOR, | ||||||
|  | 	D11UCODE_OVERSIGHT_BOMMINOR | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | int brcms_ucode_data_init(struct brcms_info *wl, struct brcms_ucode *ucode) | ||||||
|  | { | ||||||
|  | 	int rc; | ||||||
|  | 
 | ||||||
|  | 	rc = brcms_check_firmwares(wl); | ||||||
|  | 
 | ||||||
|  | 	rc = rc < 0 ? rc : | ||||||
|  | 		brcms_ucode_init_buf(wl, (void **)&ucode->d11lcn0bsinitvals24, | ||||||
|  | 				     D11LCN0BSINITVALS24); | ||||||
|  | 	rc = rc < 0 ? | ||||||
|  | 	     rc : brcms_ucode_init_buf(wl, (void **)&ucode->d11lcn0initvals24, | ||||||
|  | 				       D11LCN0INITVALS24); | ||||||
|  | 	rc = rc < 0 ? | ||||||
|  | 	     rc : brcms_ucode_init_buf(wl, (void **)&ucode->d11lcn1bsinitvals24, | ||||||
|  | 				       D11LCN1BSINITVALS24); | ||||||
|  | 	rc = rc < 0 ? | ||||||
|  | 	     rc : brcms_ucode_init_buf(wl, (void **)&ucode->d11lcn1initvals24, | ||||||
|  | 				       D11LCN1INITVALS24); | ||||||
|  | 	rc = rc < 0 ? rc : | ||||||
|  | 		brcms_ucode_init_buf(wl, (void **)&ucode->d11lcn2bsinitvals24, | ||||||
|  | 				     D11LCN2BSINITVALS24); | ||||||
|  | 	rc = rc < 0 ? | ||||||
|  | 	     rc : brcms_ucode_init_buf(wl, (void **)&ucode->d11lcn2initvals24, | ||||||
|  | 				       D11LCN2INITVALS24); | ||||||
|  | 	rc = rc < 0 ? | ||||||
|  | 	     rc : brcms_ucode_init_buf(wl, (void **)&ucode->d11n0absinitvals16, | ||||||
|  | 				       D11N0ABSINITVALS16); | ||||||
|  | 	rc = rc < 0 ? | ||||||
|  | 	     rc : brcms_ucode_init_buf(wl, (void **)&ucode->d11n0bsinitvals16, | ||||||
|  | 				       D11N0BSINITVALS16); | ||||||
|  | 	rc = rc < 0 ? | ||||||
|  | 	     rc : brcms_ucode_init_buf(wl, (void **)&ucode->d11n0initvals16, | ||||||
|  | 				       D11N0INITVALS16); | ||||||
|  | 	rc = rc < 0 ? | ||||||
|  | 	     rc : brcms_ucode_init_buf(wl, (void **)&ucode->bcm43xx_16_mimo, | ||||||
|  | 				       D11UCODE_OVERSIGHT16_MIMO); | ||||||
|  | 	rc = rc < 0 ? | ||||||
|  | 	     rc : brcms_ucode_init_uint(wl, &ucode->bcm43xx_16_mimosz, | ||||||
|  | 					D11UCODE_OVERSIGHT16_MIMOSZ); | ||||||
|  | 	rc = rc < 0 ? | ||||||
|  | 	     rc : brcms_ucode_init_buf(wl, (void **)&ucode->bcm43xx_24_lcn, | ||||||
|  | 				       D11UCODE_OVERSIGHT24_LCN); | ||||||
|  | 	rc = rc < 0 ? | ||||||
|  | 	     rc : brcms_ucode_init_uint(wl, &ucode->bcm43xx_24_lcnsz, | ||||||
|  | 					D11UCODE_OVERSIGHT24_LCNSZ); | ||||||
|  | 	rc = rc < 0 ? | ||||||
|  | 	     rc : brcms_ucode_init_buf(wl, (void **)&ucode->bcm43xx_bommajor, | ||||||
|  | 				       D11UCODE_OVERSIGHT_BOMMAJOR); | ||||||
|  | 	rc = rc < 0 ? | ||||||
|  | 	     rc : brcms_ucode_init_buf(wl, (void **)&ucode->bcm43xx_bomminor, | ||||||
|  | 				       D11UCODE_OVERSIGHT_BOMMINOR); | ||||||
|  | 	return rc; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void brcms_ucode_data_free(struct brcms_ucode *ucode) | ||||||
|  | { | ||||||
|  | 	brcms_ucode_free_buf((void *)ucode->d11lcn0bsinitvals24); | ||||||
|  | 	brcms_ucode_free_buf((void *)ucode->d11lcn0initvals24); | ||||||
|  | 	brcms_ucode_free_buf((void *)ucode->d11lcn1bsinitvals24); | ||||||
|  | 	brcms_ucode_free_buf((void *)ucode->d11lcn1initvals24); | ||||||
|  | 	brcms_ucode_free_buf((void *)ucode->d11lcn2bsinitvals24); | ||||||
|  | 	brcms_ucode_free_buf((void *)ucode->d11lcn2initvals24); | ||||||
|  | 	brcms_ucode_free_buf((void *)ucode->d11n0absinitvals16); | ||||||
|  | 	brcms_ucode_free_buf((void *)ucode->d11n0bsinitvals16); | ||||||
|  | 	brcms_ucode_free_buf((void *)ucode->d11n0initvals16); | ||||||
|  | 	brcms_ucode_free_buf((void *)ucode->bcm43xx_16_mimo); | ||||||
|  | 	brcms_ucode_free_buf((void *)ucode->bcm43xx_24_lcn); | ||||||
|  | 	brcms_ucode_free_buf((void *)ucode->bcm43xx_bommajor); | ||||||
|  | 	brcms_ucode_free_buf((void *)ucode->bcm43xx_bomminor); | ||||||
|  | } | ||||||
							
								
								
									
										58
									
								
								drivers/net/wireless/brcm80211/brcmsmac/ucode_loader.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								drivers/net/wireless/brcm80211/brcmsmac/ucode_loader.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,58 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | #ifndef	_BRCM_UCODE_H_ | ||||||
|  | #define	_BRCM_UCODE_H_ | ||||||
|  | 
 | ||||||
|  | #include "types.h"		/* forward structure declarations */ | ||||||
|  | 
 | ||||||
|  | #define MIN_FW_SIZE 40000	/* minimum firmware file size in bytes */ | ||||||
|  | #define MAX_FW_SIZE 150000 | ||||||
|  | 
 | ||||||
|  | #define UCODE_LOADER_API_VER 0 | ||||||
|  | 
 | ||||||
|  | struct d11init; | ||||||
|  | 
 | ||||||
|  | struct brcms_ucode { | ||||||
|  | 	struct d11init *d11lcn0bsinitvals24; | ||||||
|  | 	struct d11init *d11lcn0initvals24; | ||||||
|  | 	struct d11init *d11lcn1bsinitvals24; | ||||||
|  | 	struct d11init *d11lcn1initvals24; | ||||||
|  | 	struct d11init *d11lcn2bsinitvals24; | ||||||
|  | 	struct d11init *d11lcn2initvals24; | ||||||
|  | 	struct d11init *d11n0absinitvals16; | ||||||
|  | 	struct d11init *d11n0bsinitvals16; | ||||||
|  | 	struct d11init *d11n0initvals16; | ||||||
|  | 	__le32 *bcm43xx_16_mimo; | ||||||
|  | 	size_t bcm43xx_16_mimosz; | ||||||
|  | 	__le32 *bcm43xx_24_lcn; | ||||||
|  | 	size_t bcm43xx_24_lcnsz; | ||||||
|  | 	u32 *bcm43xx_bommajor; | ||||||
|  | 	u32 *bcm43xx_bomminor; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | extern int | ||||||
|  | brcms_ucode_data_init(struct brcms_info *wl, struct brcms_ucode *ucode); | ||||||
|  | 
 | ||||||
|  | extern void brcms_ucode_data_free(struct brcms_ucode *ucode); | ||||||
|  | 
 | ||||||
|  | extern int brcms_ucode_init_buf(struct brcms_info *wl, void **pbuf, | ||||||
|  | 				unsigned int idx); | ||||||
|  | extern int brcms_ucode_init_uint(struct brcms_info *wl, size_t *n_bytes, | ||||||
|  | 				 unsigned int idx); | ||||||
|  | extern void brcms_ucode_free_buf(void *); | ||||||
|  | extern int  brcms_check_firmwares(struct brcms_info *wl); | ||||||
|  | 
 | ||||||
|  | #endif	/* _BRCM_UCODE_H_ */ | ||||||
							
								
								
									
										29
									
								
								drivers/net/wireless/brcm80211/brcmutil/Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								drivers/net/wireless/brcm80211/brcmutil/Makefile
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,29 @@ | |||||||
|  | #
 | ||||||
|  | # Makefile fragment for Broadcom 802.11n Networking Device Driver Utilities
 | ||||||
|  | #
 | ||||||
|  | # Copyright (c) 2011 Broadcom Corporation
 | ||||||
|  | #
 | ||||||
|  | # 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.
 | ||||||
|  | 
 | ||||||
|  | ccflags-y :=				\
 | ||||||
|  | 	-Idrivers/net/wireless/brcm80211/brcmutil \
 | ||||||
|  | 	-Idrivers/net/wireless/brcm80211/include | ||||||
|  | 
 | ||||||
|  | BRCMUTIL_OFILES := \
 | ||||||
|  | 	utils.o \
 | ||||||
|  | 	wifi.o | ||||||
|  | 
 | ||||||
|  | MODULEPFX := brcmutil | ||||||
|  | 
 | ||||||
|  | obj-$(CONFIG_BRCMUTIL)	+= $(MODULEPFX).o | ||||||
|  | $(MODULEPFX)-objs	= $(BRCMUTIL_OFILES) | ||||||
							
								
								
									
										600
									
								
								drivers/net/wireless/brcm80211/brcmutil/utils.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										600
									
								
								drivers/net/wireless/brcm80211/brcmutil/utils.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,600 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <linux/netdevice.h> | ||||||
|  | #include <brcmu_utils.h> | ||||||
|  | 
 | ||||||
|  | MODULE_AUTHOR("Broadcom Corporation"); | ||||||
|  | MODULE_DESCRIPTION("Broadcom 802.11n wireless LAN driver utilities."); | ||||||
|  | MODULE_SUPPORTED_DEVICE("Broadcom 802.11n WLAN cards"); | ||||||
|  | MODULE_LICENSE("Dual BSD/GPL"); | ||||||
|  | 
 | ||||||
|  | struct sk_buff *brcmu_pkt_buf_get_skb(uint len) | ||||||
|  | { | ||||||
|  | 	struct sk_buff *skb; | ||||||
|  | 
 | ||||||
|  | 	skb = dev_alloc_skb(len); | ||||||
|  | 	if (skb) { | ||||||
|  | 		skb_put(skb, len); | ||||||
|  | 		skb->priority = 0; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return skb; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_pkt_buf_get_skb); | ||||||
|  | 
 | ||||||
|  | /* Free the driver packet. Free the tag if present */ | ||||||
|  | void brcmu_pkt_buf_free_skb(struct sk_buff *skb) | ||||||
|  | { | ||||||
|  | 	struct sk_buff *nskb; | ||||||
|  | 	int nest = 0; | ||||||
|  | 
 | ||||||
|  | 	/* perversion: we use skb->next to chain multi-skb packets */ | ||||||
|  | 	while (skb) { | ||||||
|  | 		nskb = skb->next; | ||||||
|  | 		skb->next = NULL; | ||||||
|  | 
 | ||||||
|  | 		if (skb->destructor) | ||||||
|  | 			/* cannot kfree_skb() on hard IRQ (net/core/skbuff.c) if
 | ||||||
|  | 			 * destructor exists | ||||||
|  | 			 */ | ||||||
|  | 			dev_kfree_skb_any(skb); | ||||||
|  | 		else | ||||||
|  | 			/* can free immediately (even in_irq()) if destructor
 | ||||||
|  | 			 * does not exist | ||||||
|  | 			 */ | ||||||
|  | 			dev_kfree_skb(skb); | ||||||
|  | 
 | ||||||
|  | 		nest++; | ||||||
|  | 		skb = nskb; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_pkt_buf_free_skb); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /* copy a buffer into a pkt buffer chain */ | ||||||
|  | uint brcmu_pktfrombuf(struct sk_buff *p, uint offset, int len, | ||||||
|  | 		unsigned char *buf) | ||||||
|  | { | ||||||
|  | 	uint n, ret = 0; | ||||||
|  | 
 | ||||||
|  | 	/* skip 'offset' bytes */ | ||||||
|  | 	for (; p && offset; p = p->next) { | ||||||
|  | 		if (offset < (uint) (p->len)) | ||||||
|  | 			break; | ||||||
|  | 		offset -= p->len; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (!p) | ||||||
|  | 		return 0; | ||||||
|  | 
 | ||||||
|  | 	/* copy the data */ | ||||||
|  | 	for (; p && len; p = p->next) { | ||||||
|  | 		n = min((uint) (p->len) - offset, (uint) len); | ||||||
|  | 		memcpy(p->data + offset, buf, n); | ||||||
|  | 		buf += n; | ||||||
|  | 		len -= n; | ||||||
|  | 		ret += n; | ||||||
|  | 		offset = 0; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_pktfrombuf); | ||||||
|  | 
 | ||||||
|  | /* return total length of buffer chain */ | ||||||
|  | uint brcmu_pkttotlen(struct sk_buff *p) | ||||||
|  | { | ||||||
|  | 	uint total; | ||||||
|  | 
 | ||||||
|  | 	total = 0; | ||||||
|  | 	for (; p; p = p->next) | ||||||
|  | 		total += p->len; | ||||||
|  | 	return total; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_pkttotlen); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * osl multiple-precedence packet queue | ||||||
|  |  * hi_prec is always >= the number of the highest non-empty precedence | ||||||
|  |  */ | ||||||
|  | struct sk_buff *brcmu_pktq_penq(struct pktq *pq, int prec, | ||||||
|  | 				      struct sk_buff *p) | ||||||
|  | { | ||||||
|  | 	struct pktq_prec *q; | ||||||
|  | 
 | ||||||
|  | 	if (pktq_full(pq) || pktq_pfull(pq, prec)) | ||||||
|  | 		return NULL; | ||||||
|  | 
 | ||||||
|  | 	q = &pq->q[prec]; | ||||||
|  | 
 | ||||||
|  | 	if (q->head) | ||||||
|  | 		q->tail->prev = p; | ||||||
|  | 	else | ||||||
|  | 		q->head = p; | ||||||
|  | 
 | ||||||
|  | 	q->tail = p; | ||||||
|  | 	q->len++; | ||||||
|  | 
 | ||||||
|  | 	pq->len++; | ||||||
|  | 
 | ||||||
|  | 	if (pq->hi_prec < prec) | ||||||
|  | 		pq->hi_prec = (u8) prec; | ||||||
|  | 
 | ||||||
|  | 	return p; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_pktq_penq); | ||||||
|  | 
 | ||||||
|  | struct sk_buff *brcmu_pktq_penq_head(struct pktq *pq, int prec, | ||||||
|  | 					   struct sk_buff *p) | ||||||
|  | { | ||||||
|  | 	struct pktq_prec *q; | ||||||
|  | 
 | ||||||
|  | 	if (pktq_full(pq) || pktq_pfull(pq, prec)) | ||||||
|  | 		return NULL; | ||||||
|  | 
 | ||||||
|  | 	q = &pq->q[prec]; | ||||||
|  | 
 | ||||||
|  | 	if (q->head == NULL) | ||||||
|  | 		q->tail = p; | ||||||
|  | 
 | ||||||
|  | 	p->prev = q->head; | ||||||
|  | 	q->head = p; | ||||||
|  | 	q->len++; | ||||||
|  | 
 | ||||||
|  | 	pq->len++; | ||||||
|  | 
 | ||||||
|  | 	if (pq->hi_prec < prec) | ||||||
|  | 		pq->hi_prec = (u8) prec; | ||||||
|  | 
 | ||||||
|  | 	return p; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_pktq_penq_head); | ||||||
|  | 
 | ||||||
|  | struct sk_buff *brcmu_pktq_pdeq(struct pktq *pq, int prec) | ||||||
|  | { | ||||||
|  | 	struct pktq_prec *q; | ||||||
|  | 	struct sk_buff *p; | ||||||
|  | 
 | ||||||
|  | 	q = &pq->q[prec]; | ||||||
|  | 
 | ||||||
|  | 	p = q->head; | ||||||
|  | 	if (p == NULL) | ||||||
|  | 		return NULL; | ||||||
|  | 
 | ||||||
|  | 	q->head = p->prev; | ||||||
|  | 	if (q->head == NULL) | ||||||
|  | 		q->tail = NULL; | ||||||
|  | 
 | ||||||
|  | 	q->len--; | ||||||
|  | 
 | ||||||
|  | 	pq->len--; | ||||||
|  | 
 | ||||||
|  | 	p->prev = NULL; | ||||||
|  | 
 | ||||||
|  | 	return p; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_pktq_pdeq); | ||||||
|  | 
 | ||||||
|  | struct sk_buff *brcmu_pktq_pdeq_tail(struct pktq *pq, int prec) | ||||||
|  | { | ||||||
|  | 	struct pktq_prec *q; | ||||||
|  | 	struct sk_buff *p, *prev; | ||||||
|  | 
 | ||||||
|  | 	q = &pq->q[prec]; | ||||||
|  | 
 | ||||||
|  | 	p = q->head; | ||||||
|  | 	if (p == NULL) | ||||||
|  | 		return NULL; | ||||||
|  | 
 | ||||||
|  | 	for (prev = NULL; p != q->tail; p = p->prev) | ||||||
|  | 		prev = p; | ||||||
|  | 
 | ||||||
|  | 	if (prev) | ||||||
|  | 		prev->prev = NULL; | ||||||
|  | 	else | ||||||
|  | 		q->head = NULL; | ||||||
|  | 
 | ||||||
|  | 	q->tail = prev; | ||||||
|  | 	q->len--; | ||||||
|  | 
 | ||||||
|  | 	pq->len--; | ||||||
|  | 
 | ||||||
|  | 	return p; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_pktq_pdeq_tail); | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | brcmu_pktq_pflush(struct pktq *pq, int prec, bool dir, | ||||||
|  | 		  bool (*fn)(struct sk_buff *, void *), void *arg) | ||||||
|  | { | ||||||
|  | 	struct pktq_prec *q; | ||||||
|  | 	struct sk_buff *p, *prev = NULL; | ||||||
|  | 
 | ||||||
|  | 	q = &pq->q[prec]; | ||||||
|  | 	p = q->head; | ||||||
|  | 	while (p) { | ||||||
|  | 		if (fn == NULL || (*fn) (p, arg)) { | ||||||
|  | 			bool head = (p == q->head); | ||||||
|  | 			if (head) | ||||||
|  | 				q->head = p->prev; | ||||||
|  | 			else | ||||||
|  | 				prev->prev = p->prev; | ||||||
|  | 			p->prev = NULL; | ||||||
|  | 			brcmu_pkt_buf_free_skb(p); | ||||||
|  | 			q->len--; | ||||||
|  | 			pq->len--; | ||||||
|  | 			p = (head ? q->head : prev->prev); | ||||||
|  | 		} else { | ||||||
|  | 			prev = p; | ||||||
|  | 			p = p->prev; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (q->head == NULL) | ||||||
|  | 		q->tail = NULL; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_pktq_pflush); | ||||||
|  | 
 | ||||||
|  | void brcmu_pktq_flush(struct pktq *pq, bool dir, | ||||||
|  | 		      bool (*fn)(struct sk_buff *, void *), void *arg) | ||||||
|  | { | ||||||
|  | 	int prec; | ||||||
|  | 	for (prec = 0; prec < pq->num_prec; prec++) | ||||||
|  | 		brcmu_pktq_pflush(pq, prec, dir, fn, arg); | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_pktq_flush); | ||||||
|  | 
 | ||||||
|  | void brcmu_pktq_init(struct pktq *pq, int num_prec, int max_len) | ||||||
|  | { | ||||||
|  | 	int prec; | ||||||
|  | 
 | ||||||
|  | 	/* pq is variable size; only zero out what's requested */ | ||||||
|  | 	memset(pq, 0, | ||||||
|  | 	      offsetof(struct pktq, q) + (sizeof(struct pktq_prec) * num_prec)); | ||||||
|  | 
 | ||||||
|  | 	pq->num_prec = (u16) num_prec; | ||||||
|  | 
 | ||||||
|  | 	pq->max = (u16) max_len; | ||||||
|  | 
 | ||||||
|  | 	for (prec = 0; prec < num_prec; prec++) | ||||||
|  | 		pq->q[prec].max = pq->max; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_pktq_init); | ||||||
|  | 
 | ||||||
|  | struct sk_buff *brcmu_pktq_peek_tail(struct pktq *pq, int *prec_out) | ||||||
|  | { | ||||||
|  | 	int prec; | ||||||
|  | 
 | ||||||
|  | 	if (pq->len == 0) | ||||||
|  | 		return NULL; | ||||||
|  | 
 | ||||||
|  | 	for (prec = 0; prec < pq->hi_prec; prec++) | ||||||
|  | 		if (pq->q[prec].head) | ||||||
|  | 			break; | ||||||
|  | 
 | ||||||
|  | 	if (prec_out) | ||||||
|  | 		*prec_out = prec; | ||||||
|  | 
 | ||||||
|  | 	return pq->q[prec].tail; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_pktq_peek_tail); | ||||||
|  | 
 | ||||||
|  | /* Return sum of lengths of a specific set of precedences */ | ||||||
|  | int brcmu_pktq_mlen(struct pktq *pq, uint prec_bmp) | ||||||
|  | { | ||||||
|  | 	int prec, len; | ||||||
|  | 
 | ||||||
|  | 	len = 0; | ||||||
|  | 
 | ||||||
|  | 	for (prec = 0; prec <= pq->hi_prec; prec++) | ||||||
|  | 		if (prec_bmp & (1 << prec)) | ||||||
|  | 			len += pq->q[prec].len; | ||||||
|  | 
 | ||||||
|  | 	return len; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_pktq_mlen); | ||||||
|  | 
 | ||||||
|  | /* Priority dequeue from a specific set of precedences */ | ||||||
|  | struct sk_buff *brcmu_pktq_mdeq(struct pktq *pq, uint prec_bmp, | ||||||
|  | 				      int *prec_out) | ||||||
|  | { | ||||||
|  | 	struct pktq_prec *q; | ||||||
|  | 	struct sk_buff *p; | ||||||
|  | 	int prec; | ||||||
|  | 
 | ||||||
|  | 	if (pq->len == 0) | ||||||
|  | 		return NULL; | ||||||
|  | 
 | ||||||
|  | 	while ((prec = pq->hi_prec) > 0 && pq->q[prec].head == NULL) | ||||||
|  | 		pq->hi_prec--; | ||||||
|  | 
 | ||||||
|  | 	while ((prec_bmp & (1 << prec)) == 0 || pq->q[prec].head == NULL) | ||||||
|  | 		if (prec-- == 0) | ||||||
|  | 			return NULL; | ||||||
|  | 
 | ||||||
|  | 	q = &pq->q[prec]; | ||||||
|  | 
 | ||||||
|  | 	p = q->head; | ||||||
|  | 	if (p == NULL) | ||||||
|  | 		return NULL; | ||||||
|  | 
 | ||||||
|  | 	q->head = p->prev; | ||||||
|  | 	if (q->head == NULL) | ||||||
|  | 		q->tail = NULL; | ||||||
|  | 
 | ||||||
|  | 	q->len--; | ||||||
|  | 
 | ||||||
|  | 	if (prec_out) | ||||||
|  | 		*prec_out = prec; | ||||||
|  | 
 | ||||||
|  | 	pq->len--; | ||||||
|  | 
 | ||||||
|  | 	p->prev = NULL; | ||||||
|  | 
 | ||||||
|  | 	return p; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_pktq_mdeq); | ||||||
|  | 
 | ||||||
|  | #if defined(BCMDBG) | ||||||
|  | /* pretty hex print a pkt buffer chain */ | ||||||
|  | void brcmu_prpkt(const char *msg, struct sk_buff *p0) | ||||||
|  | { | ||||||
|  | 	struct sk_buff *p; | ||||||
|  | 
 | ||||||
|  | 	if (msg && (msg[0] != '\0')) | ||||||
|  | 		printk(KERN_DEBUG "%s:\n", msg); | ||||||
|  | 
 | ||||||
|  | 	for (p = p0; p; p = p->next) | ||||||
|  | 		print_hex_dump_bytes("", DUMP_PREFIX_OFFSET, p->data, p->len); | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_prpkt); | ||||||
|  | #endif				/* defined(BCMDBG) */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Traverse a string of 1-byte tag/1-byte length/variable-length value | ||||||
|  |  * triples, returning a pointer to the substring whose first element | ||||||
|  |  * matches tag | ||||||
|  |  */ | ||||||
|  | struct brcmu_tlv *brcmu_parse_tlvs(void *buf, int buflen, uint key) | ||||||
|  | { | ||||||
|  | 	struct brcmu_tlv *elt; | ||||||
|  | 	int totlen; | ||||||
|  | 
 | ||||||
|  | 	elt = (struct brcmu_tlv *) buf; | ||||||
|  | 	totlen = buflen; | ||||||
|  | 
 | ||||||
|  | 	/* find tagged parameter */ | ||||||
|  | 	while (totlen >= 2) { | ||||||
|  | 		int len = elt->len; | ||||||
|  | 
 | ||||||
|  | 		/* validate remaining totlen */ | ||||||
|  | 		if ((elt->id == key) && (totlen >= (len + 2))) | ||||||
|  | 			return elt; | ||||||
|  | 
 | ||||||
|  | 		elt = (struct brcmu_tlv *) ((u8 *) elt + (len + 2)); | ||||||
|  | 		totlen -= (len + 2); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return NULL; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_parse_tlvs); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #if defined(BCMDBG) | ||||||
|  | int | ||||||
|  | brcmu_format_flags(const struct brcmu_bit_desc *bd, u32 flags, char *buf, | ||||||
|  | 		   int len) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	char *p = buf; | ||||||
|  | 	char hexstr[16]; | ||||||
|  | 	int slen = 0, nlen = 0; | ||||||
|  | 	u32 bit; | ||||||
|  | 	const char *name; | ||||||
|  | 
 | ||||||
|  | 	if (len < 2 || !buf) | ||||||
|  | 		return 0; | ||||||
|  | 
 | ||||||
|  | 	buf[0] = '\0'; | ||||||
|  | 
 | ||||||
|  | 	for (i = 0; flags != 0; i++) { | ||||||
|  | 		bit = bd[i].bit; | ||||||
|  | 		name = bd[i].name; | ||||||
|  | 		if (bit == 0 && flags != 0) { | ||||||
|  | 			/* print any unnamed bits */ | ||||||
|  | 			snprintf(hexstr, 16, "0x%X", flags); | ||||||
|  | 			name = hexstr; | ||||||
|  | 			flags = 0;	/* exit loop */ | ||||||
|  | 		} else if ((flags & bit) == 0) | ||||||
|  | 			continue; | ||||||
|  | 		flags &= ~bit; | ||||||
|  | 		nlen = strlen(name); | ||||||
|  | 		slen += nlen; | ||||||
|  | 		/* count btwn flag space */ | ||||||
|  | 		if (flags != 0) | ||||||
|  | 			slen += 1; | ||||||
|  | 		/* need NULL char as well */ | ||||||
|  | 		if (len <= slen) | ||||||
|  | 			break; | ||||||
|  | 		/* copy NULL char but don't count it */ | ||||||
|  | 		strncpy(p, name, nlen + 1); | ||||||
|  | 		p += nlen; | ||||||
|  | 		/* copy btwn flag space and NULL char */ | ||||||
|  | 		if (flags != 0) | ||||||
|  | 			p += snprintf(p, 2, " "); | ||||||
|  | 		len -= slen; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* indicate the str was too short */ | ||||||
|  | 	if (flags != 0) { | ||||||
|  | 		if (len < 2) | ||||||
|  | 			p -= 2 - len;	/* overwrite last char */ | ||||||
|  | 		p += snprintf(p, 2, ">"); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return (int)(p - buf); | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_format_flags); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * print bytes formatted as hex to a string. return the resulting | ||||||
|  |  * string length | ||||||
|  |  */ | ||||||
|  | int brcmu_format_hex(char *str, const void *bytes, int len) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	char *p = str; | ||||||
|  | 	const u8 *src = (const u8 *)bytes; | ||||||
|  | 
 | ||||||
|  | 	for (i = 0; i < len; i++) { | ||||||
|  | 		p += snprintf(p, 3, "%02X", *src); | ||||||
|  | 		src++; | ||||||
|  | 	} | ||||||
|  | 	return (int)(p - str); | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_format_hex); | ||||||
|  | #endif				/* defined(BCMDBG) */ | ||||||
|  | 
 | ||||||
|  | char *brcmu_chipname(uint chipid, char *buf, uint len) | ||||||
|  | { | ||||||
|  | 	const char *fmt; | ||||||
|  | 
 | ||||||
|  | 	fmt = ((chipid > 0xa000) || (chipid < 0x4000)) ? "%d" : "%x"; | ||||||
|  | 	snprintf(buf, len, fmt, chipid); | ||||||
|  | 	return buf; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_chipname); | ||||||
|  | 
 | ||||||
|  | uint brcmu_mkiovar(char *name, char *data, uint datalen, char *buf, uint buflen) | ||||||
|  | { | ||||||
|  | 	uint len; | ||||||
|  | 
 | ||||||
|  | 	len = strlen(name) + 1; | ||||||
|  | 
 | ||||||
|  | 	if ((len + datalen) > buflen) | ||||||
|  | 		return 0; | ||||||
|  | 
 | ||||||
|  | 	strncpy(buf, name, buflen); | ||||||
|  | 
 | ||||||
|  | 	/* append data onto the end of the name string */ | ||||||
|  | 	memcpy(&buf[len], data, datalen); | ||||||
|  | 	len += datalen; | ||||||
|  | 
 | ||||||
|  | 	return len; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_mkiovar); | ||||||
|  | 
 | ||||||
|  | /* Quarter dBm units to mW
 | ||||||
|  |  * Table starts at QDBM_OFFSET, so the first entry is mW for qdBm=153 | ||||||
|  |  * Table is offset so the last entry is largest mW value that fits in | ||||||
|  |  * a u16. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #define QDBM_OFFSET 153		/* Offset for first entry */ | ||||||
|  | #define QDBM_TABLE_LEN 40	/* Table size */ | ||||||
|  | 
 | ||||||
|  | /* Smallest mW value that will round up to the first table entry, QDBM_OFFSET.
 | ||||||
|  |  * Value is ( mW(QDBM_OFFSET - 1) + mW(QDBM_OFFSET) ) / 2 | ||||||
|  |  */ | ||||||
|  | #define QDBM_TABLE_LOW_BOUND 6493	/* Low bound */ | ||||||
|  | 
 | ||||||
|  | /* Largest mW value that will round down to the last table entry,
 | ||||||
|  |  * QDBM_OFFSET + QDBM_TABLE_LEN-1. | ||||||
|  |  * Value is ( mW(QDBM_OFFSET + QDBM_TABLE_LEN - 1) + | ||||||
|  |  * mW(QDBM_OFFSET + QDBM_TABLE_LEN) ) / 2. | ||||||
|  |  */ | ||||||
|  | #define QDBM_TABLE_HIGH_BOUND 64938	/* High bound */ | ||||||
|  | 
 | ||||||
|  | static const u16 nqdBm_to_mW_map[QDBM_TABLE_LEN] = { | ||||||
|  | /* qdBm:	+0	+1	+2	+3	+4	+5	+6	+7 */ | ||||||
|  | /* 153: */ 6683, 7079, 7499, 7943, 8414, 8913, 9441, 10000, | ||||||
|  | /* 161: */ 10593, 11220, 11885, 12589, 13335, 14125, 14962, 15849, | ||||||
|  | /* 169: */ 16788, 17783, 18836, 19953, 21135, 22387, 23714, 25119, | ||||||
|  | /* 177: */ 26607, 28184, 29854, 31623, 33497, 35481, 37584, 39811, | ||||||
|  | /* 185: */ 42170, 44668, 47315, 50119, 53088, 56234, 59566, 63096 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | u16 brcmu_qdbm_to_mw(u8 qdbm) | ||||||
|  | { | ||||||
|  | 	uint factor = 1; | ||||||
|  | 	int idx = qdbm - QDBM_OFFSET; | ||||||
|  | 
 | ||||||
|  | 	if (idx >= QDBM_TABLE_LEN) | ||||||
|  | 		/* clamp to max u16 mW value */ | ||||||
|  | 		return 0xFFFF; | ||||||
|  | 
 | ||||||
|  | 	/* scale the qdBm index up to the range of the table 0-40
 | ||||||
|  | 	 * where an offset of 40 qdBm equals a factor of 10 mW. | ||||||
|  | 	 */ | ||||||
|  | 	while (idx < 0) { | ||||||
|  | 		idx += 40; | ||||||
|  | 		factor *= 10; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* return the mW value scaled down to the correct factor of 10,
 | ||||||
|  | 	 * adding in factor/2 to get proper rounding. | ||||||
|  | 	 */ | ||||||
|  | 	return (nqdBm_to_mW_map[idx] + factor / 2) / factor; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_qdbm_to_mw); | ||||||
|  | 
 | ||||||
|  | u8 brcmu_mw_to_qdbm(u16 mw) | ||||||
|  | { | ||||||
|  | 	u8 qdbm; | ||||||
|  | 	int offset; | ||||||
|  | 	uint mw_uint = mw; | ||||||
|  | 	uint boundary; | ||||||
|  | 
 | ||||||
|  | 	/* handle boundary case */ | ||||||
|  | 	if (mw_uint <= 1) | ||||||
|  | 		return 0; | ||||||
|  | 
 | ||||||
|  | 	offset = QDBM_OFFSET; | ||||||
|  | 
 | ||||||
|  | 	/* move mw into the range of the table */ | ||||||
|  | 	while (mw_uint < QDBM_TABLE_LOW_BOUND) { | ||||||
|  | 		mw_uint *= 10; | ||||||
|  | 		offset -= 40; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for (qdbm = 0; qdbm < QDBM_TABLE_LEN - 1; qdbm++) { | ||||||
|  | 		boundary = nqdBm_to_mW_map[qdbm] + (nqdBm_to_mW_map[qdbm + 1] - | ||||||
|  | 						    nqdBm_to_mW_map[qdbm]) / 2; | ||||||
|  | 		if (mw_uint < boundary) | ||||||
|  | 			break; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	qdbm += (u8) offset; | ||||||
|  | 
 | ||||||
|  | 	return qdbm; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_mw_to_qdbm); | ||||||
|  | 
 | ||||||
|  | uint brcmu_bitcount(u8 *bitmap, uint length) | ||||||
|  | { | ||||||
|  | 	uint bitcount = 0, i; | ||||||
|  | 	u8 tmp; | ||||||
|  | 	for (i = 0; i < length; i++) { | ||||||
|  | 		tmp = bitmap[i]; | ||||||
|  | 		while (tmp) { | ||||||
|  | 			bitcount++; | ||||||
|  | 			tmp &= (tmp - 1); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return bitcount; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_bitcount); | ||||||
							
								
								
									
										136
									
								
								drivers/net/wireless/brcm80211/brcmutil/wifi.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										136
									
								
								drivers/net/wireless/brcm80211/brcmutil/wifi.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,136 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | #include <brcmu_wifi.h> | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Verify the chanspec is using a legal set of parameters, i.e. that the | ||||||
|  |  * chanspec specified a band, bw, ctl_sb and channel and that the | ||||||
|  |  * combination could be legal given any set of circumstances. | ||||||
|  |  * RETURNS: true is the chanspec is malformed, false if it looks good. | ||||||
|  |  */ | ||||||
|  | bool brcmu_chspec_malformed(u16 chanspec) | ||||||
|  | { | ||||||
|  | 	/* must be 2G or 5G band */ | ||||||
|  | 	if (!CHSPEC_IS5G(chanspec) && !CHSPEC_IS2G(chanspec)) | ||||||
|  | 		return true; | ||||||
|  | 	/* must be 20 or 40 bandwidth */ | ||||||
|  | 	if (!CHSPEC_IS40(chanspec) && !CHSPEC_IS20(chanspec)) | ||||||
|  | 		return true; | ||||||
|  | 
 | ||||||
|  | 	/* 20MHZ b/w must have no ctl sb, 40 must have a ctl sb */ | ||||||
|  | 	if (CHSPEC_IS20(chanspec)) { | ||||||
|  | 		if (!CHSPEC_SB_NONE(chanspec)) | ||||||
|  | 			return true; | ||||||
|  | 	} else if (!CHSPEC_SB_UPPER(chanspec) && !CHSPEC_SB_LOWER(chanspec)) { | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return false; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_chspec_malformed); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * This function returns the channel number that control traffic is being sent | ||||||
|  |  * on, for legacy channels this is just the channel number, for 40MHZ channels | ||||||
|  |  * it is the upper or lower 20MHZ sideband depending on the chanspec selected. | ||||||
|  |  */ | ||||||
|  | u8 brcmu_chspec_ctlchan(u16 chspec) | ||||||
|  | { | ||||||
|  | 	u8 ctl_chan; | ||||||
|  | 
 | ||||||
|  | 	/* Is there a sideband ? */ | ||||||
|  | 	if (CHSPEC_CTL_SB(chspec) == WL_CHANSPEC_CTL_SB_NONE) { | ||||||
|  | 		return CHSPEC_CHANNEL(chspec); | ||||||
|  | 	} else { | ||||||
|  | 		/*
 | ||||||
|  | 		 * we only support 40MHZ with sidebands. chanspec channel holds | ||||||
|  | 		 * the centre frequency, use that and the side band information | ||||||
|  | 		 * to reconstruct the control channel number | ||||||
|  | 		 */ | ||||||
|  | 		if (CHSPEC_CTL_SB(chspec) == WL_CHANSPEC_CTL_SB_UPPER) | ||||||
|  | 			/*
 | ||||||
|  | 			 * control chan is the upper 20 MHZ SB of the | ||||||
|  | 			 * 40MHZ channel | ||||||
|  | 			 */ | ||||||
|  | 			ctl_chan = upper_20_sb(CHSPEC_CHANNEL(chspec)); | ||||||
|  | 		else | ||||||
|  | 			/*
 | ||||||
|  | 			 * control chan is the lower 20 MHZ SB of the | ||||||
|  | 			 * 40MHZ channel | ||||||
|  | 			 */ | ||||||
|  | 			ctl_chan = lower_20_sb(CHSPEC_CHANNEL(chspec)); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return ctl_chan; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_chspec_ctlchan); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Return the channel number for a given frequency and base frequency. | ||||||
|  |  * The returned channel number is relative to the given base frequency. | ||||||
|  |  * If the given base frequency is zero, a base frequency of 5 GHz is assumed for | ||||||
|  |  * frequencies from 5 - 6 GHz, and 2.407 GHz is assumed for 2.4 - 2.5 GHz. | ||||||
|  |  * | ||||||
|  |  * Frequency is specified in MHz. | ||||||
|  |  * The base frequency is specified as (start_factor * 500 kHz). | ||||||
|  |  * Constants WF_CHAN_FACTOR_2_4_G, WF_CHAN_FACTOR_5_G are defined for | ||||||
|  |  * 2.4 GHz and 5 GHz bands. | ||||||
|  |  * | ||||||
|  |  * The returned channel will be in the range [1, 14] in the 2.4 GHz band | ||||||
|  |  * and [0, 200] otherwise. | ||||||
|  |  * -1 is returned if the start_factor is WF_CHAN_FACTOR_2_4_G and the | ||||||
|  |  * frequency is not a 2.4 GHz channel, or if the frequency is not and even | ||||||
|  |  * multiple of 5 MHz from the base frequency to the base plus 1 GHz. | ||||||
|  |  * | ||||||
|  |  * Reference 802.11 REVma, section 17.3.8.3, and 802.11B section 18.4.6.2 | ||||||
|  |  */ | ||||||
|  | int brcmu_mhz2channel(uint freq, uint start_factor) | ||||||
|  | { | ||||||
|  | 	int ch = -1; | ||||||
|  | 	uint base; | ||||||
|  | 	int offset; | ||||||
|  | 
 | ||||||
|  | 	/* take the default channel start frequency */ | ||||||
|  | 	if (start_factor == 0) { | ||||||
|  | 		if (freq >= 2400 && freq <= 2500) | ||||||
|  | 			start_factor = WF_CHAN_FACTOR_2_4_G; | ||||||
|  | 		else if (freq >= 5000 && freq <= 6000) | ||||||
|  | 			start_factor = WF_CHAN_FACTOR_5_G; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (freq == 2484 && start_factor == WF_CHAN_FACTOR_2_4_G) | ||||||
|  | 		return 14; | ||||||
|  | 
 | ||||||
|  | 	base = start_factor / 2; | ||||||
|  | 
 | ||||||
|  | 	/* check that the frequency is in 1GHz range of the base */ | ||||||
|  | 	if ((freq < base) || (freq > base + 1000)) | ||||||
|  | 		return -1; | ||||||
|  | 
 | ||||||
|  | 	offset = freq - base; | ||||||
|  | 	ch = offset / 5; | ||||||
|  | 
 | ||||||
|  | 	/* check that frequency is a 5MHz multiple from the base */ | ||||||
|  | 	if (offset != (ch * 5)) | ||||||
|  | 		return -1; | ||||||
|  | 
 | ||||||
|  | 	/* restricted channel range check for 2.4G */ | ||||||
|  | 	if (start_factor == WF_CHAN_FACTOR_2_4_G && (ch < 1 || ch > 13)) | ||||||
|  | 		return -1; | ||||||
|  | 
 | ||||||
|  | 	return ch; | ||||||
|  | } | ||||||
|  | EXPORT_SYMBOL(brcmu_mhz2channel); | ||||||
							
								
								
									
										59
									
								
								drivers/net/wireless/brcm80211/include/brcm_hw_ids.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								drivers/net/wireless/brcm80211/include/brcm_hw_ids.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,59 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef	_BRCM_HW_IDS_H_ | ||||||
|  | #define	_BRCM_HW_IDS_H_ | ||||||
|  | 
 | ||||||
|  | #define	BCM4325_D11DUAL_ID	0x431b | ||||||
|  | #define	BCM4325_D11G_ID		0x431c | ||||||
|  | #define	BCM4325_D11A_ID		0x431d | ||||||
|  | 
 | ||||||
|  | #define BCM4329_D11N2G_ID	0x432f	/* 4329 802.11n 2.4G device */ | ||||||
|  | #define BCM4329_D11N5G_ID	0x4330	/* 4329 802.11n 5G device */ | ||||||
|  | #define BCM4329_D11NDUAL_ID	0x432e | ||||||
|  | 
 | ||||||
|  | #define BCM4319_D11N_ID		0x4337	/* 4319 802.11n dualband device */ | ||||||
|  | #define BCM4319_D11N2G_ID	0x4338	/* 4319 802.11n 2.4G device */ | ||||||
|  | #define BCM4319_D11N5G_ID	0x4339	/* 4319 802.11n 5G device */ | ||||||
|  | 
 | ||||||
|  | #define BCM43224_D11N_ID	0x4353	/* 43224 802.11n dualband device */ | ||||||
|  | #define BCM43224_D11N_ID_VEN1	0x0576	/* Vendor specific 43224 802.11n db */ | ||||||
|  | 
 | ||||||
|  | #define BCM43225_D11N2G_ID	0x4357	/* 43225 802.11n 2.4GHz device */ | ||||||
|  | 
 | ||||||
|  | #define BCM43236_D11N_ID	0x4346	/* 43236 802.11n dualband device */ | ||||||
|  | #define BCM43236_D11N2G_ID	0x4347	/* 43236 802.11n 2.4GHz device */ | ||||||
|  | 
 | ||||||
|  | #define BCM4313_D11N2G_ID	0x4727	/* 4313 802.11n 2.4G device */ | ||||||
|  | 
 | ||||||
|  | /* Chip IDs */ | ||||||
|  | #define BCM4313_CHIP_ID		0x4313	/* 4313 chip id */ | ||||||
|  | #define	BCM4319_CHIP_ID		0x4319	/* 4319 chip id */ | ||||||
|  | 
 | ||||||
|  | #define	BCM43224_CHIP_ID	43224	/* 43224 chipcommon chipid */ | ||||||
|  | #define	BCM43225_CHIP_ID	43225	/* 43225 chipcommon chipid */ | ||||||
|  | #define	BCM43421_CHIP_ID	43421	/* 43421 chipcommon chipid */ | ||||||
|  | #define	BCM43235_CHIP_ID	43235	/* 43235 chipcommon chipid */ | ||||||
|  | #define	BCM43236_CHIP_ID	43236	/* 43236 chipcommon chipid */ | ||||||
|  | #define	BCM43238_CHIP_ID	43238	/* 43238 chipcommon chipid */ | ||||||
|  | #define	BCM4329_CHIP_ID		0x4329	/* 4329 chipcommon chipid */ | ||||||
|  | #define	BCM4325_CHIP_ID		0x4325	/* 4325 chipcommon chipid */ | ||||||
|  | #define	BCM4331_CHIP_ID		0x4331	/* 4331 chipcommon chipid */ | ||||||
|  | #define BCM4336_CHIP_ID		0x4336	/* 4336 chipcommon chipid */ | ||||||
|  | #define BCM4330_CHIP_ID		0x4330	/* 4330 chipcommon chipid */ | ||||||
|  | #define BCM6362_CHIP_ID		0x6362	/* 6362 chipcommon chipid */ | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCM_HW_IDS_H_ */ | ||||||
							
								
								
									
										223
									
								
								drivers/net/wireless/brcm80211/include/brcmu_utils.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										223
									
								
								drivers/net/wireless/brcm80211/include/brcmu_utils.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,223 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef	_BRCMU_UTILS_H_ | ||||||
|  | #define	_BRCMU_UTILS_H_ | ||||||
|  | 
 | ||||||
|  | #include <linux/skbuff.h> | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Spin at most 'us' microseconds while 'exp' is true. | ||||||
|  |  * Caller should explicitly test 'exp' when this completes | ||||||
|  |  * and take appropriate error action if 'exp' is still true. | ||||||
|  |  */ | ||||||
|  | #define SPINWAIT(exp, us) { \ | ||||||
|  | 	uint countdown = (us) + 9; \ | ||||||
|  | 	while ((exp) && (countdown >= 10)) {\ | ||||||
|  | 		udelay(10); \ | ||||||
|  | 		countdown -= 10; \ | ||||||
|  | 	} \ | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* osl multi-precedence packet queue */ | ||||||
|  | #define PKTQ_LEN_DEFAULT        128	/* Max 128 packets */ | ||||||
|  | #define PKTQ_MAX_PREC           16	/* Maximum precedence levels */ | ||||||
|  | 
 | ||||||
|  | #define BCME_STRLEN		64	/* Max string length for BCM errors */ | ||||||
|  | 
 | ||||||
|  | /* the largest reasonable packet buffer driver uses for ethernet MTU in bytes */ | ||||||
|  | #define	PKTBUFSZ	2048 | ||||||
|  | 
 | ||||||
|  | #ifndef setbit | ||||||
|  | #ifndef NBBY			/* the BSD family defines NBBY */ | ||||||
|  | #define	NBBY	8		/* 8 bits per byte */ | ||||||
|  | #endif				/* #ifndef NBBY */ | ||||||
|  | #define	setbit(a, i)	(((u8 *)a)[(i)/NBBY] |= 1<<((i)%NBBY)) | ||||||
|  | #define	clrbit(a, i)	(((u8 *)a)[(i)/NBBY] &= ~(1<<((i)%NBBY))) | ||||||
|  | #define	isset(a, i)	(((const u8 *)a)[(i)/NBBY] & (1<<((i)%NBBY))) | ||||||
|  | #define	isclr(a, i)	((((const u8 *)a)[(i)/NBBY] & (1<<((i)%NBBY))) == 0) | ||||||
|  | #endif				/* setbit */ | ||||||
|  | 
 | ||||||
|  | #define	NBITS(type)	(sizeof(type) * 8) | ||||||
|  | #define NBITVAL(nbits)	(1 << (nbits)) | ||||||
|  | #define MAXBITVAL(nbits)	((1 << (nbits)) - 1) | ||||||
|  | #define	NBITMASK(nbits)	MAXBITVAL(nbits) | ||||||
|  | #define MAXNBVAL(nbyte)	MAXBITVAL((nbyte) * 8) | ||||||
|  | 
 | ||||||
|  | /* crc defines */ | ||||||
|  | #define CRC16_INIT_VALUE 0xffff	/* Initial CRC16 checksum value */ | ||||||
|  | #define CRC16_GOOD_VALUE 0xf0b8	/* Good final CRC16 checksum value */ | ||||||
|  | 
 | ||||||
|  | /* 18-bytes of Ethernet address buffer length */ | ||||||
|  | #define ETHER_ADDR_STR_LEN	18 | ||||||
|  | 
 | ||||||
|  | struct pktq_prec { | ||||||
|  | 	struct sk_buff *head;	/* first packet to dequeue */ | ||||||
|  | 	struct sk_buff *tail;	/* last packet to dequeue */ | ||||||
|  | 	u16 len;		/* number of queued packets */ | ||||||
|  | 	u16 max;		/* maximum number of queued packets */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* multi-priority pkt queue */ | ||||||
|  | struct pktq { | ||||||
|  | 	u16 num_prec;	/* number of precedences in use */ | ||||||
|  | 	u16 hi_prec;	/* rapid dequeue hint (>= highest non-empty prec) */ | ||||||
|  | 	u16 max;	/* total max packets */ | ||||||
|  | 	u16 len;	/* total number of packets */ | ||||||
|  | 	/*
 | ||||||
|  | 	 * q array must be last since # of elements can be either | ||||||
|  | 	 * PKTQ_MAX_PREC or 1 | ||||||
|  | 	 */ | ||||||
|  | 	struct pktq_prec q[PKTQ_MAX_PREC]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* operations on a specific precedence in packet queue */ | ||||||
|  | 
 | ||||||
|  | static inline int pktq_plen(struct pktq *pq, int prec) | ||||||
|  | { | ||||||
|  | 	return pq->q[prec].len; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline int pktq_pavail(struct pktq *pq, int prec) | ||||||
|  | { | ||||||
|  | 	return pq->q[prec].max - pq->q[prec].len; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline bool pktq_pfull(struct pktq *pq, int prec) | ||||||
|  | { | ||||||
|  | 	return pq->q[prec].len >= pq->q[prec].max; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline bool pktq_pempty(struct pktq *pq, int prec) | ||||||
|  | { | ||||||
|  | 	return pq->q[prec].len == 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline struct sk_buff *pktq_ppeek(struct pktq *pq, int prec) | ||||||
|  | { | ||||||
|  | 	return pq->q[prec].head; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline struct sk_buff *pktq_ppeek_tail(struct pktq *pq, int prec) | ||||||
|  | { | ||||||
|  | 	return pq->q[prec].tail; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | extern struct sk_buff *brcmu_pktq_penq(struct pktq *pq, int prec, | ||||||
|  | 				 struct sk_buff *p); | ||||||
|  | extern struct sk_buff *brcmu_pktq_penq_head(struct pktq *pq, int prec, | ||||||
|  | 				      struct sk_buff *p); | ||||||
|  | extern struct sk_buff *brcmu_pktq_pdeq(struct pktq *pq, int prec); | ||||||
|  | extern struct sk_buff *brcmu_pktq_pdeq_tail(struct pktq *pq, int prec); | ||||||
|  | 
 | ||||||
|  | /* packet primitives */ | ||||||
|  | extern struct sk_buff *brcmu_pkt_buf_get_skb(uint len); | ||||||
|  | extern void brcmu_pkt_buf_free_skb(struct sk_buff *skb); | ||||||
|  | 
 | ||||||
|  | /* Empty the queue at particular precedence level */ | ||||||
|  | /* callback function fn(pkt, arg) returns true if pkt belongs to if */ | ||||||
|  | extern void brcmu_pktq_pflush(struct pktq *pq, int prec, | ||||||
|  | 	bool dir, bool (*fn)(struct sk_buff *, void *), void *arg); | ||||||
|  | 
 | ||||||
|  | /* operations on a set of precedences in packet queue */ | ||||||
|  | 
 | ||||||
|  | extern int brcmu_pktq_mlen(struct pktq *pq, uint prec_bmp); | ||||||
|  | extern struct sk_buff *brcmu_pktq_mdeq(struct pktq *pq, uint prec_bmp, | ||||||
|  | 	int *prec_out); | ||||||
|  | 
 | ||||||
|  | /* operations on packet queue as a whole */ | ||||||
|  | 
 | ||||||
|  | static inline int pktq_len(struct pktq *pq) | ||||||
|  | { | ||||||
|  | 	return (int)pq->len; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline int pktq_max(struct pktq *pq) | ||||||
|  | { | ||||||
|  | 	return (int)pq->max; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline int pktq_avail(struct pktq *pq) | ||||||
|  | { | ||||||
|  | 	return (int)(pq->max - pq->len); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline bool pktq_full(struct pktq *pq) | ||||||
|  | { | ||||||
|  | 	return pq->len >= pq->max; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline bool pktq_empty(struct pktq *pq) | ||||||
|  | { | ||||||
|  | 	return pq->len == 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | extern void brcmu_pktq_init(struct pktq *pq, int num_prec, int max_len); | ||||||
|  | /* prec_out may be NULL if caller is not interested in return value */ | ||||||
|  | extern struct sk_buff *brcmu_pktq_peek_tail(struct pktq *pq, int *prec_out); | ||||||
|  | extern void brcmu_pktq_flush(struct pktq *pq, bool dir, | ||||||
|  | 		bool (*fn)(struct sk_buff *, void *), void *arg); | ||||||
|  | 
 | ||||||
|  | /* externs */ | ||||||
|  | /* packet */ | ||||||
|  | extern uint brcmu_pktfrombuf(struct sk_buff *p, | ||||||
|  | 	uint offset, int len, unsigned char *buf); | ||||||
|  | extern uint brcmu_pkttotlen(struct sk_buff *p); | ||||||
|  | 
 | ||||||
|  | /* ip address */ | ||||||
|  | struct ipv4_addr; | ||||||
|  | 
 | ||||||
|  | #ifdef BCMDBG | ||||||
|  | extern void brcmu_prpkt(const char *msg, struct sk_buff *p0); | ||||||
|  | #else | ||||||
|  | #define brcmu_prpkt(a, b) | ||||||
|  | #endif				/* BCMDBG */ | ||||||
|  | 
 | ||||||
|  | /* brcmu_format_flags() bit description structure */ | ||||||
|  | struct brcmu_bit_desc { | ||||||
|  | 	u32 bit; | ||||||
|  | 	const char *name; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* tag_ID/length/value_buffer tuple */ | ||||||
|  | struct brcmu_tlv { | ||||||
|  | 	u8 id; | ||||||
|  | 	u8 len; | ||||||
|  | 	u8 data[1]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* externs */ | ||||||
|  | /* format/print */ | ||||||
|  | #if defined(BCMDBG) | ||||||
|  | extern int brcmu_format_flags(const struct brcmu_bit_desc *bd, u32 flags, | ||||||
|  | 			      char *buf, int len); | ||||||
|  | extern int brcmu_format_hex(char *str, const void *bytes, int len); | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | extern char *brcmu_chipname(uint chipid, char *buf, uint len); | ||||||
|  | 
 | ||||||
|  | extern struct brcmu_tlv *brcmu_parse_tlvs(void *buf, int buflen, | ||||||
|  | 					  uint key); | ||||||
|  | 
 | ||||||
|  | /* power conversion */ | ||||||
|  | extern u16 brcmu_qdbm_to_mw(u8 qdbm); | ||||||
|  | extern u8 brcmu_mw_to_qdbm(u16 mw); | ||||||
|  | 
 | ||||||
|  | extern uint brcmu_mkiovar(char *name, char *data, uint datalen, | ||||||
|  | 			  char *buf, uint len); | ||||||
|  | extern uint brcmu_bitcount(u8 *bitmap, uint bytelength); | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCMU_UTILS_H_ */ | ||||||
							
								
								
									
										275
									
								
								drivers/net/wireless/brcm80211/include/brcmu_wifi.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										275
									
								
								drivers/net/wireless/brcm80211/include/brcmu_wifi.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,275 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef	_BRCMU_WIFI_H_ | ||||||
|  | #define	_BRCMU_WIFI_H_ | ||||||
|  | 
 | ||||||
|  | #include <linux/if_ether.h>		/* for ETH_ALEN */ | ||||||
|  | #include <linux/ieee80211.h>		/* for WLAN_PMKID_LEN */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * A chanspec (u16) holds the channel number, band, bandwidth and control | ||||||
|  |  * sideband | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* channel defines */ | ||||||
|  | #define CH_UPPER_SB			0x01 | ||||||
|  | #define CH_LOWER_SB			0x02 | ||||||
|  | #define CH_EWA_VALID			0x04 | ||||||
|  | #define CH_20MHZ_APART			4 | ||||||
|  | #define CH_10MHZ_APART			2 | ||||||
|  | #define CH_5MHZ_APART			1 /* 2G band channels are 5 Mhz apart */ | ||||||
|  | #define CH_MAX_2G_CHANNEL		14	/* Max channel in 2G band */ | ||||||
|  | #define BRCM_MAX_2G_CHANNEL	CH_MAX_2G_CHANNEL	/* legacy define */ | ||||||
|  | 
 | ||||||
|  | /* bandstate array indices */ | ||||||
|  | #define BAND_2G_INDEX		0	/* wlc->bandstate[x] index */ | ||||||
|  | #define BAND_5G_INDEX		1	/* wlc->bandstate[x] index */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * max # supported channels. The max channel no is 216, this is that + 1 | ||||||
|  |  * rounded up to a multiple of NBBY (8). DO NOT MAKE it > 255: channels are | ||||||
|  |  * u8's all over | ||||||
|  | */ | ||||||
|  | #define	MAXCHANNEL		224 | ||||||
|  | 
 | ||||||
|  | #define WL_CHANSPEC_CHAN_MASK		0x00ff | ||||||
|  | #define WL_CHANSPEC_CHAN_SHIFT		0 | ||||||
|  | 
 | ||||||
|  | #define WL_CHANSPEC_CTL_SB_MASK		0x0300 | ||||||
|  | #define WL_CHANSPEC_CTL_SB_SHIFT	     8 | ||||||
|  | #define WL_CHANSPEC_CTL_SB_LOWER	0x0100 | ||||||
|  | #define WL_CHANSPEC_CTL_SB_UPPER	0x0200 | ||||||
|  | #define WL_CHANSPEC_CTL_SB_NONE		0x0300 | ||||||
|  | 
 | ||||||
|  | #define WL_CHANSPEC_BW_MASK		0x0C00 | ||||||
|  | #define WL_CHANSPEC_BW_SHIFT		    10 | ||||||
|  | #define WL_CHANSPEC_BW_10		0x0400 | ||||||
|  | #define WL_CHANSPEC_BW_20		0x0800 | ||||||
|  | #define WL_CHANSPEC_BW_40		0x0C00 | ||||||
|  | 
 | ||||||
|  | #define WL_CHANSPEC_BAND_MASK		0xf000 | ||||||
|  | #define WL_CHANSPEC_BAND_SHIFT		12 | ||||||
|  | #define WL_CHANSPEC_BAND_5G		0x1000 | ||||||
|  | #define WL_CHANSPEC_BAND_2G		0x2000 | ||||||
|  | #define INVCHANSPEC			255 | ||||||
|  | 
 | ||||||
|  | /* used to calculate the chan_freq = chan_factor * 500Mhz + 5 * chan_number */ | ||||||
|  | #define WF_CHAN_FACTOR_2_4_G		4814	/* 2.4 GHz band, 2407 MHz */ | ||||||
|  | #define WF_CHAN_FACTOR_5_G		10000	/* 5   GHz band, 5000 MHz */ | ||||||
|  | #define WF_CHAN_FACTOR_4_G		8000	/* 4.9 GHz band for Japan */ | ||||||
|  | 
 | ||||||
|  | #define CHSPEC_CHANNEL(chspec)	((u8)((chspec) & WL_CHANSPEC_CHAN_MASK)) | ||||||
|  | #define CHSPEC_BAND(chspec)	((chspec) & WL_CHANSPEC_BAND_MASK) | ||||||
|  | 
 | ||||||
|  | #define CHSPEC_CTL_SB(chspec)	((chspec) & WL_CHANSPEC_CTL_SB_MASK) | ||||||
|  | #define CHSPEC_BW(chspec)	((chspec) & WL_CHANSPEC_BW_MASK) | ||||||
|  | 
 | ||||||
|  | #define CHSPEC_IS10(chspec) \ | ||||||
|  | 	(((chspec) & WL_CHANSPEC_BW_MASK) == WL_CHANSPEC_BW_10) | ||||||
|  | 
 | ||||||
|  | #define CHSPEC_IS20(chspec) \ | ||||||
|  | 	(((chspec) & WL_CHANSPEC_BW_MASK) == WL_CHANSPEC_BW_20) | ||||||
|  | 
 | ||||||
|  | #ifndef CHSPEC_IS40 | ||||||
|  | #define CHSPEC_IS40(chspec) \ | ||||||
|  | 	(((chspec) & WL_CHANSPEC_BW_MASK) == WL_CHANSPEC_BW_40) | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #define CHSPEC_IS5G(chspec) \ | ||||||
|  | 	(((chspec) & WL_CHANSPEC_BAND_MASK) == WL_CHANSPEC_BAND_5G) | ||||||
|  | 
 | ||||||
|  | #define CHSPEC_IS2G(chspec) \ | ||||||
|  | 	(((chspec) & WL_CHANSPEC_BAND_MASK) == WL_CHANSPEC_BAND_2G) | ||||||
|  | 
 | ||||||
|  | #define CHSPEC_SB_NONE(chspec) \ | ||||||
|  | 	(((chspec) & WL_CHANSPEC_CTL_SB_MASK) == WL_CHANSPEC_CTL_SB_NONE) | ||||||
|  | 
 | ||||||
|  | #define CHSPEC_SB_UPPER(chspec) \ | ||||||
|  | 	(((chspec) & WL_CHANSPEC_CTL_SB_MASK) == WL_CHANSPEC_CTL_SB_UPPER) | ||||||
|  | 
 | ||||||
|  | #define CHSPEC_SB_LOWER(chspec) \ | ||||||
|  | 	(((chspec) & WL_CHANSPEC_CTL_SB_MASK) == WL_CHANSPEC_CTL_SB_LOWER) | ||||||
|  | 
 | ||||||
|  | #define CHSPEC_CTL_CHAN(chspec) \ | ||||||
|  | 	((CHSPEC_SB_LOWER(chspec)) ? \ | ||||||
|  | 	(lower_20_sb(((chspec) & WL_CHANSPEC_CHAN_MASK))) : \ | ||||||
|  | 	(upper_20_sb(((chspec) & WL_CHANSPEC_CHAN_MASK)))) | ||||||
|  | 
 | ||||||
|  | #define CHSPEC2BAND(chspec) (CHSPEC_IS5G(chspec) ? BRCM_BAND_5G : BRCM_BAND_2G) | ||||||
|  | 
 | ||||||
|  | #define CHANSPEC_STR_LEN    8 | ||||||
|  | 
 | ||||||
|  | static inline int lower_20_sb(int channel) | ||||||
|  | { | ||||||
|  | 	return channel > CH_10MHZ_APART ? (channel - CH_10MHZ_APART) : 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline int upper_20_sb(int channel) | ||||||
|  | { | ||||||
|  | 	return (channel < (MAXCHANNEL - CH_10MHZ_APART)) ? | ||||||
|  | 	       channel + CH_10MHZ_APART : 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline int chspec_bandunit(u16 chspec) | ||||||
|  | { | ||||||
|  | 	return CHSPEC_IS5G(chspec) ? BAND_5G_INDEX : BAND_2G_INDEX; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline u16 ch20mhz_chspec(int channel) | ||||||
|  | { | ||||||
|  | 	u16 rc = channel <= CH_MAX_2G_CHANNEL ? | ||||||
|  | 		 WL_CHANSPEC_BAND_2G : WL_CHANSPEC_BAND_5G; | ||||||
|  | 
 | ||||||
|  | 	return	(u16)((u16)channel | WL_CHANSPEC_BW_20 | | ||||||
|  | 		      WL_CHANSPEC_CTL_SB_NONE | rc); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline int next_20mhz_chan(int channel) | ||||||
|  | { | ||||||
|  | 	return channel < (MAXCHANNEL - CH_20MHZ_APART) ? | ||||||
|  | 	       channel + CH_20MHZ_APART : 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* defined rate in 500kbps */ | ||||||
|  | #define BRCM_MAXRATE	108	/* in 500kbps units */ | ||||||
|  | #define BRCM_RATE_1M	2	/* in 500kbps units */ | ||||||
|  | #define BRCM_RATE_2M	4	/* in 500kbps units */ | ||||||
|  | #define BRCM_RATE_5M5	11	/* in 500kbps units */ | ||||||
|  | #define BRCM_RATE_11M	22	/* in 500kbps units */ | ||||||
|  | #define BRCM_RATE_6M	12	/* in 500kbps units */ | ||||||
|  | #define BRCM_RATE_9M	18	/* in 500kbps units */ | ||||||
|  | #define BRCM_RATE_12M	24	/* in 500kbps units */ | ||||||
|  | #define BRCM_RATE_18M	36	/* in 500kbps units */ | ||||||
|  | #define BRCM_RATE_24M	48	/* in 500kbps units */ | ||||||
|  | #define BRCM_RATE_36M	72	/* in 500kbps units */ | ||||||
|  | #define BRCM_RATE_48M	96	/* in 500kbps units */ | ||||||
|  | #define BRCM_RATE_54M	108	/* in 500kbps units */ | ||||||
|  | 
 | ||||||
|  | #define BRCM_2G_25MHZ_OFFSET		5	/* 2.4GHz band channel offset */ | ||||||
|  | 
 | ||||||
|  | #define MCSSET_LEN	16 | ||||||
|  | 
 | ||||||
|  | static inline bool ac_bitmap_tst(u8 bitmap, int prec) | ||||||
|  | { | ||||||
|  | 	return (bitmap & (1 << (prec))) != 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Verify the chanspec is using a legal set of parameters, i.e. that the | ||||||
|  |  * chanspec specified a band, bw, ctl_sb and channel and that the | ||||||
|  |  * combination could be legal given any set of circumstances. | ||||||
|  |  * RETURNS: true is the chanspec is malformed, false if it looks good. | ||||||
|  |  */ | ||||||
|  | extern bool brcmu_chspec_malformed(u16 chanspec); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * This function returns the channel number that control traffic is being sent | ||||||
|  |  * on, for legacy channels this is just the channel number, for 40MHZ channels | ||||||
|  |  * it is the upper or lower 20MHZ sideband depending on the chanspec selected. | ||||||
|  |  */ | ||||||
|  | extern u8 brcmu_chspec_ctlchan(u16 chspec); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Return the channel number for a given frequency and base frequency. | ||||||
|  |  * The returned channel number is relative to the given base frequency. | ||||||
|  |  * If the given base frequency is zero, a base frequency of 5 GHz is assumed for | ||||||
|  |  * frequencies from 5 - 6 GHz, and 2.407 GHz is assumed for 2.4 - 2.5 GHz. | ||||||
|  |  * | ||||||
|  |  * Frequency is specified in MHz. | ||||||
|  |  * The base frequency is specified as (start_factor * 500 kHz). | ||||||
|  |  * Constants WF_CHAN_FACTOR_2_4_G, WF_CHAN_FACTOR_5_G are defined for | ||||||
|  |  * 2.4 GHz and 5 GHz bands. | ||||||
|  |  * | ||||||
|  |  * The returned channel will be in the range [1, 14] in the 2.4 GHz band | ||||||
|  |  * and [0, 200] otherwise. | ||||||
|  |  * -1 is returned if the start_factor is WF_CHAN_FACTOR_2_4_G and the | ||||||
|  |  * frequency is not a 2.4 GHz channel, or if the frequency is not and even | ||||||
|  |  * multiple of 5 MHz from the base frequency to the base plus 1 GHz. | ||||||
|  |  * | ||||||
|  |  * Reference 802.11 REVma, section 17.3.8.3, and 802.11B section 18.4.6.2 | ||||||
|  |  */ | ||||||
|  | extern int brcmu_mhz2channel(uint freq, uint start_factor); | ||||||
|  | 
 | ||||||
|  | /* Enumerate crypto algorithms */ | ||||||
|  | #define	CRYPTO_ALGO_OFF			0 | ||||||
|  | #define	CRYPTO_ALGO_WEP1		1 | ||||||
|  | #define	CRYPTO_ALGO_TKIP		2 | ||||||
|  | #define	CRYPTO_ALGO_WEP128		3 | ||||||
|  | #define CRYPTO_ALGO_AES_CCM		4 | ||||||
|  | #define CRYPTO_ALGO_AES_RESERVED1	5 | ||||||
|  | #define CRYPTO_ALGO_AES_RESERVED2	6 | ||||||
|  | #define CRYPTO_ALGO_NALG		7 | ||||||
|  | 
 | ||||||
|  | /* wireless security bitvec */ | ||||||
|  | 
 | ||||||
|  | #define WEP_ENABLED		0x0001 | ||||||
|  | #define TKIP_ENABLED		0x0002 | ||||||
|  | #define AES_ENABLED		0x0004 | ||||||
|  | #define WSEC_SWFLAG		0x0008 | ||||||
|  | /* to go into transition mode without setting wep */ | ||||||
|  | #define SES_OW_ENABLED		0x0040 | ||||||
|  | 
 | ||||||
|  | /* WPA authentication mode bitvec */ | ||||||
|  | #define WPA_AUTH_DISABLED	0x0000	/* Legacy (i.e., non-WPA) */ | ||||||
|  | #define WPA_AUTH_NONE		0x0001	/* none (IBSS) */ | ||||||
|  | #define WPA_AUTH_UNSPECIFIED	0x0002	/* over 802.1x */ | ||||||
|  | #define WPA_AUTH_PSK		0x0004	/* Pre-shared key */ | ||||||
|  | #define WPA_AUTH_RESERVED1	0x0008 | ||||||
|  | #define WPA_AUTH_RESERVED2	0x0010 | ||||||
|  | 
 | ||||||
|  | #define WPA2_AUTH_RESERVED1	0x0020 | ||||||
|  | #define WPA2_AUTH_UNSPECIFIED	0x0040	/* over 802.1x */ | ||||||
|  | #define WPA2_AUTH_PSK		0x0080	/* Pre-shared key */ | ||||||
|  | #define WPA2_AUTH_RESERVED3	0x0200 | ||||||
|  | #define WPA2_AUTH_RESERVED4	0x0400 | ||||||
|  | #define WPA2_AUTH_RESERVED5	0x0800 | ||||||
|  | 
 | ||||||
|  | /* pmkid */ | ||||||
|  | #define	MAXPMKID		16 | ||||||
|  | 
 | ||||||
|  | #define DOT11_DEFAULT_RTS_LEN		2347 | ||||||
|  | #define DOT11_DEFAULT_FRAG_LEN		2346 | ||||||
|  | 
 | ||||||
|  | #define DOT11_ICV_AES_LEN		8 | ||||||
|  | #define DOT11_QOS_LEN			2 | ||||||
|  | #define DOT11_IV_MAX_LEN		8 | ||||||
|  | #define DOT11_A4_HDR_LEN		30 | ||||||
|  | 
 | ||||||
|  | #define HT_CAP_RX_STBC_NO		0x0 | ||||||
|  | #define HT_CAP_RX_STBC_ONE_STREAM	0x1 | ||||||
|  | 
 | ||||||
|  | struct pmkid { | ||||||
|  | 	u8 BSSID[ETH_ALEN]; | ||||||
|  | 	u8 PMKID[WLAN_PMKID_LEN]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct pmkid_list { | ||||||
|  | 	u32 npmkid; | ||||||
|  | 	struct pmkid pmkid[1]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct pmkid_cand { | ||||||
|  | 	u8 BSSID[ETH_ALEN]; | ||||||
|  | 	u8 preauth; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct pmkid_cand_list { | ||||||
|  | 	u32 npmkid_cand; | ||||||
|  | 	struct pmkid_cand pmkid_cand[1]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCMU_WIFI_H_ */ | ||||||
							
								
								
									
										284
									
								
								drivers/net/wireless/brcm80211/include/chipcommon.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										284
									
								
								drivers/net/wireless/brcm80211/include/chipcommon.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,284 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef	_SBCHIPC_H | ||||||
|  | #define	_SBCHIPC_H | ||||||
|  | 
 | ||||||
|  | #include "defs.h"		/* for PAD macro */ | ||||||
|  | 
 | ||||||
|  | struct chipcregs { | ||||||
|  | 	u32 chipid;		/* 0x0 */ | ||||||
|  | 	u32 capabilities; | ||||||
|  | 	u32 corecontrol;	/* corerev >= 1 */ | ||||||
|  | 	u32 bist; | ||||||
|  | 
 | ||||||
|  | 	/* OTP */ | ||||||
|  | 	u32 otpstatus;	/* 0x10, corerev >= 10 */ | ||||||
|  | 	u32 otpcontrol; | ||||||
|  | 	u32 otpprog; | ||||||
|  | 	u32 otplayout;	/* corerev >= 23 */ | ||||||
|  | 
 | ||||||
|  | 	/* Interrupt control */ | ||||||
|  | 	u32 intstatus;	/* 0x20 */ | ||||||
|  | 	u32 intmask; | ||||||
|  | 
 | ||||||
|  | 	/* Chip specific regs */ | ||||||
|  | 	u32 chipcontrol;	/* 0x28, rev >= 11 */ | ||||||
|  | 	u32 chipstatus;	/* 0x2c, rev >= 11 */ | ||||||
|  | 
 | ||||||
|  | 	/* Jtag Master */ | ||||||
|  | 	u32 jtagcmd;		/* 0x30, rev >= 10 */ | ||||||
|  | 	u32 jtagir; | ||||||
|  | 	u32 jtagdr; | ||||||
|  | 	u32 jtagctrl; | ||||||
|  | 
 | ||||||
|  | 	/* serial flash interface registers */ | ||||||
|  | 	u32 flashcontrol;	/* 0x40 */ | ||||||
|  | 	u32 flashaddress; | ||||||
|  | 	u32 flashdata; | ||||||
|  | 	u32 PAD[1]; | ||||||
|  | 
 | ||||||
|  | 	/* Silicon backplane configuration broadcast control */ | ||||||
|  | 	u32 broadcastaddress;	/* 0x50 */ | ||||||
|  | 	u32 broadcastdata; | ||||||
|  | 
 | ||||||
|  | 	/* gpio - cleared only by power-on-reset */ | ||||||
|  | 	u32 gpiopullup;	/* 0x58, corerev >= 20 */ | ||||||
|  | 	u32 gpiopulldown;	/* 0x5c, corerev >= 20 */ | ||||||
|  | 	u32 gpioin;		/* 0x60 */ | ||||||
|  | 	u32 gpioout;		/* 0x64 */ | ||||||
|  | 	u32 gpioouten;	/* 0x68 */ | ||||||
|  | 	u32 gpiocontrol;	/* 0x6C */ | ||||||
|  | 	u32 gpiointpolarity;	/* 0x70 */ | ||||||
|  | 	u32 gpiointmask;	/* 0x74 */ | ||||||
|  | 
 | ||||||
|  | 	/* GPIO events corerev >= 11 */ | ||||||
|  | 	u32 gpioevent; | ||||||
|  | 	u32 gpioeventintmask; | ||||||
|  | 
 | ||||||
|  | 	/* Watchdog timer */ | ||||||
|  | 	u32 watchdog;	/* 0x80 */ | ||||||
|  | 
 | ||||||
|  | 	/* GPIO events corerev >= 11 */ | ||||||
|  | 	u32 gpioeventintpolarity; | ||||||
|  | 
 | ||||||
|  | 	/* GPIO based LED powersave registers corerev >= 16 */ | ||||||
|  | 	u32 gpiotimerval;	/* 0x88 */ | ||||||
|  | 	u32 gpiotimeroutmask; | ||||||
|  | 
 | ||||||
|  | 	/* clock control */ | ||||||
|  | 	u32 clockcontrol_n;	/* 0x90 */ | ||||||
|  | 	u32 clockcontrol_sb;	/* aka m0 */ | ||||||
|  | 	u32 clockcontrol_pci;	/* aka m1 */ | ||||||
|  | 	u32 clockcontrol_m2;	/* mii/uart/mipsref */ | ||||||
|  | 	u32 clockcontrol_m3;	/* cpu */ | ||||||
|  | 	u32 clkdiv;		/* corerev >= 3 */ | ||||||
|  | 	u32 gpiodebugsel;	/* corerev >= 28 */ | ||||||
|  | 	u32 capabilities_ext;	/* 0xac  */ | ||||||
|  | 
 | ||||||
|  | 	/* pll delay registers (corerev >= 4) */ | ||||||
|  | 	u32 pll_on_delay;	/* 0xb0 */ | ||||||
|  | 	u32 fref_sel_delay; | ||||||
|  | 	u32 slow_clk_ctl;	/* 5 < corerev < 10 */ | ||||||
|  | 	u32 PAD; | ||||||
|  | 
 | ||||||
|  | 	/* Instaclock registers (corerev >= 10) */ | ||||||
|  | 	u32 system_clk_ctl;	/* 0xc0 */ | ||||||
|  | 	u32 clkstatestretch; | ||||||
|  | 	u32 PAD[2]; | ||||||
|  | 
 | ||||||
|  | 	/* Indirect backplane access (corerev >= 22) */ | ||||||
|  | 	u32 bp_addrlow;	/* 0xd0 */ | ||||||
|  | 	u32 bp_addrhigh; | ||||||
|  | 	u32 bp_data; | ||||||
|  | 	u32 PAD; | ||||||
|  | 	u32 bp_indaccess; | ||||||
|  | 	u32 PAD[3]; | ||||||
|  | 
 | ||||||
|  | 	/* More clock dividers (corerev >= 32) */ | ||||||
|  | 	u32 clkdiv2; | ||||||
|  | 	u32 PAD[2]; | ||||||
|  | 
 | ||||||
|  | 	/* In AI chips, pointer to erom */ | ||||||
|  | 	u32 eromptr;		/* 0xfc */ | ||||||
|  | 
 | ||||||
|  | 	/* ExtBus control registers (corerev >= 3) */ | ||||||
|  | 	u32 pcmcia_config;	/* 0x100 */ | ||||||
|  | 	u32 pcmcia_memwait; | ||||||
|  | 	u32 pcmcia_attrwait; | ||||||
|  | 	u32 pcmcia_iowait; | ||||||
|  | 	u32 ide_config; | ||||||
|  | 	u32 ide_memwait; | ||||||
|  | 	u32 ide_attrwait; | ||||||
|  | 	u32 ide_iowait; | ||||||
|  | 	u32 prog_config; | ||||||
|  | 	u32 prog_waitcount; | ||||||
|  | 	u32 flash_config; | ||||||
|  | 	u32 flash_waitcount; | ||||||
|  | 	u32 SECI_config;	/* 0x130 SECI configuration */ | ||||||
|  | 	u32 PAD[3]; | ||||||
|  | 
 | ||||||
|  | 	/* Enhanced Coexistence Interface (ECI) registers (corerev >= 21) */ | ||||||
|  | 	u32 eci_output;	/* 0x140 */ | ||||||
|  | 	u32 eci_control; | ||||||
|  | 	u32 eci_inputlo; | ||||||
|  | 	u32 eci_inputmi; | ||||||
|  | 	u32 eci_inputhi; | ||||||
|  | 	u32 eci_inputintpolaritylo; | ||||||
|  | 	u32 eci_inputintpolaritymi; | ||||||
|  | 	u32 eci_inputintpolarityhi; | ||||||
|  | 	u32 eci_intmasklo; | ||||||
|  | 	u32 eci_intmaskmi; | ||||||
|  | 	u32 eci_intmaskhi; | ||||||
|  | 	u32 eci_eventlo; | ||||||
|  | 	u32 eci_eventmi; | ||||||
|  | 	u32 eci_eventhi; | ||||||
|  | 	u32 eci_eventmasklo; | ||||||
|  | 	u32 eci_eventmaskmi; | ||||||
|  | 	u32 eci_eventmaskhi; | ||||||
|  | 	u32 PAD[3]; | ||||||
|  | 
 | ||||||
|  | 	/* SROM interface (corerev >= 32) */ | ||||||
|  | 	u32 sromcontrol;	/* 0x190 */ | ||||||
|  | 	u32 sromaddress; | ||||||
|  | 	u32 sromdata; | ||||||
|  | 	u32 PAD[17]; | ||||||
|  | 
 | ||||||
|  | 	/* Clock control and hardware workarounds (corerev >= 20) */ | ||||||
|  | 	u32 clk_ctl_st;	/* 0x1e0 */ | ||||||
|  | 	u32 hw_war; | ||||||
|  | 	u32 PAD[70]; | ||||||
|  | 
 | ||||||
|  | 	/* UARTs */ | ||||||
|  | 	u8 uart0data;	/* 0x300 */ | ||||||
|  | 	u8 uart0imr; | ||||||
|  | 	u8 uart0fcr; | ||||||
|  | 	u8 uart0lcr; | ||||||
|  | 	u8 uart0mcr; | ||||||
|  | 	u8 uart0lsr; | ||||||
|  | 	u8 uart0msr; | ||||||
|  | 	u8 uart0scratch; | ||||||
|  | 	u8 PAD[248];		/* corerev >= 1 */ | ||||||
|  | 
 | ||||||
|  | 	u8 uart1data;	/* 0x400 */ | ||||||
|  | 	u8 uart1imr; | ||||||
|  | 	u8 uart1fcr; | ||||||
|  | 	u8 uart1lcr; | ||||||
|  | 	u8 uart1mcr; | ||||||
|  | 	u8 uart1lsr; | ||||||
|  | 	u8 uart1msr; | ||||||
|  | 	u8 uart1scratch; | ||||||
|  | 	u32 PAD[126]; | ||||||
|  | 
 | ||||||
|  | 	/* PMU registers (corerev >= 20) */ | ||||||
|  | 	u32 pmucontrol;	/* 0x600 */ | ||||||
|  | 	u32 pmucapabilities; | ||||||
|  | 	u32 pmustatus; | ||||||
|  | 	u32 res_state; | ||||||
|  | 	u32 res_pending; | ||||||
|  | 	u32 pmutimer; | ||||||
|  | 	u32 min_res_mask; | ||||||
|  | 	u32 max_res_mask; | ||||||
|  | 	u32 res_table_sel; | ||||||
|  | 	u32 res_dep_mask; | ||||||
|  | 	u32 res_updn_timer; | ||||||
|  | 	u32 res_timer; | ||||||
|  | 	u32 clkstretch; | ||||||
|  | 	u32 pmuwatchdog; | ||||||
|  | 	u32 gpiosel;		/* 0x638, rev >= 1 */ | ||||||
|  | 	u32 gpioenable;	/* 0x63c, rev >= 1 */ | ||||||
|  | 	u32 res_req_timer_sel; | ||||||
|  | 	u32 res_req_timer; | ||||||
|  | 	u32 res_req_mask; | ||||||
|  | 	u32 PAD; | ||||||
|  | 	u32 chipcontrol_addr;	/* 0x650 */ | ||||||
|  | 	u32 chipcontrol_data;	/* 0x654 */ | ||||||
|  | 	u32 regcontrol_addr; | ||||||
|  | 	u32 regcontrol_data; | ||||||
|  | 	u32 pllcontrol_addr; | ||||||
|  | 	u32 pllcontrol_data; | ||||||
|  | 	u32 pmustrapopt;	/* 0x668, corerev >= 28 */ | ||||||
|  | 	u32 pmu_xtalfreq;	/* 0x66C, pmurev >= 10 */ | ||||||
|  | 	u32 PAD[100]; | ||||||
|  | 	u16 sromotp[768]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* chipid */ | ||||||
|  | #define	CID_ID_MASK		0x0000ffff	/* Chip Id mask */ | ||||||
|  | #define	CID_REV_MASK		0x000f0000	/* Chip Revision mask */ | ||||||
|  | #define	CID_REV_SHIFT		16	/* Chip Revision shift */ | ||||||
|  | #define	CID_PKG_MASK		0x00f00000	/* Package Option mask */ | ||||||
|  | #define	CID_PKG_SHIFT		20	/* Package Option shift */ | ||||||
|  | #define	CID_CC_MASK		0x0f000000	/* CoreCount (corerev >= 4) */ | ||||||
|  | #define CID_CC_SHIFT		24 | ||||||
|  | #define	CID_TYPE_MASK		0xf0000000	/* Chip Type */ | ||||||
|  | #define CID_TYPE_SHIFT		28 | ||||||
|  | 
 | ||||||
|  | /* capabilities */ | ||||||
|  | #define	CC_CAP_UARTS_MASK	0x00000003	/* Number of UARTs */ | ||||||
|  | #define CC_CAP_MIPSEB		0x00000004	/* MIPS is in big-endian mode */ | ||||||
|  | #define CC_CAP_UCLKSEL		0x00000018	/* UARTs clock select */ | ||||||
|  | /* UARTs are driven by internal divided clock */ | ||||||
|  | #define CC_CAP_UINTCLK		0x00000008 | ||||||
|  | #define CC_CAP_UARTGPIO		0x00000020	/* UARTs own GPIOs 15:12 */ | ||||||
|  | #define CC_CAP_EXTBUS_MASK	0x000000c0	/* External bus mask */ | ||||||
|  | #define CC_CAP_EXTBUS_NONE	0x00000000	/* No ExtBus present */ | ||||||
|  | #define CC_CAP_EXTBUS_FULL	0x00000040	/* ExtBus: PCMCIA, IDE & Prog */ | ||||||
|  | #define CC_CAP_EXTBUS_PROG	0x00000080	/* ExtBus: ProgIf only */ | ||||||
|  | #define	CC_CAP_FLASH_MASK	0x00000700	/* Type of flash */ | ||||||
|  | #define	CC_CAP_PLL_MASK		0x00038000	/* Type of PLL */ | ||||||
|  | #define CC_CAP_PWR_CTL		0x00040000	/* Power control */ | ||||||
|  | #define CC_CAP_OTPSIZE		0x00380000	/* OTP Size (0 = none) */ | ||||||
|  | #define CC_CAP_OTPSIZE_SHIFT	19	/* OTP Size shift */ | ||||||
|  | #define CC_CAP_OTPSIZE_BASE	5	/* OTP Size base */ | ||||||
|  | #define CC_CAP_JTAGP		0x00400000	/* JTAG Master Present */ | ||||||
|  | #define CC_CAP_ROM		0x00800000	/* Internal boot rom active */ | ||||||
|  | #define CC_CAP_BKPLN64		0x08000000	/* 64-bit backplane */ | ||||||
|  | #define	CC_CAP_PMU		0x10000000	/* PMU Present, rev >= 20 */ | ||||||
|  | #define	CC_CAP_SROM		0x40000000	/* Srom Present, rev >= 32 */ | ||||||
|  | /* Nand flash present, rev >= 35 */ | ||||||
|  | #define	CC_CAP_NFLASH		0x80000000 | ||||||
|  | 
 | ||||||
|  | #define	CC_CAP2_SECI		0x00000001	/* SECI Present, rev >= 36 */ | ||||||
|  | /* GSIO (spi/i2c) present, rev >= 37 */ | ||||||
|  | #define	CC_CAP2_GSIO		0x00000002 | ||||||
|  | 
 | ||||||
|  | /* pmucapabilities */ | ||||||
|  | #define PCAP_REV_MASK	0x000000ff | ||||||
|  | #define PCAP_RC_MASK	0x00001f00 | ||||||
|  | #define PCAP_RC_SHIFT	8 | ||||||
|  | #define PCAP_TC_MASK	0x0001e000 | ||||||
|  | #define PCAP_TC_SHIFT	13 | ||||||
|  | #define PCAP_PC_MASK	0x001e0000 | ||||||
|  | #define PCAP_PC_SHIFT	17 | ||||||
|  | #define PCAP_VC_MASK	0x01e00000 | ||||||
|  | #define PCAP_VC_SHIFT	21 | ||||||
|  | #define PCAP_CC_MASK	0x1e000000 | ||||||
|  | #define PCAP_CC_SHIFT	25 | ||||||
|  | #define PCAP5_PC_MASK	0x003e0000	/* PMU corerev >= 5 */ | ||||||
|  | #define PCAP5_PC_SHIFT	17 | ||||||
|  | #define PCAP5_VC_MASK	0x07c00000 | ||||||
|  | #define PCAP5_VC_SHIFT	22 | ||||||
|  | #define PCAP5_CC_MASK	0xf8000000 | ||||||
|  | #define PCAP5_CC_SHIFT	27 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  | * Maximum delay for the PMU state transition in us. | ||||||
|  | * This is an upper bound intended for spinwaits etc. | ||||||
|  | */ | ||||||
|  | #define PMU_MAX_TRANSITION_DLY	15000 | ||||||
|  | 
 | ||||||
|  | #endif				/* _SBCHIPC_H */ | ||||||
							
								
								
									
										104
									
								
								drivers/net/wireless/brcm80211/include/defs.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										104
									
								
								drivers/net/wireless/brcm80211/include/defs.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,104 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef	_BRCM_DEFS_H_ | ||||||
|  | #define	_BRCM_DEFS_H_ | ||||||
|  | 
 | ||||||
|  | #include <linux/types.h> | ||||||
|  | 
 | ||||||
|  | #define	SI_BUS			0 | ||||||
|  | #define	PCI_BUS			1 | ||||||
|  | #define	PCMCIA_BUS		2 | ||||||
|  | #define SDIO_BUS		3 | ||||||
|  | #define JTAG_BUS		4 | ||||||
|  | #define USB_BUS			5 | ||||||
|  | #define SPI_BUS			6 | ||||||
|  | 
 | ||||||
|  | #define	OFF	0 | ||||||
|  | #define	ON	1		/* ON = 1 */ | ||||||
|  | #define	AUTO	(-1)		/* Auto = -1 */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Priority definitions according 802.1D | ||||||
|  |  */ | ||||||
|  | #define	PRIO_8021D_NONE		2 | ||||||
|  | #define	PRIO_8021D_BK		1 | ||||||
|  | #define	PRIO_8021D_BE		0 | ||||||
|  | #define	PRIO_8021D_EE		3 | ||||||
|  | #define	PRIO_8021D_CL		4 | ||||||
|  | #define	PRIO_8021D_VI		5 | ||||||
|  | #define	PRIO_8021D_VO		6 | ||||||
|  | #define	PRIO_8021D_NC		7 | ||||||
|  | 
 | ||||||
|  | #define	MAXPRIO			7 | ||||||
|  | #define NUMPRIO			(MAXPRIO + 1) | ||||||
|  | 
 | ||||||
|  | #define WL_NUMRATES		16	/* max # of rates in a rateset */ | ||||||
|  | 
 | ||||||
|  | #define BRCM_CNTRY_BUF_SZ	4	/* Country string is 3 bytes + NUL */ | ||||||
|  | 
 | ||||||
|  | #define BRCM_SET_CHANNEL	30 | ||||||
|  | #define BRCM_SET_SRL		32 | ||||||
|  | #define BRCM_SET_LRL		34 | ||||||
|  | #define BRCM_SET_BCNPRD		76 | ||||||
|  | 
 | ||||||
|  | #define BRCM_GET_CURR_RATESET	114	/* current rateset */ | ||||||
|  | #define BRCM_GET_PHYLIST	180 | ||||||
|  | 
 | ||||||
|  | /* Bit masks for radio disabled status - returned by WL_GET_RADIO */ | ||||||
|  | 
 | ||||||
|  | #define WL_RADIO_SW_DISABLE		(1<<0) | ||||||
|  | #define WL_RADIO_HW_DISABLE		(1<<1) | ||||||
|  | #define WL_RADIO_MPC_DISABLE		(1<<2) | ||||||
|  | /* some countries don't support any channel */ | ||||||
|  | #define WL_RADIO_COUNTRY_DISABLE	(1<<3) | ||||||
|  | 
 | ||||||
|  | /* Override bit for SET_TXPWR.  if set, ignore other level limits */ | ||||||
|  | #define WL_TXPWR_OVERRIDE	(1U<<31) | ||||||
|  | 
 | ||||||
|  | /* band types */ | ||||||
|  | #define	BRCM_BAND_AUTO		0	/* auto-select */ | ||||||
|  | #define	BRCM_BAND_5G		1	/* 5 Ghz */ | ||||||
|  | #define	BRCM_BAND_2G		2	/* 2.4 Ghz */ | ||||||
|  | #define	BRCM_BAND_ALL		3	/* all bands */ | ||||||
|  | 
 | ||||||
|  | /* Values for PM */ | ||||||
|  | #define PM_OFF	0 | ||||||
|  | #define PM_MAX	1 | ||||||
|  | 
 | ||||||
|  | /* Message levels */ | ||||||
|  | #define LOG_ERROR_VAL		0x00000001 | ||||||
|  | #define LOG_TRACE_VAL		0x00000002 | ||||||
|  | 
 | ||||||
|  | #define PM_OFF	0 | ||||||
|  | #define PM_MAX	1 | ||||||
|  | #define PM_FAST 2 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Sonics Configuration Space Registers. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* core sbconfig regs are top 256bytes of regs */ | ||||||
|  | #define	SBCONFIGOFF		0xf00 | ||||||
|  | 
 | ||||||
|  | /* cpp contortions to concatenate w/arg prescan */ | ||||||
|  | #ifndef	PAD | ||||||
|  | #define	_PADLINE(line)	pad ## line | ||||||
|  | #define	_XSTR(line)	_PADLINE(line) | ||||||
|  | #define	PAD		_XSTR(__LINE__) | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCM_DEFS_H_ */ | ||||||
							
								
								
									
										90
									
								
								drivers/net/wireless/brcm80211/include/soc.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								drivers/net/wireless/brcm80211/include/soc.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,90 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2010 Broadcom Corporation | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef	_BRCM_SOC_H | ||||||
|  | #define	_BRCM_SOC_H | ||||||
|  | 
 | ||||||
|  | #define SI_ENUM_BASE		0x18000000	/* Enumeration space base */ | ||||||
|  | 
 | ||||||
|  | /* core codes */ | ||||||
|  | #define	NODEV_CORE_ID		0x700	/* Invalid coreid */ | ||||||
|  | #define	CC_CORE_ID		0x800	/* chipcommon core */ | ||||||
|  | #define	ILINE20_CORE_ID		0x801	/* iline20 core */ | ||||||
|  | #define	SRAM_CORE_ID		0x802	/* sram core */ | ||||||
|  | #define	SDRAM_CORE_ID		0x803	/* sdram core */ | ||||||
|  | #define	PCI_CORE_ID		0x804	/* pci core */ | ||||||
|  | #define	MIPS_CORE_ID		0x805	/* mips core */ | ||||||
|  | #define	ENET_CORE_ID		0x806	/* enet mac core */ | ||||||
|  | #define	CODEC_CORE_ID		0x807	/* v90 codec core */ | ||||||
|  | #define	USB_CORE_ID		0x808	/* usb 1.1 host/device core */ | ||||||
|  | #define	ADSL_CORE_ID		0x809	/* ADSL core */ | ||||||
|  | #define	ILINE100_CORE_ID	0x80a	/* iline100 core */ | ||||||
|  | #define	IPSEC_CORE_ID		0x80b	/* ipsec core */ | ||||||
|  | #define	UTOPIA_CORE_ID		0x80c	/* utopia core */ | ||||||
|  | #define	PCMCIA_CORE_ID		0x80d	/* pcmcia core */ | ||||||
|  | #define	SOCRAM_CORE_ID		0x80e	/* internal memory core */ | ||||||
|  | #define	MEMC_CORE_ID		0x80f	/* memc sdram core */ | ||||||
|  | #define	OFDM_CORE_ID		0x810	/* OFDM phy core */ | ||||||
|  | #define	EXTIF_CORE_ID		0x811	/* external interface core */ | ||||||
|  | #define	D11_CORE_ID		0x812	/* 802.11 MAC core */ | ||||||
|  | #define	APHY_CORE_ID		0x813	/* 802.11a phy core */ | ||||||
|  | #define	BPHY_CORE_ID		0x814	/* 802.11b phy core */ | ||||||
|  | #define	GPHY_CORE_ID		0x815	/* 802.11g phy core */ | ||||||
|  | #define	MIPS33_CORE_ID		0x816	/* mips3302 core */ | ||||||
|  | #define	USB11H_CORE_ID		0x817	/* usb 1.1 host core */ | ||||||
|  | #define	USB11D_CORE_ID		0x818	/* usb 1.1 device core */ | ||||||
|  | #define	USB20H_CORE_ID		0x819	/* usb 2.0 host core */ | ||||||
|  | #define	USB20D_CORE_ID		0x81a	/* usb 2.0 device core */ | ||||||
|  | #define	SDIOH_CORE_ID		0x81b	/* sdio host core */ | ||||||
|  | #define	ROBO_CORE_ID		0x81c	/* roboswitch core */ | ||||||
|  | #define	ATA100_CORE_ID		0x81d	/* parallel ATA core */ | ||||||
|  | #define	SATAXOR_CORE_ID		0x81e	/* serial ATA & XOR DMA core */ | ||||||
|  | #define	GIGETH_CORE_ID		0x81f	/* gigabit ethernet core */ | ||||||
|  | #define	PCIE_CORE_ID		0x820	/* pci express core */ | ||||||
|  | #define	NPHY_CORE_ID		0x821	/* 802.11n 2x2 phy core */ | ||||||
|  | #define	SRAMC_CORE_ID		0x822	/* SRAM controller core */ | ||||||
|  | #define	MINIMAC_CORE_ID		0x823	/* MINI MAC/phy core */ | ||||||
|  | #define	ARM11_CORE_ID		0x824	/* ARM 1176 core */ | ||||||
|  | #define	ARM7S_CORE_ID		0x825	/* ARM7tdmi-s core */ | ||||||
|  | #define	LPPHY_CORE_ID		0x826	/* 802.11a/b/g phy core */ | ||||||
|  | #define	PMU_CORE_ID		0x827	/* PMU core */ | ||||||
|  | #define	SSNPHY_CORE_ID		0x828	/* 802.11n single-stream phy core */ | ||||||
|  | #define	SDIOD_CORE_ID		0x829	/* SDIO device core */ | ||||||
|  | #define	ARMCM3_CORE_ID		0x82a	/* ARM Cortex M3 core */ | ||||||
|  | #define	HTPHY_CORE_ID		0x82b	/* 802.11n 4x4 phy core */ | ||||||
|  | #define	MIPS74K_CORE_ID		0x82c	/* mips 74k core */ | ||||||
|  | #define	GMAC_CORE_ID		0x82d	/* Gigabit MAC core */ | ||||||
|  | #define	DMEMC_CORE_ID		0x82e	/* DDR1/2 memory controller core */ | ||||||
|  | #define	PCIERC_CORE_ID		0x82f	/* PCIE Root Complex core */ | ||||||
|  | #define	OCP_CORE_ID		0x830	/* OCP2OCP bridge core */ | ||||||
|  | #define	SC_CORE_ID		0x831	/* shared common core */ | ||||||
|  | #define	AHB_CORE_ID		0x832	/* OCP2AHB bridge core */ | ||||||
|  | #define	SPIH_CORE_ID		0x833	/* SPI host core */ | ||||||
|  | #define	I2S_CORE_ID		0x834	/* I2S core */ | ||||||
|  | #define	DMEMS_CORE_ID		0x835	/* SDR/DDR1 memory controller core */ | ||||||
|  | #define	DEF_SHIM_COMP		0x837	/* SHIM component in ubus/6362 */ | ||||||
|  | #define OOB_ROUTER_CORE_ID	0x367	/* OOB router core ID */ | ||||||
|  | /* Default component, in ai chips it maps all unused address ranges */ | ||||||
|  | #define	DEF_AI_COMP		0xfff | ||||||
|  | 
 | ||||||
|  | /* Common core control flags */ | ||||||
|  | #define	SICF_BIST_EN		0x8000 | ||||||
|  | #define	SICF_PME_EN		0x4000 | ||||||
|  | #define	SICF_CORE_BITS		0x3ffc | ||||||
|  | #define	SICF_FGC		0x0002 | ||||||
|  | #define	SICF_CLOCK_EN		0x0001 | ||||||
|  | 
 | ||||||
|  | #endif				/* _BRCM_SOC_H */ | ||||||
| @ -44,7 +44,7 @@ source "drivers/staging/wlan-ng/Kconfig" | |||||||
| 
 | 
 | ||||||
| source "drivers/staging/echo/Kconfig" | source "drivers/staging/echo/Kconfig" | ||||||
| 
 | 
 | ||||||
| source "drivers/staging/brcm80211/Kconfig" | 
 | ||||||
| 
 | 
 | ||||||
| source "drivers/staging/comedi/Kconfig" | source "drivers/staging/comedi/Kconfig" | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -14,8 +14,8 @@ obj-$(CONFIG_USBIP_CORE)	+= usbip/ | |||||||
| obj-$(CONFIG_W35UND)		+= winbond/ | obj-$(CONFIG_W35UND)		+= winbond/ | ||||||
| obj-$(CONFIG_PRISM2_USB)	+= wlan-ng/ | obj-$(CONFIG_PRISM2_USB)	+= wlan-ng/ | ||||||
| obj-$(CONFIG_ECHO)		+= echo/ | obj-$(CONFIG_ECHO)		+= echo/ | ||||||
| obj-$(CONFIG_BRCMSMAC)		+= brcm80211/ | 
 | ||||||
| obj-$(CONFIG_BRCMFMAC)		+= brcm80211/ | 
 | ||||||
| obj-$(CONFIG_COMEDI)		+= comedi/ | obj-$(CONFIG_COMEDI)		+= comedi/ | ||||||
| obj-$(CONFIG_FB_OLPC_DCON)	+= olpc_dcon/ | obj-$(CONFIG_FB_OLPC_DCON)	+= olpc_dcon/ | ||||||
| obj-$(CONFIG_ASUS_OLED)		+= asus_oled/ | obj-$(CONFIG_ASUS_OLED)		+= asus_oled/ | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Arend van Spriel
						Arend van Spriel