diff options
Diffstat (limited to 'target/linux/realtek/files/drivers/net/wireless/rtl8192cd/8192cd_osdep.c')
-rw-r--r-- | target/linux/realtek/files/drivers/net/wireless/rtl8192cd/8192cd_osdep.c | 7899 |
1 files changed, 7899 insertions, 0 deletions
diff --git a/target/linux/realtek/files/drivers/net/wireless/rtl8192cd/8192cd_osdep.c b/target/linux/realtek/files/drivers/net/wireless/rtl8192cd/8192cd_osdep.c new file mode 100644 index 000000000..a85745b42 --- /dev/null +++ b/target/linux/realtek/files/drivers/net/wireless/rtl8192cd/8192cd_osdep.c @@ -0,0 +1,7899 @@ +/* + * Routines to handle OS dependent jobs and interfaces + * + * $Id: 8192cd_osdep.c,v 1.61.2.28 2011/01/11 13:48:37 button Exp $ + * + * Copyright (c) 2009 Realtek Semiconductor Corp. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ + +/*----------------------------------------------------------------------------- + This file is for OS related functions. +------------------------------------------------------------------------------*/ +#define _8192CD_OSDEP_C_ + +#ifdef __KERNEL__ +#include <linux/module.h> +#include <linux/kernel.h> +#include <linux/init.h> +#include <linux/ioport.h> +#include <linux/netdevice.h> +#include <linux/etherdevice.h> +#include <linux/in.h> +#include <linux/if.h> +#include <linux/ip.h> +#include <asm/io.h> +#include <linux/skbuff.h> +#include <linux/socket.h> +#include <linux/fs.h> +#include <linux/major.h> +#include <linux/fcntl.h> +#include <linux/signal.h> +#include <asm/uaccess.h> +#include <linux/sched.h> +#endif + +#ifdef __DRAYTEK_OS__ +#include <draytek/softimer.h> +#include <draytek/skbuff.h> +#include <draytek/wl_dev.h> +#endif + +#include "./8192cd_cfg.h" + +#ifdef __KERNEL__ +#ifdef __LINUX_2_6__ +#include <linux/syscalls.h> +#include <linux/file.h> +#include <asm/unistd.h> +#endif +#if defined(RTK_BR_EXT) || defined(BR_SHORTCUT) +#ifdef __LINUX_2_6__ +#include <linux/syscalls.h> +#else +#include <linux/fs.h> +#endif +#include <../net/bridge/br_private.h> +#endif +#else +#include "./sys-support.h" +#endif + +#include "./8192cd.h" +#include "./8192cd_hw.h" +#include "./8192cd_headers.h" +#include "./8192cd_rx.h" +#include "./8192cd_debug.h" + +#ifdef RTL8192CD_VARIABLE_USED_DMEM +#include "./8192cd_dmem.h" +#endif + +#ifdef CONFIG_RTL_92D_DMDP +u32 if_priv[NUM_WLAN_IFACE]; +#endif + +#ifdef CONFIG_RTK_MESH +#include "./mesh_ext/mesh_route.h" // Note : Not include in #ifdef CONFIG_RTK_MESH, Because use in wlan_device[] +#include "./mesh_ext/mesh_util.h" +#endif // CONFIG_RTK_MESH + +#ifdef CONFIG_RTL_KERNEL_MIPS16_WLAN +#include <asm/mips16_lib.h> +#endif + +#ifndef CONFIG_RTK_MESH //move below from mesh_route.h ; plus 0119 +#define MESH_SHIFT 0 +#define MESH_NUM_CFG 0 +#else +#define MESH_SHIFT 8 // ACCESS_MASK uses 2 bits, WDS_MASK use 4 bits +#define MESH_MASK 0x3 +#define MESH_NUM_CFG NUM_MESH +#endif + + +#if defined(CONFIG_RTL_8198) +#define CONFIG_RTL8198_REVISION_B 1 +//#define CONFIG_PHY_EAT_40MHZ 1 +#endif + +#ifdef CONFIG_RTL8672 +#include <platform.h> +#else +#if !defined(CONFIG_NET_PCI) && defined(CONFIG_RTL8196B) +#include <asm/rtl865x/platform.h> +#endif + +#if !defined(CONFIG_NET_PCI) && defined(CONFIG_RTL8196C) +#include <asm/rtl865x/platform.h> +#endif +#endif + +#if !defined(CONFIG_NET_PCI) && defined(CONFIG_RTL_819X) && defined(__LINUX_2_6__) +#if !defined(USE_RLX_BSP) +#include <platform.h> +#else +#include <bsp/bspchip.h> +#endif +#endif + +#if defined(CONFIG_RTL_WAPI_SUPPORT) +#include "wapiCrypto.h" +#endif +#ifdef CONFIG_RTL_CUSTOM_PASSTHRU //mark_test +#include <linux/proc_fs.h> +#endif +#ifdef CONFIG_WIRELESS_LAN_MODULE +extern int (*wirelessnet_hook)(void); +#ifdef BR_SHORTCUT +extern struct net_device* (*wirelessnet_hook_shortcut)(unsigned char *da); +#endif +#ifdef PERF_DUMP +extern int rtl8651_romeperfEnterPoint(unsigned int index); +extern int rtl8651_romeperfExitPoint(unsigned int index); +extern int (*Fn_rtl8651_romeperfEnterPoint)(unsigned int index); +extern int (*Fn_rtl8651_romeperfExitPoint)(unsigned int index); +#endif +#ifdef CONFIG_RTL8190_PRIV_SKB +extern int (*wirelessnet_hook_is_priv_buf)(void); +extern void (*wirelessnet_hook_free_priv_buf)(unsigned char *head); +#endif +#endif // CONFIG_WIRELESS_LAN_MODULE + +struct _device_info_ wlan_device[] = +{ +#if defined(USE_RTL8186_SDK) + //{(MESH_NUM_CFG<<MESH_SHIFT) |(WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_EMBEDDED<<TYPE_SHIFT), 0, 0xbd400000, 2, NULL}, + #ifdef CONFIG_NET_PCI + {(MESH_NUM_CFG<<MESH_SHIFT) |(WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_PCI_BIOS<<TYPE_SHIFT) | ACCESS_SWAP_MEM, 0, 0, 0, NULL}, + #else + #ifdef IO_MAPPING + {(WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_PCI_DIRECT<<TYPE_SHIFT) | ACCESS_SWAP_MEM, PCI_SLOT1_CONFIG_ADDR, 0xb8C00000, 3, NULL}, + #else + #if defined(CONFIG_RTL8196B) || defined(CONFIG_RTL8196C) || defined(CONFIG_RTL_819X) + #if defined(__LINUX_2_6__) && defined(USE_RLX_BSP) + #if defined(CONFIG_RTL_92D_SUPPORT) + #if (RTL_USED_PCIE_SLOT==0) + {(MESH_NUM_CFG<<MESH_SHIFT) |(WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_PCI_DIRECT<<TYPE_SHIFT) | ACCESS_SWAP_MEM, BSP_PCIE0_D_CFG0, BSP_PCIE0_D_MEM, BSP_PCIE_IRQ, NULL}, + #ifdef CONFIG_RTL_92D_DMDP + {(MESH_NUM_CFG<<MESH_SHIFT) |(WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_PCI_DIRECT<<TYPE_SHIFT) | ACCESS_SWAP_MEM, BSP_PCIE0_D_CFG0, BSP_PCIE0_F1_D_MEM, BSP_PCIE_IRQ, NULL}, + #endif + #elif (RTL_USED_PCIE_SLOT==1) + {(MESH_NUM_CFG<<MESH_SHIFT) |(WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_PCI_DIRECT<<TYPE_SHIFT) | ACCESS_SWAP_MEM, BSP_PCIE1_D_CFG0, BSP_PCIE1_D_MEM, BSP_PCIE2_IRQ, NULL}, + #ifdef CONFIG_RTL_92D_DMDP + {(MESH_NUM_CFG<<MESH_SHIFT) |(WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_PCI_DIRECT<<TYPE_SHIFT) | ACCESS_SWAP_MEM, BSP_PCIE1_D_CFG0, BSP_PCIE1_F1_D_MEM, BSP_PCIE2_IRQ, NULL}, + #endif + #if defined(CONFIG_RTL_DUAL_PCIESLOT_BIWLAN_D) || defined(CONFIG_RTL_DUAL_PCIESLOT_BIWLAN) + {(MESH_NUM_CFG<<MESH_SHIFT) |(WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_PCI_DIRECT<<TYPE_SHIFT) | ACCESS_SWAP_MEM, BSP_PCIE0_D_CFG0, BSP_PCIE0_D_MEM, BSP_PCIE_IRQ, NULL}, + #endif + + + + #else + #error "define pcie error" + #endif + #else + {(MESH_NUM_CFG<<MESH_SHIFT) |(WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_PCI_DIRECT<<TYPE_SHIFT) | ACCESS_SWAP_MEM, BSP_PCIE0_D_CFG0, BSP_PCIE0_D_MEM, BSP_PCIE_IRQ, NULL}, + #if defined(CONFIG_RTL_DUAL_PCIESLOT_BIWLAN) + {(MESH_NUM_CFG<<MESH_SHIFT) |(WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_PCI_DIRECT<<TYPE_SHIFT) | ACCESS_SWAP_MEM, BSP_PCIE1_D_CFG0, BSP_PCIE1_D_MEM, BSP_PCIE2_IRQ, NULL} + #endif + #endif +#else + {(MESH_NUM_CFG<<MESH_SHIFT) |(WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_PCI_DIRECT<<TYPE_SHIFT) | ACCESS_SWAP_MEM, PCIE0_D_CFG0, PCIE0_D_MEM, PCIE_IRQ, NULL}, + #ifdef CONFIG_RTL_92D_DMDP + {(MESH_NUM_CFG<<MESH_SHIFT) |(WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_PCI_DIRECT<<TYPE_SHIFT) | ACCESS_SWAP_MEM, PCIE0_D_CFG0, PCIE0_F1_D_MEM, PCIE_IRQ, NULL}, + #endif +#endif + //{(MESH_NUM_CFG<<MESH_SHIFT) |(WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_PCI_BIOS<<TYPE_SHIFT) | ACCESS_SWAP_MEM, 0, 0, 0, NULL}, + #else + {(MESH_NUM_CFG<<MESH_SHIFT) | (WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_PCI_DIRECT<<TYPE_SHIFT) | ACCESS_SWAP_MEM, PCI_SLOT1_CONFIG_ADDR, 0xb9000000, 3, NULL}, + #endif + #endif + #endif +#elif defined(NOT_RTK_BSP) + {(MESH_NUM_CFG<<MESH_SHIFT) |(WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_PCI_BIOS<<TYPE_SHIFT) | ACCESS_SWAP_MEM, 0, 0, 0, NULL} +#elif defined(CONFIG_RTL8671) + #ifdef IO_MAPPING + {(WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_PCI_DIRECT<<TYPE_SHIFT) | ACCESS_SWAP_MEM, PCI_SLOT0_CONFIG_ADDR, 0xbd200000, 7, NULL}, + #else + {(MESH_NUM_CFG<<MESH_SHIFT) |(WDS_NUM_CFG<<WDS_SHIFT) | (TYPE_PCI_DIRECT<<TYPE_SHIFT) | ACCESS_SWAP_MEM, PCI_SLOT0_CONFIG_ADDR, 0xbd300000, 7, NULL}, + #endif +#else + #error No_System_Defined +#endif +}; + +static int wlan_index=0; + +#ifdef CONFIG_RTL8672 +// Added by Mason Yu +// MBSSID Port Mapping +struct port_map wlanDev[RTL8192CD_NUM_VWLAN+1]; +static int wlanDevNum=0; +#endif + +#ifdef PRIV_STA_BUF +#ifdef CONCURRENT_MODE +static struct rtl8192cd_hw hw_info[NUM_WLAN_IFACE]; +static struct priv_shared_info shared_info[NUM_WLAN_IFACE]; +static struct wlan_hdr_poll hdr_pool[NUM_WLAN_IFACE]; +static struct wlanllc_hdr_poll llc_pool[NUM_WLAN_IFACE]; +static struct wlanbuf_poll buf_pool[NUM_WLAN_IFACE]; +static struct wlanicv_poll icv_pool[NUM_WLAN_IFACE]; +static struct wlanmic_poll mic_pool[NUM_WLAN_IFACE]; +static unsigned char desc_buf[NUM_WLAN_IFACE][DESC_DMA_PAGE_SIZE]; +static int wlandev_idx = 0; + +#else +static struct rtl8192cd_hw hw_info; +static struct priv_shared_info shared_info; +static struct wlan_hdr_poll hdr_pool; +static struct wlanllc_hdr_poll llc_pool; +static struct wlanbuf_poll buf_pool; +static struct wlanicv_poll icv_pool; +static struct wlanmic_poll mic_pool; +static unsigned char desc_buf[DESC_DMA_PAGE_SIZE]; +#endif +#endif + +// init and remove char device +#ifdef CONFIG_WIRELESS_LAN_MODULE +extern int rtl8192cd_chr_init(void); +extern void rtl8192cd_chr_exit(void); +#else +int rtl8192cd_chr_init(void); +void rtl8192cd_chr_exit(void); +#endif +struct rtl8192cd_priv *rtl8192cd_chr_reg(unsigned int minor, struct rtl8192cd_chr_priv *priv); +void rtl8192cd_chr_unreg(unsigned int minor); +int rtl8192cd_fileopen(const char *filename, int flags, int mode); + +__MIPS16 +__IRAM_IN_865X +struct net_device *get_shortcut_dev(unsigned char *da); + +void force_stop_wlan_hw(void); + + +#if defined(_INCLUDE_PROC_FS_) && defined(PERF_DUMP) +#include "romeperf.h" +static int read_perf_dump(struct file *file, const char *buffer, + unsigned long count, void *data) +{ + struct net_device *dev = (struct net_device *)data; + struct rtl8192cd_priv *priv = (struct rtl8192cd_priv *)dev->priv; + unsigned long x; + + SAVE_INT_AND_CLI(x); + + rtl8651_romeperfDump(1, 1); + + RESTORE_INT(x); + return count; +} + + +static int flush_perf_dump(struct file *file, const char *buffer, + unsigned long count, void *data) +{ + struct net_device *dev = (struct net_device *)data; + struct rtl8192cd_priv *priv = (struct rtl8192cd_priv *)dev->priv; + unsigned long x; + + SAVE_INT_AND_CLI(x); + + rtl8651_romeperfReset(); + + RESTORE_INT(x); + return count; +} +#endif // _INCLUDE_PROC_FS_ && PERF_DUMP + +#define GPIO_BASE 0xB8003500 +#define PEFGHCNR_REG (0x01C + GPIO_BASE) /* Port EFGH control */ +#define PEFGHPTYPE_REG (0x020 + GPIO_BASE) /* Port EFGH type */ +#define PEFGHDIR_REG (0x024 + GPIO_BASE) /* Port EFGH direction */ +#define PEFGHDAT_REG (0x028 + GPIO_BASE) /* Port EFGH data */ +#ifndef REG32 + #define REG32(reg) (*(volatile unsigned int *)(reg)) +#endif + +#ifdef CONFIG_RTL_92D_DMDP +void Sw_PCIE_Func(int func) +{ +#if (RTL_USED_PCIE_SLOT==1) + REG32(0xb8b2100c)=REG32(0xb8b2100c)|func; // switch to function # +#else + REG32(0xb8b0100c)=REG32(0xb8b0100c)|func; // switch to function # +#endif +} +#endif +#if !defined(CONFIG_NET_PCI) && (defined(CONFIG_RTL8196B) || defined(CONFIG_RTL_819X)) +#define MAX_PAYLOAD_SIZE_128B 0x00 + +#if !(defined(CONFIG_RTL8196C) || defined(CONFIG_RTL_8196C)) +#ifdef CONFIG_RTL_KERNEL_MIPS16_WLAN +__NOMIPS16 +#endif +static int rtl8196b_pci_reset(unsigned long conf_addr) +{ + /* If PCI needs to be reset, put code here. + * Note: + * Software may need to do hot reset for a period of time, say ~100us. + * Here we put 2ms. + */ + //Modified for PCIE PHY parameter due to RD center suggestion by Jason 12252009 + WRITE_MEM32(0xb8000044, 0x9);//Enable PCIE PLL + delay_ms(10); + REG32(0xb8000010)=REG32(0xb8000010)|(0x500); //Active LX & PCIE Clock in 8196B system register + delay_ms(10); + WRITE_MEM32(0xb800003C, 0x1);//PORT0 PCIE PHY MDIO Reset + delay_ms(10); + WRITE_MEM32(0xb800003C, 0x3);//PORT0 PCIE PHY MDIO Reset + delay_ms(10); + WRITE_MEM32(0xb8000040, 0x1);//PORT1 PCIE PHY MDIO Reset + delay_ms(10); + WRITE_MEM32(0xb8000040, 0x3);//PORT1 PCIE PHY MDIO Reset + delay_ms(10); + WRITE_MEM32(0xb8b01008, 0x1);// PCIE PHY Reset Close:Port 0 + delay_ms(10); + WRITE_MEM32(0xb8b01008, 0x81);// PCIE PHY Reset On:Port 0 + delay_ms(10); +#ifdef PIN_208 + WRITE_MEM32(0xb8b21008, 0x1);// PCIE PHY Reset Close:Port 1 + delay_ms(10); + WRITE_MEM32(0xb8b21008, 0x81);// PCIE PHY Reset On:Port 1 + delay_ms(10); +#endif +#ifdef OUT_CYSTALL + WRITE_MEM32(0xb8b01000, 0xcc011901);// PCIE PHY Reset On:Port 0 + delay_ms(10); +#ifdef PIN_208 + WRITE_MEM32(0xb8b21000, 0xcc011901);// PCIE PHY Reset On:Port 1 + delay_ms(10); +#endif +#endif + REG32(0xb8000010)=REG32(0xb8000010)|(0x01000000); //PCIE PHY Reset On:Port 0 + delay_ms(10); + +#if defined(__LINUX_2_6__) && defined(USE_RLX_BSP) + WRITE_MEM32(BSP_PCIE0_H_PWRCR, READ_MEM32(BSP_PCIE0_H_PWRCR) & 0xFFFFFF7F); +#ifdef PIN_208 + WRITE_MEM32(BSP_PCIE1_H_PWRCR, READ_MEM32(BSP_PCIE1_H_PWRCR) & 0xFFFFFF7F); +#endif + delay_ms(100); + WRITE_MEM32(BSP_PCIE0_H_PWRCR, READ_MEM32(BSP_PCIE0_H_PWRCR) | 0x00000080); +#ifdef PIN_208 + WRITE_MEM32(BSP_PCIE1_H_PWRCR, READ_MEM32(BSP_PCIE1_H_PWRCR) | 0x00000080); +#endif +#else + WRITE_MEM32(PCIE0_H_PWRCR, READ_MEM32(PCIE0_H_PWRCR) & 0xFFFFFF7F); +#ifdef PIN_208 + WRITE_MEM32(PCIE1_H_PWRCR, READ_MEM32(PCIE1_H_PWRCR) & 0xFFFFFF7F); +#endif + delay_ms(100); + WRITE_MEM32(PCIE0_H_PWRCR, READ_MEM32(PCIE0_H_PWRCR) | 0x00000080); +#ifdef PIN_208 + WRITE_MEM32(PCIE1_H_PWRCR, READ_MEM32(PCIE1_H_PWRCR) | 0x00000080); +#endif +#endif + + delay_ms(10); + + if ((READ_MEM32(0xb8b00728)&0x1f)!=0x11) + { + _DEBUG_INFO("PCIE LINK FAIL\n"); + return FAIL; + } + + // Enable PCIE host +#if defined(__LINUX_2_6__) && defined(USE_RLX_BSP) + WRITE_MEM32(BSP_PCIE0_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(BSP_PCIE0_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B +#else + WRITE_MEM32(PCIE0_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(PCIE0_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B +#endif + return SUCCESS; +} +#endif // !defined(CONFIG_NET_PCI) && (defined(CONFIG_RTL8196B) || defined(CONFIG_RTL_819X)) +#endif + + +#if defined(CONFIG_RTL_8198) || defined(CONFIG_RTL8196C) || defined(CONFIG_RTL_8196C) +#define MAX_PAYLOAD_SIZE_128B 0x00 + +#define CLK_MANAGE 0xb8000010 +#define PCIE0_RC_EXT_BASE (0xb8b01000) +#define PCIE1_RC_EXT_BASE (0xb8b21000) +//RC Extended register +#define PCIE0_MDIO (PCIE0_RC_EXT_BASE+0x00) +#define PCIE1_MDIO (PCIE1_RC_EXT_BASE+0x00) +//MDIO +#define PCIE_MDIO_DATA_OFFSET (16) +#define PCIE_MDIO_DATA_MASK (0xffff <<PCIE_MDIO_DATA_OFFSET) +#define PCIE_MDIO_REG_OFFSET (8) +#define PCIE_MDIO_RDWR_OFFSET (0) +#if !defined(CONFIG_NET_PCI) && defined(CONFIG_RTL_8198) +#define PHY_EAT_40MHZ 1 +#endif + +#if !defined(CONFIG_NET_PCI) && !defined(CONFIG_PCI) +#if defined(__LINUX_2_6__) && defined(USE_RLX_BSP) +#define PCIE0_H_CFG BSP_PCIE0_H_CFG +#define PCIE1_H_CFG BSP_PCIE1_H_CFG +#endif + +void HostPCIe_SetPhyMdioWrite(unsigned int portnum, unsigned int regaddr, unsigned short val) +{ + unsigned int mdioaddr; + volatile int i; + + if(portnum==0) mdioaddr=PCIE0_MDIO; + else if(portnum==1) mdioaddr=PCIE1_MDIO; + else return; + + REG32(mdioaddr)= ( (regaddr&0x1f)<<PCIE_MDIO_REG_OFFSET) | ((val&0xffff)<<PCIE_MDIO_DATA_OFFSET) | (1<<PCIE_MDIO_RDWR_OFFSET) ; + //delay + for(i=0;i<5555;i++) ; + mdelay(1); +} +#if defined(CONFIG_RTL_8196CS) || defined(CONFIG_RTL_8197B) +static void GPIO6_PCIE_Device_PERST(void) + +{ + +#if defined(CONFIG_RTL_8197B) + REG32(0xb8000040)=REG32(0xb8000040)|0x00018; + + REG32(0xb8003500)=REG32(0xb8003500)&(~0x10); + + REG32(0xb8003508)= REG32(0xb8003508)|0x10; + + REG32(0xb800350c)= REG32(0xb800350c)|0x10; + + delay_ms(500); + + delay_ms(500); + + + + // 6. PCIE Device Reset + + //REG32(CLK_MANAGE) &= ~(1<<26); //perst=0 off. + + REG32(0xb800350c)= REG32(0xb800350c)&(~0x10); + + delay_ms(500); //PCIE standadrd: poweron: 100us, after poweron: 100ms + + delay_ms(500); + + REG32(0xb800350c)= REG32(0xb800350c)|0x10; +#else + REG32(0xb8000040)=REG32(0xb8000040)|0x300000; + + REG32(0xb8003500)=REG32(0xb8003500)&(~0x40); + + REG32(0xb8003508)= REG32(0xb8003508)|0x40; + + REG32(0xb800350c)= REG32(0xb800350c)|0x40; + + delay_ms(500); + + delay_ms(500); + + + + // 6. PCIE Device Reset + + //REG32(CLK_MANAGE) &= ~(1<<26); //perst=0 off. + + REG32(0xb800350c)= REG32(0xb800350c)&(~0x40); + + delay_ms(500); //PCIE standadrd: poweron: 100us, after poweron: 100ms + + delay_ms(500); + + REG32(0xb800350c)= REG32(0xb800350c)|0x40; + + // REG32(CLK_MANAGE) |= (1<<26); //PERST=1 +#endif +} +#endif +#ifdef CONFIG_RTL_8196C_iNIC //mark_inic +static void iNIC_PCIE_Device_PERST(void) + +{ + + REG32(0xb8000040)=REG32(0xb8000040)|0x300000; + + REG32(0xb8003500)=REG32(0xb8003500)&(~0x40); + + REG32(0xb8003508)= REG32(0xb8003508)|0x40; + + REG32(0xb800350c)= REG32(0xb800350c)|0x40; + + delay_ms(500); + + delay_ms(500); + + + + // 6. PCIE Device Reset + + //REG32(CLK_MANAGE) &= ~(1<<26); //perst=0 off. + + REG32(0xb800350c)= REG32(0xb800350c)&(~0x40); + + delay_ms(500); //PCIE standadrd: poweron: 100us, after poweron: 100ms + + delay_ms(500); + + REG32(0xb800350c)= REG32(0xb800350c)|0x40; + + // REG32(CLK_MANAGE) |= (1<<26); //PERST=1 + +} +#endif + +#if defined(CONFIG_RTL_8198) +static int at2_mode=0; +static void PCIE_Device_PERST(int portnum) +{ + if (portnum==0) + { + REG32(CLK_MANAGE) &= ~(1<<26); //perst=0 off. + mdelay(500); //PCIE standadrd: poweron: 100us, after poweron: 100ms + mdelay(500); + REG32(CLK_MANAGE) |= (1<<26); //PERST=1 + } + else if (portnum==1) + { + /* PCIE Device Reset + * The pcei1 slot reset register depends on the hw + */ + +#if defined(CONFIG_RTL_DUAL_PCIESLOT_BIWLAN) || (RTL_USED_PCIE_SLOT==1) || defined(CONFIG_RTL_DUAL_PCIESLOT_BIWLAN_D) +#if defined(CONFIG_RTL_8197B) + GPIO6_PCIE_Device_PERST(); +#else + REG32(PEFGHDAT_REG) &= ~(0x1000); //perst=0 off. + mdelay(300); //PCIE standadrd: poweron: 100us, after poweron: 100ms + mdelay(300); + REG32(PEFGHDAT_REG) |= (0x1000); //PERST=1 +#endif +#elif defined(CONFIG_RTL_92D_SUPPORT) + REG32(CLK_MANAGE) &= ~(1<<26); //perst=0 off. + mdelay(500); //PCIE standadrd: poweron: 100us, after poweron: 100ms + mdelay(500); + REG32(CLK_MANAGE) |= (1<<26); //PERST=1 +#endif + } + else + return; +} + +void PCIE_MDIO_Reset(unsigned int portnum) +{ + #define SYS_PCIE_PHY0 (0xb8000000 +0x50) + #define SYS_PCIE_PHY1 (0xb8000000 +0x54) + + unsigned int sys_pcie_phy; + + if(portnum==0) sys_pcie_phy=SYS_PCIE_PHY0; + else if(portnum==1) sys_pcie_phy=SYS_PCIE_PHY1; + else return; + + // 3.MDIO Reset + REG32(sys_pcie_phy) = (1<<3) |(0<<1) | (0<<0); //mdio reset=0, + REG32(sys_pcie_phy) = (1<<3) |(0<<1) | (1<<0); //mdio reset=1, + REG32(sys_pcie_phy) = (1<<3) |(1<<1) | (1<<0); //bit1 load_done=1 +} + +void PCIE_PHY_Reset(unsigned int portnum) +{ + #define PCIE_PHY0 0xb8b01008 + #define PCIE_PHY1 0xb8b21008 + + unsigned int pcie_phy; + + if(portnum==0) pcie_phy=BSP_PCIE0_H_PWRCR; + else if(portnum==1) pcie_phy=BSP_PCIE1_H_PWRCR; + else return; + + //4. PCIE PHY Reset + REG32(pcie_phy) = 0x01; //bit7:PHY reset=0 bit0: Enable LTSSM=1 + REG32(pcie_phy) = 0x81; //bit7: PHY reset=1 bit0: Enable LTSSM=1 +} + +int PCIE_Check_Link(unsigned int portnum) +{ + unsigned int dbgaddr; + unsigned int cfgaddr=0; + int i=10; + + if(portnum==0) dbgaddr=0xb8b00728; + else if(portnum==1) dbgaddr=0xb8b20728; + else return 1; + + + //wait for LinkUP + + while(--i) + { + if( (REG32(dbgaddr)&0x1f)==0x11) + break; + + mdelay(100); + } + + if(i==0) + { if(at2_mode==0) //not auto test, show message + printk("i=%x Cannot LinkUP \n",i); + return 0; + } + else + { + if(portnum==0) cfgaddr=0xb8b10000; + else if(portnum==1) cfgaddr=0xb8b30000; + + if(at2_mode==0) + printk("Find Port=%x Device:Vender ID=%x\n", portnum, REG32(cfgaddr) ); + } + return 1; +} + +int PCIE_reset_procedure(int portnum, int Use_External_PCIE_CLK, int mdio_reset, unsigned long conf_addr) +{ + static int pcie_reset_done[RTL_MAX_PCIE_SLOT_NUM] = {0}; + int status=FAIL; + + + #if defined(CONFIG_RTL_DUAL_PCIESLOT_BIWLAN) || (RTL_USED_PCIE_SLOT==1) || defined(CONFIG_RTL_DUAL_PCIESLOT_BIWLAN_D) + if (portnum==0) + { + REG32(CLK_MANAGE) |= (1<<26); //PERST=1 + } + else if (portnum==1) + { + #if defined(CONFIG_RTL_8197B) + REG32(0xb8000040)|=0x18; + REG32(0xb8003500) &= ~(0x10); /*port Abit 4 */ + REG32(0xb8003508) |= (0x10); /*port A bit 4 */ + REG32(0xb800350c) |= (0x10); //PERST=1 + + #else + REG32(0xb8000040)|=0x300; + REG32(PEFGHCNR_REG) &= ~(0x1000); /*port F bit 4 */ + REG32(PEFGHDIR_REG) |= (0x1000); /*port F bit 4 */ + REG32(PEFGHDAT_REG) |= (0x1000); //PERST=1 + #endif + } + #endif + + if (pcie_reset_done[portnum]) + goto SET_BAR; + + printk("PCIE reset (%d) \n", pcie_reset_done[portnum]); + if(portnum==0) REG32(CLK_MANAGE) |= (1<<14); //enable active_pcie0 + else if(portnum==1) REG32(CLK_MANAGE) |= (1<<16); //enable active_pcie1 + else return 0; + + mdelay(500); + + #ifdef CONFIG_RTL8198_REVISION_B + if(portnum==1) + { + #define PAD_CONTROL 0xb8000048 + REG32(PAD_CONTROL)|=(1<<27); + } + #endif + mdelay(500); + + + if(mdio_reset) + { + if(at2_mode==0) //no auto test, show message + printk("Do MDIO_RESET\n"); + + // 3.MDIO Reset + PCIE_MDIO_Reset(portnum); + } + + /* + PCIE_PHY_Reset(portnum); + */ + mdelay(500); + mdelay(500); + + //---------------------------------------- + if(mdio_reset) + { + //fix 8198 test chip pcie tx problem. + #ifdef CONFIG_RTL8198_REVISION_B + if (REG32(BSP_REVR) >= BSP_RTL8198_REVISION_B) + { + HostPCIe_SetPhyMdioWrite(portnum, 0, 0xD087); //bokai tell, and fix + + HostPCIe_SetPhyMdioWrite(portnum, 1, 0x0003); + HostPCIe_SetPhyMdioWrite(portnum, 2, 0x4d18); + #if CONFIG_PHY_EAT_40MHZ + HostPCIe_SetPhyMdioWrite(portnum, 5, 0x0BCB); //40M + #endif + + #if CONFIG_PHY_EAT_40MHZ + printk("98 - 40MHz Clock Source\n"); + HostPCIe_SetPhyMdioWrite(portnum, 6, 0xF148); //40M + #else + printk("98 - 25MHz Clock Source\n"); + HostPCIe_SetPhyMdioWrite(portnum, 6, 0xf848); //25M + #endif + + HostPCIe_SetPhyMdioWrite(portnum, 7, 0x31ff); + HostPCIe_SetPhyMdioWrite(portnum, 8, 0x18d7); //peisi tune + + //saving more power, 8196c pe-si tune + HostPCIe_SetPhyMdioWrite(portnum, 0x09, 0x539c); + HostPCIe_SetPhyMdioWrite(portnum, 0x0a, 0x20eb); + HostPCIe_SetPhyMdioWrite(portnum, 0x0d, 0x1766); + + HostPCIe_SetPhyMdioWrite(portnum, 0x0b, 0x0511); //for sloving low performance + + + HostPCIe_SetPhyMdioWrite(portnum, 0xf, 0x0a00); + HostPCIe_SetPhyMdioWrite(portnum, 0x19, 0xFCE0); + + HostPCIe_SetPhyMdioWrite(portnum, 0x1a, 0x7e40); //formal chip, reg 0x1a.4=0 + HostPCIe_SetPhyMdioWrite(portnum, 0x1b, 0xFC01); //formal chip reg 0x1b.0=1 + + HostPCIe_SetPhyMdioWrite(portnum, 0x1e, 0xC280); + } + else + #endif + { + HostPCIe_SetPhyMdioWrite(portnum, 0, 0xD087); + + HostPCIe_SetPhyMdioWrite(portnum, 1, 0x0003); + HostPCIe_SetPhyMdioWrite(portnum, 6, 0xf448); //new + HostPCIe_SetPhyMdioWrite(portnum, 6, 0x408); //avoid noise infuse //15-12=0, 7-5=0, 0448 + + HostPCIe_SetPhyMdioWrite(portnum, 7, 0x31ff); + HostPCIe_SetPhyMdioWrite(portnum, 8, 0x18d5); //new + HostPCIe_SetPhyMdioWrite(portnum, 9, 0x531c); + + HostPCIe_SetPhyMdioWrite(portnum, 0xd, 0x1766); + HostPCIe_SetPhyMdioWrite(portnum, 0xf, 0x0010);//ori + + HostPCIe_SetPhyMdioWrite(portnum, 0x19, 0xFCE0); + HostPCIe_SetPhyMdioWrite(portnum, 0x1e, 0xC280); + } + } + + //--------------------------------------- + PCIE_Device_PERST(portnum); + PCIE_PHY_Reset(portnum); + mdelay(500); + mdelay(500); + status=PCIE_Check_Link(portnum); + if(status==FAIL) + return FAIL; + + +SET_BAR: + if (portnum==0) + { + // Enable PCIE host + if (pcie_reset_done[portnum] == 0) { +#if defined(__LINUX_2_6__) && defined(USE_RLX_BSP) + WRITE_MEM32(BSP_PCIE0_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(BSP_PCIE0_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B +#else + WRITE_MEM32(PCIE0_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(PCIE0_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B +#endif + pcie_reset_done[portnum] = 1; + } +#ifdef CONFIG_RTL_92D_DMDP + else { + Sw_PCIE_Func(1); + //choose the PCIE port number + WRITE_MEM32(PCIE0_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(PCIE0_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B + Sw_PCIE_Func(0); + } +#endif + } + else if (portnum==1) + { + // Enable PCIE host +#if defined(__LINUX_2_6__) && defined(USE_RLX_BSP) + if (pcie_reset_done[portnum] == 0) { + WRITE_MEM32(BSP_PCIE1_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(BSP_PCIE1_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B + pcie_reset_done[portnum] = 1; + } + #ifdef CONFIG_RTL_92D_DMDP + else { + Sw_PCIE_Func(1); + //choose the PCIE port number + WRITE_MEM32(BSP_PCIE1_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(BSP_PCIE1_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B + Sw_PCIE_Func(0); + } + #endif + #else /* defined(__LINUX_2_6__) && defined(USE_RLX_BSP) */ + WRITE_MEM32(PCIE1_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(PCIE1_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B + #endif /* defined(__LINUX_2_6__) && defined(USE_RLX_BSP) */ + } + else + return FAIL; + + mdelay(500); + return SUCCESS; + +} +#elif defined(CONFIG_RTL8196C) || defined(CONFIG_RTL_8196C) + +#ifdef CONFIG_RTL8672 +int PCIE_reset_procedure(int PCIE_Port0and1_8196B_208pin, int Use_External_PCIE_CLK, int mdio_reset,unsigned long conf_addr) +{ + #define SYS_PCIE_PHY0 0xb8003400 + #define Module_Enable 0xb800330c + //PCIE Register + #define CLK_MANAGE 0xb800350c + #define PCIE_PHY0_REG 0xb8b01000 + //#define PCIE_PHY1_REG 0xb8b21000 + #define PCIE_PHY0 0xb8b01008 + // #define PCIE_PHY1 0xb8b21008 + #define PCIE_gpio_RST 5 + + //1. PCIE phy mdio reset + REG32(SYS_PCIE_PHY0) = 0x1d400000; + REG32(SYS_PCIE_PHY0) = 0x1d500000; + + //2. PCIE MAC reset + REG32(Module_Enable) &= ~(1<<9); + REG32(Module_Enable) |= (1<<9); + + //3.Active LX & PCIE Clock + gpioSet(PCIE_gpio_RST); + delay_ms(100); + + //4. GPIO output + REG32(0xb8003508) |= (1<<5); + delay_ms(100); + + if(mdio_reset) + { + //printk("Do MDIO_RESET\n"); + // 5.MDIO Reset + REG32(SYS_PCIE_PHY0) = 0x1d500000; + } + //6. PCIE PHY Reset + REG32(PCIE_PHY0) = 0x1; //bit7 PHY reset=0 bit0 Enable LTSSM=1 + REG32(PCIE_PHY0) = 0x81; //bit7 PHY reset=1 bit0 Enable LTSSM=1 + delay_ms(100); + + delay_ms(100); + + //---------------------------------------- + if(mdio_reset) + { + int port=0; + HostPCIe_SetPhyMdioWrite(port, 0, 0x5027); + HostPCIe_SetPhyMdioWrite(port, 2, 0x6d18); + //HostPCIe_SetPhyMdioWrite(port, 6, 0x8028); + if( (IS_RLE0315 ||IS_6166) && (REG32(MISC_PINSR)&(0x200000))){ + HostPCIe_SetPhyMdioWrite(port, 6, 0x20c8); //35.328 clock source + HostPCIe_SetPhyMdioWrite(port, 5, 0x0bcb); + printk("Clock source is 35.328MHz\n"); + }else if( (IS_RLE0315 ||IS_6166) && ~(REG32(MISC_PINSR)&(0x200000))){ + HostPCIe_SetPhyMdioWrite(port, 6, 0xf848); //25M clock source + HostPCIe_SetPhyMdioWrite(port, 5, 0x08ab); + printk("Clock source is 25MHz\n"); + } + HostPCIe_SetPhyMdioWrite(port, 7, 0x30ff); + //HostPCIe_SetPhyMdioWrite(port, 8, 0x18dd); + HostPCIe_SetPhyMdioWrite(port, 8, 0x18d7); + HostPCIe_SetPhyMdioWrite(port, 0xa, 0xe9); + HostPCIe_SetPhyMdioWrite(port, 0xb, 0x0511); + //HostPCIe_SetPhyMdioWrite(port, 0xd, 0x15b6); + HostPCIe_SetPhyMdioWrite(port, 0xd, 0x15a6); + HostPCIe_SetPhyMdioWrite(port, 0xf, 0x0f0f); + //HostPCIe_SetPhyMdioWrite(port, 0x9, 0xe950); //czyao adds to leave L0 mode + } + + //--------------------------------------- + //delay_ms(10000); + + // 7. PCIE Device Reset + gpioClear(PCIE_gpio_RST); + delay_ms(100); + + gpioSet(PCIE_gpio_RST); + + //4. PCIE PHY Reset + REG32(PCIE_PHY0) = 0x1; //bit7 PHY reset=0 bit0 Enable LTSSM=1 + REG32(PCIE_PHY0) = 0x81; //bit7 PHY reset=1 bit0 Enable LTSSM=1 + delay_ms(100); + + //8. Set BAR + REG32(0xb8b10010) = 0x18c00001; + REG32(0xb8b10018) = 0x19000004; + REG32(0xb8b10004) = 0x00180007; + REG32(0xb8b00004) = 0x00100007; + + //8. Fine tune parameter, and depends on RD center's suggestion + //REG32(0xb8003204) = 0x4cc3106d; //This is added in bootcode +#if 1 //wait for LinkUP + int i=100; + while(--i) + { + if( (REG32(0xb8b00728)&0x1f)==0x11) + break; + delay_ms(100); + } + if(i==0) + { + printk("i=%x Cannot LinkUP \n",i); + return FAIL; + } +#endif + // Enable PCIE host +#if defined(__LINUX_2_6__) && defined(USE_RLX_BSP) + WRITE_MEM32(BSP_PCIE0_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(BSP_PCIE0_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B +#else + WRITE_MEM32(PCIE0_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(PCIE0_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B +#endif + return SUCCESS; +} +#else /* defined(CONFIG_RTL8672) */ +static int pcie_reset_done = 0; +int PCIE_reset_procedure(int PCIE_Port0and1_8196B_208pin, int Use_External_PCIE_CLK, int mdio_reset,unsigned long conf_addr) +{ + + int i=100; +#define SYS_PCIE_PHY0 (0xb8000000 +0x50) + //PCIE Register +#define CLK_MANAGE 0xb8000010 +#define PCIE_PHY0_REG 0xb8b01000 + //#define PCIE_PHY1_REG 0xb8b21000 +#define PCIE_PHY0 0xb8b01008 + // #define PCIE_PHY1 0xb8b21008 + int port =0; + + if (pcie_reset_done) + goto SET_BAR; + + //2.Active LX & PCIE Clock + REG32(CLK_MANAGE) |= (1<<11); //enable active_pcie0 + delay_ms(100); + +#if 1 + if(mdio_reset) + { + //printk("Do MDIO_RESET\n"); + // 3.MDIO Reset + REG32(SYS_PCIE_PHY0) = (1<<3) |(0<<1) | (0<<0); //mdio reset=0, + REG32(SYS_PCIE_PHY0) = (1<<3) |(0<<1) | (1<<0); //mdio reset=1, + REG32(SYS_PCIE_PHY0) = (1<<3) |(1<<1) | (1<<0); //bit1 load_done=1 + } + //4. PCIE PHY Reset + REG32(PCIE_PHY0) = 0x1; //bit7 PHY reset=0 bit0 Enable LTSSM=1 + REG32(PCIE_PHY0) = 0x81; //bit7 PHY reset=1 bit0 Enable LTSSM=1 + delay_ms(100); +#endif + + delay_ms(100); + + //---------------------------------------- + if(mdio_reset) + { + if (REG32(REVR) == RTL8196C_REVISION_A) + { + HostPCIe_SetPhyMdioWrite(port, 0, 0x5027); + HostPCIe_SetPhyMdioWrite(port, 2, 0x6d18); + HostPCIe_SetPhyMdioWrite(port, 6, 0x8828); + HostPCIe_SetPhyMdioWrite(port, 7, 0x30ff); + HostPCIe_SetPhyMdioWrite(port, 8, 0x18d7); + HostPCIe_SetPhyMdioWrite(port, 0xa, 0xe9); + HostPCIe_SetPhyMdioWrite(port, 0xb, 0x0511); + HostPCIe_SetPhyMdioWrite(port, 0xd, 0x15b6); + HostPCIe_SetPhyMdioWrite(port, 0xf, 0x0f0f); +#if 1 // PHY_EAT_40MHZ + HostPCIe_SetPhyMdioWrite(port, 5, 0xbcb); //[9:3]=1111001 (binary) 121 (10) + HostPCIe_SetPhyMdioWrite(port, 6, 0x8128); //[11]=0 [9:8]=01 +#endif + /* + emdiow 0 5027 + emdiow 2 6d18 + emdiow 6 8828 + emdiow 7 30ff + emdiow 8 18dd + emdiow a e9 + emdiow b 0511 + emdiow d 15b6 + emdiow f 0f0f + */ + } + else + { + HostPCIe_SetPhyMdioWrite(port, 0, 0xD087); + HostPCIe_SetPhyMdioWrite(port, 1, 0x0003); + HostPCIe_SetPhyMdioWrite(port, 2, 0x4d18); + +#ifdef CONFIG_PHY_EAT_40MHZ + + printk("96C - 40MHz Clock Source\n"); +#ifdef HIGH_POWER_EXT_PA + HostPCIe_SetPhyMdioWrite(port, 5, 0x0BF3); //40M +#else + HostPCIe_SetPhyMdioWrite(port, 5, 0x0BCB); //40M +#endif + HostPCIe_SetPhyMdioWrite(port, 6, 0xF148); //40M + +#else + printk("96C - 25MHz Clock Source\n"); + HostPCIe_SetPhyMdioWrite(port, 6, 0xf848); //25M +#endif + + HostPCIe_SetPhyMdioWrite(port, 7, 0x31ff); + HostPCIe_SetPhyMdioWrite(port, 8, 0x18d7); + HostPCIe_SetPhyMdioWrite(port, 9, 0x539c); + HostPCIe_SetPhyMdioWrite(port, 0xa, 0x20eb); + HostPCIe_SetPhyMdioWrite(port, 0xb, 0x0511); + HostPCIe_SetPhyMdioWrite(port, 0xd, 0x1764); + HostPCIe_SetPhyMdioWrite(port, 0xf, 0x0a00); + +#ifdef HAVING_FIB + HostPCIe_SetPhyMdioWrite(port,8, 0x18dd); + HostPCIe_SetPhyMdioWrite(port, 0xd, 0x1776); +#endif + + HostPCIe_SetPhyMdioWrite(port, 0x19, 0xFCE0); + HostPCIe_SetPhyMdioWrite(port, 0x1e, 0xC280); + } + } + + //--------------------------------------- + // 6. PCIE Device Reset +#ifndef CONFIG_RTL_8196C_iNIC + REG32(CLK_MANAGE) &= ~(1<<12); //perst=0 off. + delay_ms(300); +#endif + //mark_inic , from jason patch +#ifdef CONFIG_RTL_8196C_iNIC + iNIC_PCIE_Device_PERST(); +#endif +#if defined(CONFIG_RTL_8196CS) || defined(CONFIG_RTL_8197B) + GPIO6_PCIE_Device_PERST(); +#endif + //4. PCIE PHY Reset + REG32(PCIE_PHY0) = 0x1; //bit7 PHY reset=0 bit0 Enable LTSSM=1 + REG32(PCIE_PHY0) = 0x81; //bit7 PHY reset=1 bit0 Enable LTSSM=1 + delay_ms(300); +#ifndef CONFIG_RTL_8196C_iNIC + REG32(CLK_MANAGE) |= (1<<12); //PERST=1 +#endif + //4. PCIE PHY Reset + + //prom_printf("\nCLK_MANAGE(0x%x)=0x%x\n\n",CLK_MANAGE,READ_MEM32(CLK_MANAGE)); + delay_ms(100); +#if 1 //wait for LinkUP + while(--i) + { + if( (REG32(0xb8b00728)&0x1f)==0x11) + break; + delay_ms(100); + } + if(i==0) + { + printk("i=%x Cannot LinkUP \n",i); + return FAIL; + } +#endif + +SET_BAR: + if (port==0) + { + // Enable PCIE host + if (pcie_reset_done == 0) { +#if defined(__LINUX_2_6__) && defined(USE_RLX_BSP) + WRITE_MEM32(BSP_PCIE0_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(BSP_PCIE0_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B +#else + WRITE_MEM32(PCIE0_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(PCIE0_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B +#endif + pcie_reset_done = 1; + } +#ifdef CONFIG_RTL_92D_DMDP + else { + Sw_PCIE_Func(1); + //choose the PCIE port number +#if defined(__LINUX_2_6__) && defined(USE_RLX_BSP) + WRITE_MEM32(BSP_PCIE0_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(BSP_PCIE0_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B +#else + WRITE_MEM32(PCIE0_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(PCIE0_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B +#endif + Sw_PCIE_Func(0); + } +#endif + } + else if (port==1) + { + // Enable PCIE host + if (pcie_reset_done == 0) { +#if defined(__LINUX_2_6__) && defined(USE_RLX_BSP) + WRITE_MEM32(BSP_PCIE1_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(BSP_PCIE1_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B +#else + WRITE_MEM32(PCIE1_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(PCIE1_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B +#endif + pcie_reset_done = 1; + } +#ifdef CONFIG_RTL_92D_DMDP + else { + Sw_PCIE_Func(1); +#if defined(__LINUX_2_6__) && defined(USE_RLX_BSP) + //choose the PCIE port number + WRITE_MEM32(BSP_PCIE1_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(BSP_PCIE1_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B +#else + WRITE_MEM32(PCIE1_H_CFG + 0x04, 0x00100007); + WRITE_MEM8(PCIE1_H_CFG + 0x78, (READ_MEM8(conf_addr + 0x78) & (~0xE0)) | MAX_PAYLOAD_SIZE_128B); // Set MAX_PAYLOAD_SIZE to 128B +#endif + Sw_PCIE_Func(0); + } +#endif + } + else + return FAIL; + return SUCCESS; +} + +#endif /* defined(CONFIG_RTL8672) */ + +void HostPCIe_Close(void) +{ + REG32(0xb8b10044) &= (~(3)); + REG32(0xb8b10044) |= (3); + HostPCIe_SetPhyMdioWrite(0, 0xf, 0x0708); + //.DeActive LX & PCIE Clock + REG32(CLK_MANAGE) &=(0xFFFFFFFF- (1<<11)); //enable active_pcie0 + pcie_reset_done=0; +} +#endif /* #elif (defined(CONFIG_RTL8196C) || defined(CONFIG_RTL_8196C)) */ + +#endif /* !defined(CONFIG_NET_PCI) */ +#endif +static void rtl8192cd_bcnProc(struct rtl8192cd_priv *priv, unsigned int bcnInt, + unsigned int bcnDmaOk, unsigned int bcnOk, unsigned int bcnErr, unsigned int status) +{ +#ifdef MBSSID + int i; +#endif + + /* ================================================================ + Process Beacon interrupt + ================================================================ */ + // + // Update beacon content + // + if (bcnInt) { + unsigned char val8; + if (status & HIMR_BCNDMA0) { +#ifdef UNIVERSAL_REPEATER + if ((OPMODE & WIFI_STATION_STATE) && GET_VXD_PRIV(priv) && + (GET_VXD_PRIV(priv)->drv_state & DRV_STATE_VXD_AP_STARTED)) { + if (GET_VXD_PRIV(priv)->timoffset) { + update_beacon(GET_VXD_PRIV(priv)); + } + } else +#endif + { +#if 0 //TESTCHIP_SUPPORT +#ifdef SMART_CONCURRENT_92D // -- fwdebug + if (priv->pshare->wlandev_idx == 1) { + DMDP_RTL_W8(0, 0x663, DMDP_RTL_R8(0, 0x663)|BIT(2)); // write MAC'0 0x663 to notify DRVERLY_INT + DMDP_RTL_W8(0, 0x1cf, DMDP_RTL_R8(0, 0x1cf)|BIT(4)); // write MAC'0 0x1cf to notify firmware INT + if (priv->smcc_state==0) { + unsigned int now, timeout; + now = RTL_R32(TSFTR); + timeout = now + (priv->pmib->dot11RFEntry.smcc_t * 1000); + //printk("wlan%d start timer now=%d timeout=%d timeout1=%d\n", priv->pshare->wlandev_idx, now, timeout, timeout1); + setup_timer2(priv, timeout); +#ifdef PHASE2_TEST + timeout = now + (priv->pmib->dot11RFEntry.smcc_p * 1000); + setup_timer1(priv, timeout); +#endif + } + } +#endif +#endif +#if 0//def SMART_CONCURRENT_92D + priv->pshare->bcnCount++; + if ((priv->pshare->bcnCount % 3)==0) + smcc_signin_linkstate(priv, 1, 5, priv->smcc_state); + else if ((priv->pshare->bcnCount % 3)==1) + smcc_signin_linkstate(priv, 1, priv->pmib->dot11RFEntry.smcc_t, priv->smcc_state); +#endif + + if (priv->timoffset) { + update_beacon(priv); + } + } + } +#ifdef MBSSID + else { + if (GET_ROOT(priv)->pmib->miscEntry.vap_enable) { + for (i=0; i<RTL8192CD_NUM_VWLAN; i++) { + if ((priv->pvap_priv[i]->vap_init_seq > 0) && IS_DRV_OPEN(priv->pvap_priv[i]) + && (status & (HIMR_BCNDMA1 << (priv->pvap_priv[i]->vap_init_seq-1)))) + { + if (priv->pvap_priv[i]->timoffset) { + update_beacon(priv->pvap_priv[i]); + } + } + } + } + } +#endif + + // + // Polling highQ as there is multicast waiting for tx... + // +#ifdef UNIVERSAL_REPEATER + struct rtl8192cd_priv *priv_root=NULL; + if ((OPMODE & WIFI_STATION_STATE) && GET_VXD_PRIV(priv) && + (GET_VXD_PRIV(priv)->drv_state & DRV_STATE_VXD_AP_STARTED)) { + priv_root = priv; + priv = GET_VXD_PRIV(priv); + } +#endif + + if ((OPMODE & WIFI_AP_STATE)) { + + if (status & HIMR_BCNDMA0) { + val8 = *((unsigned char *)priv->beaconbuf + priv->timoffset + 4); + if (val8 & 0x01) { + if(RTL_R8(BCN_CTRL) & DIS_ATIM) + RTL_W8(BCN_CTRL, (RTL_R8(BCN_CTRL) & (~DIS_ATIM))); + process_mcast_dzqueue(priv); + priv->pkt_in_dtimQ = 0; + } else { + if(!(RTL_R8(BCN_CTRL) & DIS_ATIM)) + RTL_W8(BCN_CTRL, (RTL_R8(BCN_CTRL) | DIS_ATIM)); + } +#ifdef MBSSID + priv->pshare->bcnDOk_priv = priv; +#endif + } +#ifdef MBSSID + else if (GET_ROOT(priv)->pmib->miscEntry.vap_enable) { + for (i=0; i<RTL8192CD_NUM_VWLAN; i++) { + if ((priv->pvap_priv[i]->vap_init_seq > 0) && IS_DRV_OPEN(priv->pvap_priv[i]) + && (status & (HIMR_BCNDMA1 << (priv->pvap_priv[i]->vap_init_seq-1)))) { + val8 = *((unsigned char *)priv->pvap_priv[i]->beaconbuf + priv->pvap_priv[i]->timoffset + 4); + if (val8 & 0x01) { + if(RTL_R8(BCN_CTRL) & DIS_ATIM) + RTL_W8(BCN_CTRL, (RTL_R8(BCN_CTRL) & (~DIS_ATIM))); + process_mcast_dzqueue(priv->pvap_priv[i]); + priv->pvap_priv[i]->pkt_in_dtimQ = 0; + } else { + if(!(RTL_R8(BCN_CTRL) & DIS_ATIM)) + RTL_W8(BCN_CTRL, (RTL_R8(BCN_CTRL) | DIS_ATIM)); + } + + priv->pshare->bcnDOk_priv = priv->pvap_priv[i]; + } + } + } +#endif + + } + + if (priv->pshare->pkt_in_hiQ) { + int pre_head = get_txhead(priv->pshare->phw, MCAST_QNUM); + do { + txdesc_rollback(&pre_head); + } while ((get_txdesc_info(priv->pshare->pdesc_info, MCAST_QNUM) + pre_head)->type != _PRE_ALLOCLLCHDR_); + if (get_desc((get_txdesc(priv->pshare->phw, MCAST_QNUM) + pre_head)->Dword0) & TX_OWN) { + unsigned short *phdr = (unsigned short *)((get_txdesc_info(priv->pshare->pdesc_info, MCAST_QNUM) + pre_head)->pframe); +#ifdef __MIPSEB__ + phdr = (unsigned short *)KSEG1ADDR(phdr); +#endif + ClearMData(phdr); + } + tx_poll(priv, MCAST_QNUM); + priv->pshare->pkt_in_hiQ = 0; + } + +#ifdef UNIVERSAL_REPEATER + if (priv_root != NULL) + priv = priv_root; +#endif + } + + /* ================================================================ + Process Beacon OK/ERROR interrupt + ================================================================ */ + if (bcnDmaOk || bcnOk || bcnErr) + { + +#ifdef UNIVERSAL_REPEATER + struct rtl8192cd_priv *priv_root=NULL; + if ((OPMODE & WIFI_STATION_STATE) && GET_VXD_PRIV(priv) && + (GET_VXD_PRIV(priv)->drv_state & DRV_STATE_VXD_AP_STARTED)) { + priv_root = priv; + priv = GET_VXD_PRIV(priv); + } +#endif + + // + // Polling highQ as there is multicast waiting for tx... + // + if ((OPMODE & WIFI_AP_STATE) && bcnDmaOk) { +#ifdef PCIE_POWER_SAVING + if ((priv->offload_ctrl & 1) && (priv->offload_ctrl >> 7) && priv->pshare->rf_ft_var.power_save) { + priv->offload_ctrl &= (~1); + update_beacon(priv); + RTL_W16(0x100 , RTL_R16(0x100) & ~BIT(8)); // disable sw beacon + return; + } +#endif + + } + + // + // Statistics and LED counting + // + if (bcnOk) { + // for SW LED + if (((LED_TYPE >= LEDTYPE_SW_LINK_TXRX) && (LED_TYPE <= LEDTYPE_SW_LINKTXRX)) || + (LED_TYPE == LEDTYPE_SW_LED2_GPIO8_LINKTXRX)||(LED_TYPE == LEDTYPE_SW_LED2_GPIO10_LINKTXRX)|| + (LED_TYPE == LEDTYPE_SW_LED1_GPIO10_LINKTXRX)) + priv->pshare->LED_tx_cnt++; +#ifdef MBSSID + if (priv->pshare->bcnDOk_priv) + priv->pshare->bcnDOk_priv->ext_stats.beacon_ok++; +#else + priv->ext_stats.beacon_ok++; +#endif + SNMP_MIB_INC(dot11TransmittedFragmentCount, 1); + } else if (bcnErr) { +#ifdef MBSSID + if (priv->pshare->bcnDOk_priv) + priv->pshare->bcnDOk_priv->ext_stats.beacon_er++; +#else + priv->ext_stats.beacon_er++; +#endif + } + +#ifdef UNIVERSAL_REPEATER + if (priv_root != NULL) + priv = priv_root; +#endif + } + +#ifdef CLIENT_MODE + // + // Ad-hoc beacon status + // + if (OPMODE & WIFI_ADHOC_STATE) { + if (bcnOk) + priv->ibss_tx_beacon = TRUE; + if (bcnErr) + priv->ibss_tx_beacon = FALSE; + } +#endif +} + + +void check_dma_error(struct rtl8192cd_priv *priv, unsigned int status, unsigned int status_ext) +{ + unsigned char reg_rxdma; + unsigned int reg_txdma; + int clear_isr=0; + + if ((status & HIMR_RXFOVW) | (status_ext & HIMRE_RXERR)) { + reg_rxdma = RTL_R8(RXDMA_STATUS); + if (reg_rxdma) { + RTL_W8(RXDMA_STATUS, reg_rxdma); + //panic_printk("RXDMA_STATUS %02x\n", reg_rxdma); + clear_isr = 1; + } + } + + if ((status & HIMR_TXFOVW) | (status_ext & HIMRE_TXERR)) { + reg_txdma = RTL_R32(TXDMA_STATUS); + if (reg_txdma) { + RTL_W32(TXDMA_STATUS, reg_txdma); + //panic_printk("TXDMA_STATUS %08x\n", reg_txdma); + clear_isr = 1; + } + } + + if (clear_isr) { + RTL_W32(HISR, status); + RTL_W32(HISRE, status_ext); + } +} + +#define RTL_WLAN_INT_RETRY_MAX (32) +__MIPS16 +__IRAM_IN_865X +__inline__ static int __rtl8192cd_interrupt(void *dev_instance) +{ + struct net_device *dev; + struct rtl8192cd_priv *priv; + struct rtl8192cd_hw *phw; + + unsigned int status, status_ext, retry_cnt; + unsigned int caseBcnInt, caseBcnDmaOK, caseBcnStatusOK, caseBcnStatusER; + unsigned int caseRxStatus, caseRxRDU; + #if 0 //TESTCHIP_SUPPORT || defined(SMART_CONCURRENT_92D) // --fwdebug + unsigned int caseTimer1; + #endif + #ifdef SUPPORT_TX_AMSDU + unsigned int caseTimer2; + #endif +#if defined (SUPPORT_TX_AMSDU) || defined (SMART_CONCURRENT_92D) + unsigned long current_value, timeout; +#endif + + dev = (struct net_device *)dev_instance; +#ifdef NETDEV_NO_PRIV + priv = ((struct rtl8192cd_priv *)netdev_priv(dev))->wlan_priv; +#else + priv = (struct rtl8192cd_priv *)dev->priv; +#endif + +#ifdef CONFIG_RTL_92D_DMDP + if (GET_CHIP_VER(priv)==VERSION_8192D) { + #if (RTL_USED_PCIE_SLOT==1) + if (!((REG32(0xb8b21004)& 0x01) && (priv->pshare->wlandev_idx ==0)) && + !((REG32(0xb8b21004)& 0x02) && (priv->pshare->wlandev_idx ==1))) { + //printk("INT=[%02x] WLAN(%d)\n",(REG32(0xb8b21004)& 0x0f),(priv->pshare->wlandev_idx)); + goto int_exit; + } + #else + if (!((REG32(0xb8b01004)& 0x01) && (priv->pshare->wlandev_idx ==0)) && + !((REG32(0xb8b01004)& 0x02) && (priv->pshare->wlandev_idx ==1))) { + //printk("INT=[%02x] WLAN(%d)\n",(REG32(0xb8b21004)& 0x0f),(priv->pshare->wlandev_idx)); + goto int_exit; + } + #endif + } +#endif + +#ifdef PCIE_POWER_SAVING + if ((priv->pwr_state == L2) || (priv->pwr_state == L1)) { +#ifdef CONFIG_RTL_92D_DMDP + REG32(0xb8003000) = REG32(0xb8003000)|BIT(22); +#endif + goto int_exit; + } +#endif + + status = RTL_R32(HISR); + RTL_W32(HISR, status); + status_ext = RTL_R32(HISRE); + RTL_W32(HISRE, status_ext); + if (status == 0 && status_ext == 0) { + goto int_exit; + } + + retry_cnt = 0; +retry_process: + + caseBcnInt = caseBcnDmaOK = caseBcnStatusOK = caseBcnStatusER = 0; + caseRxStatus = caseRxRDU = 0; + #if 0 //TESTCHIP_SUPPORT || defined(SMART_CONCURRENT_92D) // --fwdebug + caseTimer1 = 0; + #endif + #ifdef SUPPORT_TX_AMSDU + caseTimer2 = 0 + #endif + +#ifdef TXREPORT + if(status_ext & BIT(9) ) { + C2H_isr(priv); + } +#endif + + check_dma_error(priv, status, status_ext); + + if (status & (HIMR_BCNDMA0 | HIMR_BCNDMA1 | HIMR_BCNDMA2 | HIMR_BCNDMA3 | HIMR_BCNDMA4 | HIMR_BCNDMA5 | HIMR_BCNDMA6 | HIMR_BCNDMA7)) + caseBcnInt = 1; + + if (status & (HIMR_BCNDOK0 | HIMR_BCNDOK1 | HIMR_BCNDOK2 | HIMR_BCNDOK3 | HIMR_BCNDOK4 | HIMR_BCNDOK5 | HIMR_BCNDOK6 | HIMR_BCNDOK7)) + caseBcnDmaOK = 1; + + if (status & HIMR_TXBCNOK) + caseBcnStatusOK = 1; + + if (status & HIMR_TXBCNERR) + caseBcnStatusER = 1; + + if (status & (HIMR_ROK | HIMR_RDU)) + caseRxStatus = 1; + + if (status & HIMR_RDU) { + priv->ext_stats.rx_rdu++; + caseRxRDU = 1; + priv->pshare->skip_mic_chk = SKIP_MIC_NUM; + } + + if (status & HIMR_RXFOVW) { + priv->ext_stats.rx_fifoO++; + priv->pshare->skip_mic_chk = SKIP_MIC_NUM; + } + + #if 0 //TESTCHIP_SUPPORT || defined(SMART_CONCURRENT_92D) // --fwdebug + if (status & HIMR_TIMEOUT1) + caseTimer1 = 1; + #endif + + #ifdef SUPPORT_TX_AMSDU + if (status & HIMR_TIMEOUT2) + caseTimer2 = 1; + #endif + + if (caseBcnInt || caseBcnDmaOK || caseBcnStatusOK || caseBcnStatusER){ + rtl8192cd_bcnProc(priv, caseBcnInt, caseBcnDmaOK, caseBcnStatusOK, caseBcnStatusER, status); + } + + // + // Rx interrupt + // + if (caseRxStatus) + { + // stop RX first +#ifdef __KERNEL__ +#if defined(RTL8190_ISR_RX) && defined(RTL8190_DIRECT_RX) +#if defined(RX_TASKLET) + if (!priv->pshare->has_triggered_rx_tasklet) { + priv->pshare->has_triggered_rx_tasklet = 1; + //RTL_W32(HIMR, priv->pshare->InterruptMask & ~(HIMR_RXFOVW | HIMR_RDU | HIMR_ROK)); + RTL_W32(HIMR, priv->pshare->InterruptMask & ~(HIMR_RXFOVW | HIMR_ROK)); + tasklet_hi_schedule(&priv->pshare->rx_tasklet); + } +#else + rtl8192cd_rx_isr(priv); +#endif + +#else // !(defined RTL8190_ISR_RX && RTL8190_DIRECT_RX) + if (caseRxRDU) { + rtl8192cd_rx_isr(priv); + tasklet_hi_schedule(&priv->pshare->rx_tasklet); + } + else { + if (priv->pshare->rxInt_useTsklt) + tasklet_hi_schedule(&priv->pshare->rx_tasklet); + else + rtl8192cd_rx_dsr((unsigned long)priv); + } +#endif + +#else // !__KERNEL__ + rtl8192cd_rx_dsr((unsigned long)priv); +#endif + } + + // + // Tx interrupt + // + phw = GET_HW(priv); +#ifdef MP_TEST +#ifdef SMP_SYNC + if (OPMODE & WIFI_MP_STATE){ + if (!priv->pshare->has_triggered_tx_tasklet) { + tasklet_schedule(&priv->pshare->tx_tasklet); + priv->pshare->has_triggered_tx_tasklet = 1; + } + } +#else + if (OPMODE & WIFI_MP_STATE) + rtl8192cd_tx_dsr((unsigned long)priv); +#endif + else +#endif + if ((CIRC_CNT_RTK(phw->txhead0, phw->txtail0, NUM_TX_DESC) > 10) || + (CIRC_CNT_RTK(phw->txhead1, phw->txtail1, NUM_TX_DESC) > 10) || + (CIRC_CNT_RTK(phw->txhead2, phw->txtail2, NUM_TX_DESC) > 10) || + (CIRC_CNT_RTK(phw->txhead3, phw->txtail3, NUM_TX_DESC) > 10) || + (CIRC_CNT_RTK(phw->txhead4, phw->txtail4, NUM_TX_DESC) > 10) || + (CIRC_CNT_RTK(phw->txhead5, phw->txtail5, NUM_TX_DESC) > 10) + ) { +#ifdef __KERNEL__ + if (!priv->pshare->has_triggered_tx_tasklet) { + tasklet_schedule(&priv->pshare->tx_tasklet); + priv->pshare->has_triggered_tx_tasklet = 1; + } +#else +#ifdef SMP_SYNC + if (!priv->pshare->has_triggered_tx_tasklet) { + tasklet_schedule(&priv->pshare->tx_tasklet); + priv->pshare->has_triggered_tx_tasklet = 1; + } +#else + rtl8192cd_tx_dsr((unsigned long)priv); +#endif +#endif + } + +#if 0 //TESTCHIP_SUPPORT +#ifdef SMART_CONCURRENT_92D // --fwdebug + +#ifdef PHASE2_TEST + if (caseTimer1 && priv->pshare->wlandev_idx == 1) { + cancel_timer1(priv); + current_value = RTL_R32(TSFTR) ; + timeout = RTL_R32(TIMER0); + if (TSF_LESS(current_value, timeout)){ + setup_timer1(priv, timeout); + }else { + //printk("TIMER1 %d \n", current_value); + RTL_W8_Phy0(priv, 0x1cf, RTL_R8_Phy0(priv, 0x1cf)|BIT(2)); // write MAC'0 0x1cf to notify firmware h2c cmd + } + } +#endif + if (caseTimer2 && priv->pshare->wlandev_idx == 1) { + cancel_timer2(priv); + + current_value = RTL_R32(TSFTR) ; + timeout = RTL_R32(TIMER1); + if (TSF_LESS(current_value, timeout)){ + setup_timer2(priv, timeout); + }else { + //printk("TIMER2 %d \n", current_value); + DMDP_RTL_W8(0, 0x1cf, DMDP_RTL_R8(0, 0x1cf)|BIT(3)); // write MAC'0 0x1cf to notify firmware h2c cmd + } + } +#endif +#endif + +#ifdef SUPPORT_TX_AMSDU + if (caseTimer2) { + RTL_W32(IMR, RTL_R32(IMR) & ~IMR_TIMEOUT2); + + current_value = RTL_R32(TSFR) ; + timeout = RTL_R32(TIMER1); + if (TSF_LESS(current_value, timeout)) + setup_timer2(priv, timeout); + else + amsdu_timeout(priv, current_value); + } +#endif + + status = RTL_R32(HISR); + status_ext = RTL_R32(HISRE); + + if ((status!=0||status_ext!=0) && ((retry_cnt++)<RTL_WLAN_INT_RETRY_MAX)) { + #if defined(CONFIG_RTL865X_WTDOG) || defined(CONFIG_RTL_WTDOG) + REG32(BSP_WDTCNR) |= 1 << 23; + #endif + RTL_W32(HISR, status); + RTL_W32(HISRE, status_ext); + + goto retry_process; + } + +int_exit: + return SUCCESS; +} + + +#ifdef __LINUX_2_6__ +__MIPS16 +__IRAM_IN_865X +irqreturn_t rtl8192cd_interrupt(int irq, void *dev_instance) +{ + __rtl8192cd_interrupt(dev_instance); + return IRQ_HANDLED; +} +#else +__MIPS16 +__IRAM_IN_865X +void rtl8192cd_interrupt(int irq, void *dev_instance, struct pt_regs *regs) +{ + __rtl8192cd_interrupt(dev_instance); + return; +} +#endif + + +static void rtl8192cd_set_rx_mode(struct net_device *dev) +{ + +} + + +static struct net_device_stats *rtl8192cd_get_stats(struct net_device *dev) +{ +#ifdef NETDEV_NO_PRIV + struct rtl8192cd_priv *priv = ((struct rtl8192cd_priv *)netdev_priv(dev))->wlan_priv; +#else + struct rtl8192cd_priv *priv = (struct rtl8192cd_priv *)dev->priv; +#endif + +#ifdef SMP_SYNC + unsigned long flags; +#endif + + SMP_LOCK(flags); + +#ifdef WDS + int idx; + struct stat_info *pstat; + + if (dev->base_addr == 0) { + idx = getWdsIdxByDev(priv, dev); + if (idx < 0) { + memset(&priv->wds_stats[NUM_WDS-1], 0, sizeof(struct net_device_stats)); + SMP_UNLOCK(flags); + return &priv->wds_stats[NUM_WDS-1]; + } + + if (netif_running(dev) && netif_running(priv->dev)) { + pstat = get_stainfo(priv, priv->pmib->dot11WdsInfo.entry[idx].macAddr); + if (pstat == NULL) { + DEBUG_ERR("%s: get_stainfo() wds fail!\n", (char *)__FUNCTION__); + memset(&priv->wds_stats[idx], 0, sizeof(struct net_device_stats)); + } + else { + priv->wds_stats[idx].tx_packets = pstat->tx_pkts; + priv->wds_stats[idx].tx_errors = pstat->tx_fail; + priv->wds_stats[idx].tx_bytes = pstat->tx_bytes; + priv->wds_stats[idx].rx_packets = pstat->rx_pkts; + priv->wds_stats[idx].rx_bytes = pstat->rx_bytes; + } + } + SMP_UNLOCK(flags); + return &priv->wds_stats[idx]; + } +#endif + +#ifdef CONFIG_RTK_MESH + + if (dev->base_addr == 1) { + if(priv->mesh_dev != dev) + { + SMP_UNLOCK(flags); + return NULL; + } + + SMP_UNLOCK(flags); + return &priv->mesh_stats; + } +#endif // CONFIG_RTK_MESH + + SMP_UNLOCK(flags); + return &(priv->net_stats); +} + +static int rtl8192cd_init_sw(struct rtl8192cd_priv *priv) +{ + // All the index/counters should be reset to zero... + struct rtl8192cd_hw *phw=NULL; + unsigned long offset; + unsigned int i; + struct sk_buff *pskb; + unsigned char *page_ptr; + struct wlan_hdr_poll *pwlan_hdr_poll; + struct wlanllc_hdr_poll *pwlanllc_hdr_poll; + struct wlanbuf_poll *pwlanbuf_poll; + struct wlanicv_poll *pwlanicv_poll; + struct wlanmic_poll *pwlanmic_poll; + struct wlan_acl_poll *pwlan_acl_poll; +#ifdef _MESH_ACL_ENABLE_ + struct mesh_acl_poll *pmesh_acl_poll; +#endif + unsigned long ring_virt_addr; + unsigned long ring_dma_addr; + unsigned int ring_buf_len; + unsigned char MIMO_TR_hw_support; + unsigned int NumTotalRFPath; +#ifndef PRIV_STA_BUF + unsigned long alloc_dma_buf; +#endif +#if defined(CLIENT_MODE) && defined(CHECK_HANGUP) + unsigned char *pbackup=NULL; + unsigned long backup_len=0; +#endif +#ifdef _11s_TEST_MODE_ + struct Galileo_poll *pgalileo_poll; +#endif + +#ifdef DFS + /* + * For JAPAN : prevent switching to channels 52, 56, 60, and 64 in adhoc mode + */ + if (((priv->pmib->dot11StationConfigEntry.dot11RegDomain == DOMAIN_MKK) || + (priv->pmib->dot11StationConfigEntry.dot11RegDomain == DOMAIN_MKK3)) && + (OPMODE & WIFI_ADHOC_STATE)) { + if (priv->pmib->dot11BssType.net_work_type & WIRELESS_11A) { + /* block channels 52~64 and place them in NOP_chnl */ + if (!timer_pending(&priv->ch52_timer)) + InsertChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 52); + if (!timer_pending(&priv->ch56_timer)) + InsertChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 56); + if (!timer_pending(&priv->ch60_timer)) + InsertChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 60); + if (!timer_pending(&priv->ch64_timer)) + InsertChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 64); + } + + /* if users select an illegal channel, the driver will switch to channel 36~48 */ + if ((priv->pmib->dot11RFEntry.dot11channel >= 52) && (priv->pmib->dot11RFEntry.dot11channel <= 64)) { + PRINT_INFO("Channel %d is illegal in ad-hoc mode in Japan!\n", priv->pmib->dot11RFEntry.dot11channel); + priv->pmib->dot11RFEntry.dot11channel = DFS_SelectChannel(priv); + PRINT_INFO("Swiching to channel %d!\n", priv->pmib->dot11RFEntry.dot11channel); + } + } + + /* if users select a blocked channel, the driver will switch to unblocked channel */ + if (!priv->pmib->dot11DFSEntry.disable_DFS && + ((timer_pending(&priv->ch52_timer) && (priv->pmib->dot11RFEntry.dot11channel == 52)) || + (timer_pending(&priv->ch56_timer) && (priv->pmib->dot11RFEntry.dot11channel == 56)) || + (timer_pending(&priv->ch60_timer) && (priv->pmib->dot11RFEntry.dot11channel == 60)) || + (timer_pending(&priv->ch64_timer) && (priv->pmib->dot11RFEntry.dot11channel == 64)) || + (timer_pending(&priv->ch100_timer) && (priv->pmib->dot11RFEntry.dot11channel == 100)) || + (timer_pending(&priv->ch104_timer) && (priv->pmib->dot11RFEntry.dot11channel == 104)) || + (timer_pending(&priv->ch108_timer) && (priv->pmib->dot11RFEntry.dot11channel == 108)) || + (timer_pending(&priv->ch112_timer) && (priv->pmib->dot11RFEntry.dot11channel == 112)) || + (timer_pending(&priv->ch116_timer) && (priv->pmib->dot11RFEntry.dot11channel == 116)) || + (timer_pending(&priv->ch120_timer) && (priv->pmib->dot11RFEntry.dot11channel == 120)) || + (timer_pending(&priv->ch124_timer) && (priv->pmib->dot11RFEntry.dot11channel == 124)) || + (timer_pending(&priv->ch128_timer) && (priv->pmib->dot11RFEntry.dot11channel == 128)) || + (timer_pending(&priv->ch132_timer) && (priv->pmib->dot11RFEntry.dot11channel == 132)) || + (timer_pending(&priv->ch136_timer) && (priv->pmib->dot11RFEntry.dot11channel == 136)) || + (timer_pending(&priv->ch140_timer) && (priv->pmib->dot11RFEntry.dot11channel == 140)))) { + PRINT_INFO("Channel %d is still in none occupancy period!\n", priv->pmib->dot11RFEntry.dot11channel); + priv->pmib->dot11RFEntry.dot11channel = DFS_SelectChannel(priv); + PRINT_INFO("Swiching to channel %d!\n", priv->pmib->dot11RFEntry.dot11channel); + } + + /* disable all of the transmissions during channel availability check */ +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { + priv->pmib->dot11DFSEntry.disable_tx = 0; + if (!priv->pmib->dot11DFSEntry.disable_DFS && + (((priv->pmib->dot11RFEntry.dot11channel >= 52) && + (priv->pmib->dot11RFEntry.dot11channel <= 64)) || + ((priv->pmib->dot11RFEntry.dot11channel >= 100) && + (priv->pmib->dot11RFEntry.dot11channel <= 140))) && + (OPMODE & WIFI_AP_STATE)) + priv->pmib->dot11DFSEntry.disable_tx = 1; + } +#endif + +#ifdef ENABLE_RTL_SKB_STATS + rtl_atomic_set(&priv->rtl_tx_skb_cnt, 0); + rtl_atomic_set(&priv->rtl_rx_skb_cnt, 0); +#endif + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { +#ifdef __KERNEL__ +#ifdef DFS + /* will not initialize the tasklet if the driver is rebooting due to the detection of radar */ + if (!priv->pmib->dot11DFSEntry.DFS_detected) +#endif + { +#ifdef CHECK_HANGUP + if (!priv->reset_hangup) +#endif + { +#ifdef PCIE_POWER_SAVING + tasklet_init(&priv->pshare->ps_tasklet, PCIe_power_save_tasklet, (unsigned long)priv); +#endif +#if !(defined(RTL8190_ISR_RX) && defined(RTL8190_DIRECT_RX)) + tasklet_init(&priv->pshare->rx_tasklet, rtl8192cd_rx_dsr, (unsigned long)priv); +#else +#ifdef RX_TASKLET + tasklet_init(&priv->pshare->rx_tasklet, rtl8192cd_rx_tkl_isr, (unsigned long)priv); +#endif +#endif + tasklet_init(&priv->pshare->tx_tasklet, rtl8192cd_tx_dsr, (unsigned long)priv); + tasklet_init(&priv->pshare->oneSec_tasklet, rtl8192cd_expire_timer, (unsigned long)priv); + } + } +#endif // __KERNEL__ + +#ifdef DFS + if (priv->pmib->dot11DFSEntry.DFS_detected) + priv->pmib->dot11DFSEntry.DFS_detected = 0; +#endif + + phw = GET_HW(priv); + + // save descriptor virtual address before reset, david + ring_virt_addr = phw->ring_virt_addr; + ring_dma_addr = phw->ring_dma_addr; + ring_buf_len = phw->ring_buf_len; +#ifndef PRIV_STA_BUF + alloc_dma_buf = phw->alloc_dma_buf; +#endif + + // save RF related settings before reset + MIMO_TR_hw_support = phw->MIMO_TR_hw_support; + NumTotalRFPath = phw->NumTotalRFPath; + + memset((void *)phw, 0, sizeof(struct rtl8192cd_hw)); + phw->ring_virt_addr = ring_virt_addr; + phw->ring_buf_len = ring_buf_len; +#ifndef PRIV_STA_BUF + phw->alloc_dma_buf = alloc_dma_buf; +#endif + +#ifdef CONFIG_NET_PCI + if (IS_PCIBIOS_TYPE) + phw->ring_dma_addr = ring_dma_addr; +#endif + phw->MIMO_TR_hw_support = MIMO_TR_hw_support; + phw->NumTotalRFPath = NumTotalRFPath; + +#if defined(DUALBAND_ONLY) && defined(CONFIG_RTL8190_PRIV_SKB) + if (priv->pmib->dot11RFEntry.macPhyMode == DUALMAC_DUALPHY) + split_pool(priv); + else + merge_pool(priv); +#endif + } + +#if defined(CLIENT_MODE) && defined(CHECK_HANGUP) + if (priv->reset_hangup && + (OPMODE & (WIFI_STATION_STATE | WIFI_ADHOC_STATE))) { + backup_len = ((unsigned long)&((struct rtl8192cd_priv *)0)->br_ip) - + ((unsigned long)&((struct rtl8192cd_priv *)0)->join_res)+4; + pbackup = kmalloc(backup_len, GFP_ATOMIC); + if (pbackup) + memcpy(pbackup, &priv->join_res, backup_len); + } +#endif + + offset = (unsigned long)(&((struct rtl8192cd_priv *)0)->net_stats); + // zero all data members below (including) stats + memset((void *)((unsigned long)priv + offset), 0, sizeof(struct rtl8192cd_priv)-offset); + +#ifdef CHECK_HANGUP + if (!priv->reset_hangup) +#endif + { + priv->up_time = 0; + } + +#if defined(CLIENT_MODE) && defined(CHECK_HANGUP) + if (priv->reset_hangup && pbackup) { + memcpy(&priv->join_res, pbackup, backup_len); + kfree(pbackup); + } +#endif + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { + // zero all data members below (including) LED_Timer of share_info + offset = (unsigned long)(&((struct priv_shared_info*)0)->LED_Timer); + memset((void *)((unsigned long)priv->pshare+ offset), 0, sizeof(struct priv_shared_info)-offset); + +#ifdef CONFIG_RTK_MESH + memset((void *)&priv->pshare->meshare, 0, sizeof(struct MESH_Share)); + get_random_bytes((void *)&priv->pshare->meshare.seq, sizeof(priv->pshare->meshare.seq)); +#if (MESH_DBG_LV & MESH_DBG_COMPLEX) + init_timer(&priv->pshare->meshare.mesh_test_sme_timer); + priv->pshare->meshare.mesh_test_sme_timer.data = (unsigned long) priv; + priv->pshare->meshare.mesh_test_sme_timer.function = mesh_test_sme_timer; + mod_timer(&priv->pshare->meshare.mesh_test_sme_timer, jiffies + RTL_SECONDS_TO_JIFFIES(2)); +#endif // (MESH_DBG_LV & MESH_DBG_COMPLEX) + +#if (MESH_DBG_LV & MESH_DBG_TEST) + init_timer(&priv->pshare->meshare.mesh_test_sme_timer2); + priv->pshare->meshare.mesh_test_sme_timer2.data = (unsigned long) priv; + priv->pshare->meshare.mesh_test_sme_timer2.function = mesh_test_sme_timer2; + mod_timer(&priv->pshare->meshare.mesh_test_sme_timer2, jiffies + RTL_SECONDS_TO_JIFFIES(50)); +#endif // (MESH_DBG_LV & MESH_DBG_TEST) + +#endif // CONFIG_RTK_MESH + + pwlan_hdr_poll = priv->pshare->pwlan_hdr_poll; + pwlanllc_hdr_poll = priv->pshare->pwlanllc_hdr_poll; + pwlanbuf_poll = priv->pshare->pwlanbuf_poll; + pwlanicv_poll = priv->pshare->pwlanicv_poll; + pwlanmic_poll = priv->pshare->pwlanmic_poll; + +#ifdef _11s_TEST_MODE_ + pgalileo_poll = priv->pshare->galileo_poll; + pgalileo_poll->count = AODV_RREQ_TABLE_SIZE; +#endif + pwlan_hdr_poll->count = PRE_ALLOCATED_HDR; + pwlanllc_hdr_poll->count = PRE_ALLOCATED_HDR; + pwlanbuf_poll->count = PRE_ALLOCATED_MMPDU; + pwlanicv_poll->count = PRE_ALLOCATED_HDR; + pwlanmic_poll->count = PRE_ALLOCATED_HDR; + + // initialize all the hdr/buf node, and list to the poll_list + INIT_LIST_HEAD(&priv->pshare->wlan_hdrlist); + INIT_LIST_HEAD(&priv->pshare->wlanllc_hdrlist); + INIT_LIST_HEAD(&priv->pshare->wlanbuf_list); + INIT_LIST_HEAD(&priv->pshare->wlanicv_list); + INIT_LIST_HEAD(&priv->pshare->wlanmic_list); + +#ifdef _11s_TEST_MODE_ //Galileo + + memset(priv->rvTestPacket, 0, 3000); + + INIT_LIST_HEAD(&priv->pshare->galileo_list); + INIT_LIST_HEAD(&priv->mtb_list); + + for(i=0; i< AODV_RREQ_TABLE_SIZE; i++) + { + INIT_LIST_HEAD(&(pgalileo_poll->node[i].list)); + list_add_tail(&(pgalileo_poll->node[i].list), &priv->pshare->galileo_list); + init_timer(&pgalileo_poll->node[i].data.expire_timer); + pgalileo_poll->node[i].data.priv = priv; + pgalileo_poll->node[i].data.expire_timer.function = galileo_timer; + } +#endif + + for(i=0; i< PRE_ALLOCATED_HDR; i++) + { + INIT_LIST_HEAD(&(pwlan_hdr_poll->hdrnode[i].list)); + list_add_tail(&(pwlan_hdr_poll->hdrnode[i].list), &priv->pshare->wlan_hdrlist); + + INIT_LIST_HEAD(&(pwlanllc_hdr_poll->hdrnode[i].list)); + list_add_tail( &(pwlanllc_hdr_poll->hdrnode[i].list), &priv->pshare->wlanllc_hdrlist); + + INIT_LIST_HEAD(&(pwlanicv_poll->hdrnode[i].list)); + list_add_tail( &(pwlanicv_poll->hdrnode[i].list), &priv->pshare->wlanicv_list); + + INIT_LIST_HEAD(&(pwlanmic_poll->hdrnode[i].list)); + list_add_tail( &(pwlanmic_poll->hdrnode[i].list), &priv->pshare->wlanmic_list); + } + + for(i=0; i< PRE_ALLOCATED_MMPDU; i++) + { + INIT_LIST_HEAD(&(pwlanbuf_poll->hdrnode[i].list)); + list_add_tail( &(pwlanbuf_poll->hdrnode[i].list), &priv->pshare->wlanbuf_list); + } + + DEBUG_INFO("hdrlist=%x, llc_hdrlist=%x, buf_list=%x, icv_list=%x, mic_list=%X\n", + (UINT)&priv->pshare->wlan_hdrlist, (UINT)&priv->pshare->wlanllc_hdrlist, (UINT)&priv->pshare->wlanbuf_list, + (UINT)&priv->pshare->wlanicv_list, (UINT)&priv->pshare->wlanmic_list); + + page_ptr = (unsigned char *)phw->ring_virt_addr; + memset(page_ptr, 0, phw->ring_buf_len); // this is vital! + +#ifndef __KERNEL__ + if ((unsigned int)page_ptr & (~(PAGE_SIZE - 1))) + page_ptr = (unsigned char *)(((unsigned int)page_ptr & (~(PAGE_SIZE - 1))) + PAGE_SIZE); +#endif +#ifdef CONFIG_NET_PCI + if (!IS_PCIBIOS_TYPE) +#endif + phw->ring_dma_addr = virt_to_bus(page_ptr); + + phw->rx_ring_addr = phw->ring_dma_addr; + phw->tx_ring0_addr = phw->ring_dma_addr + NUM_RX_DESC * sizeof(struct rx_desc); + phw->tx_ring1_addr = phw->tx_ring0_addr + NUM_TX_DESC * sizeof(struct tx_desc); + phw->tx_ring2_addr = phw->tx_ring1_addr + NUM_TX_DESC * sizeof(struct tx_desc); + phw->tx_ring3_addr = phw->tx_ring2_addr + NUM_TX_DESC * sizeof(struct tx_desc); + phw->tx_ring4_addr = phw->tx_ring3_addr + NUM_TX_DESC * sizeof(struct tx_desc); + phw->tx_ring5_addr = phw->tx_ring4_addr + NUM_TX_DESC * sizeof(struct tx_desc); + phw->rxcmd_ring_addr = phw->tx_ring5_addr + NUM_TX_DESC * sizeof(struct tx_desc); + phw->txcmd_ring_addr = phw->rxcmd_ring_addr + NUM_CMD_DESC * sizeof(struct rx_desc); + phw->tx_ringB_addr = phw->txcmd_ring_addr + NUM_CMD_DESC * sizeof(struct tx_desc); + + phw->rx_descL = (struct rx_desc *)page_ptr; + phw->tx_desc0 = (struct tx_desc *)(page_ptr + NUM_RX_DESC * sizeof(struct rx_desc)); + phw->tx_desc1 = (struct tx_desc *)((unsigned long)phw->tx_desc0 + NUM_TX_DESC * sizeof(struct tx_desc)); + phw->tx_desc2 = (struct tx_desc *)((unsigned long)phw->tx_desc1 + NUM_TX_DESC * sizeof(struct tx_desc)); + phw->tx_desc3 = (struct tx_desc *)((unsigned long)phw->tx_desc2 + NUM_TX_DESC * sizeof(struct tx_desc)); + phw->tx_desc4 = (struct tx_desc *)((unsigned long)phw->tx_desc3 + NUM_TX_DESC * sizeof(struct tx_desc)); + phw->tx_desc5 = (struct tx_desc *)((unsigned long)phw->tx_desc4 + NUM_TX_DESC * sizeof(struct tx_desc)); + phw->rxcmd_desc = (struct rx_desc *)((unsigned long)phw->tx_desc5 + NUM_TX_DESC * sizeof(struct tx_desc)); + phw->txcmd_desc = (struct tx_desc *)((unsigned long)phw->rxcmd_desc + NUM_CMD_DESC * sizeof(struct rx_desc)); + phw->tx_descB = (struct tx_desc *)((unsigned long)phw->txcmd_desc + NUM_CMD_DESC * sizeof(struct tx_desc)); + + /* To set the DMA address for both RX/TX ring */ + { +#ifndef USE_RTL8186_SDK + int txDescRingIdx; + + struct tx_desc *tx_desc_array[] = { + phw->tx_desc0, + phw->tx_desc1, + phw->tx_desc2, + phw->tx_desc3, + phw->tx_desc4, + phw->tx_desc5, + phw->tx_descB, + 0 + }; +#ifdef CONFIG_NET_PCI + unsigned long *tx_desc_dma_array[] = { + (unsigned long*)(phw->tx_desc0_dma_addr), + (unsigned long*)(phw->tx_desc1_dma_addr), + (unsigned long*)(phw->tx_desc2_dma_addr), + (unsigned long*)(phw->tx_desc3_dma_addr), + (unsigned long*)(phw->tx_desc4_dma_addr), + (unsigned long*)(phw->tx_desc5_dma_addr), + (unsigned long*)(phw->tx_descB_dma_addr), + (unsigned long*)0 + }; +#endif +#endif // !USE_RTL8186_SDK + + /* RX RING */ +#if defined(NOT_RTK_BSP) + for (i=0; i<NUM_RX_DESC; i++) { + phw->rx_descL_dma_addr[i] = phw->rx_ring_addr + i*(sizeof(struct rx_desc)); + } +#else + for (i=0; i<NUM_RX_DESC; i++) { + phw->rx_descL_dma_addr[i] = get_physical_addr(priv, (void *)(&phw->rx_descL[i]), + sizeof(struct rx_desc), PCI_DMA_TODEVICE); + } +#endif + +#ifndef USE_RTL8186_SDK + +#if defined(NOT_RTK_BSP) + unsigned int tx_dma_start; + tx_dma_start = phw->ring_dma_addr + NUM_RX_DESC * sizeof(struct rx_desc); +#endif + + /* TX RING */ + txDescRingIdx = 0; + + while (tx_desc_array[txDescRingIdx] != 0) { +#ifdef CONFIG_NET_PCI + unsigned long *tx_desc_dma_ptr = tx_desc_dma_array[txDescRingIdx]; +#endif + struct tx_desc *tx_desc_ptr = tx_desc_array[txDescRingIdx]; + +#if defined(NOT_RTK_BSP) + for (i=0; i<NUM_TX_DESC; i++) { + tx_desc_dma_ptr[i]= tx_dma_start + txDescRingIdx*NUM_TX_DESC * sizeof(struct tx_desc) + i*sizeof(struct tx_desc); + } +#else + for (i=0; i<NUM_TX_DESC; i++) { + tx_desc_dma_ptr[i] = get_physical_addr(priv, (void *)(&(tx_desc_ptr[i])), + sizeof(struct tx_desc), PCI_DMA_TODEVICE); + } +#endif + + txDescRingIdx ++; + } +#endif // !USE_RTL8186_SDK + +#if defined(NOT_RTK_BSP) + for (i=0; i<NUM_CMD_DESC; i++) { + phw->rxcmd_desc_dma_addr[i] = phw->rxcmd_ring_addr + i*sizeof(struct rx_desc); + phw->txcmd_desc_dma_addr[i] = phw->txcmd_ring_addr + i*sizeof(struct tx_desc); + } + +#else + for (i=0; i<NUM_CMD_DESC; i++) { + phw->rxcmd_desc_dma_addr[i] = get_physical_addr(priv, (void *)(&phw->rxcmd_desc[i]), + sizeof(struct rx_desc), PCI_DMA_TODEVICE); + phw->txcmd_desc_dma_addr[i] = get_physical_addr(priv, (void *)(&phw->txcmd_desc[i]), + sizeof(struct tx_desc), PCI_DMA_TODEVICE); + } +#endif + } + + DEBUG_INFO("rx_descL=%08x tx_desc0=%08x, tx_desc1=%08x, tx_desc2=%08x, tx_desc3=%08x, tx_desc4=%08x, " + "tx_desc5=%08x, rxcmd_desc=%08x, txcmd_desc=%08x, tx_descB=%08x\n", + (UINT)phw->rx_descL, (UINT)phw->tx_desc0, (UINT)phw->tx_desc1, (UINT)phw->tx_desc2, + (UINT)phw->tx_desc3, (UINT)phw->tx_desc4, (UINT)phw->tx_desc5, + (UINT)phw->rxcmd_desc, (UINT)phw->txcmd_desc, (UINT)phw->tx_descB); +#ifdef RTK_QUE + rtk_queue_init(&priv->pshare->skb_queue); +#else + skb_queue_head_init(&priv->pshare->skb_queue); +#endif + + // Now for Rx desc... + for(i=0; i<NUM_RX_DESC; i++) + { + pskb = rtl_dev_alloc_skb(priv, RX_BUF_LEN, _SKB_RX_, 1); + if (pskb == NULL) { + printk("can't allocate skbuff for RX, abort!\n"); + return 1; + } + init_rxdesc(pskb, i, priv); + } + + // Nothing to do for Tx desc... + for(i=0; i<NUM_TX_DESC; i++) + { + init_txdesc(priv, phw->tx_desc0, phw->tx_ring0_addr, i); + init_txdesc(priv, phw->tx_desc1, phw->tx_ring1_addr, i); + init_txdesc(priv, phw->tx_desc2, phw->tx_ring2_addr, i); + init_txdesc(priv, phw->tx_desc3, phw->tx_ring3_addr, i); + init_txdesc(priv, phw->tx_desc4, phw->tx_ring4_addr, i); + init_txdesc(priv, phw->tx_desc5, phw->tx_ring5_addr, i); + } + +#ifdef MBSSID + for(i=0; i<(RTL8192CD_NUM_VWLAN+1); i++) { + if (GET_ROOT(priv)->pmib->miscEntry.vap_enable) { + if (i == RTL8192CD_NUM_VWLAN) + (phw->tx_descB + i)->Dword10 = set_desc(phw->tx_ringB_addr); + //(phw->tx_descB + i)->NextTxDescAddress = set_desc(phw->tx_ringB_addr); + else + (phw->tx_descB + i)->Dword10 = set_desc(phw->tx_ringB_addr + (i+1) * sizeof(struct tx_desc)); + //(phw->tx_descB + i)->NextTxDescAddress = set_desc(phw->tx_ringB_addr + (i+1) * sizeof(struct tx_desc)); + } + } +#endif + + //Family add: must re-initialize use_txdesc_cnt after realloc txdesc +#ifdef RESERVE_TXDESC_FOR_EACH_IF + for(i=0; i<=HIGH_QUEUE; ++i) { + priv->use_txdesc_cnt[i] = 0; + } + +#ifdef MBSSID + if (priv->pmib->miscEntry.vap_enable){ + int j; + for (i=0; i<RTL8192CD_NUM_VWLAN; ++i){ + for (j=0; j<=HIGH_QUEUE; ++j){ + priv->pvap_priv[i]->use_txdesc_cnt[j] = 0; + } + } + } +#endif // #ifdef MBSSID +#ifdef UNIVERSAL_REPEATER + if (GET_VXD_PRIV(priv)) { + for(i=0; i<=HIGH_QUEUE; ++i) { + GET_VXD_PRIV(priv)->use_txdesc_cnt[i] = 0; + } + } +#endif // #ifdef UNIVERSAL_REPEATER +#endif // #ifdef RESERVE_TXDESC_FOR_EACH_IF + for(i=0; i<NUM_CMD_DESC; i++) { + if (i == (NUM_CMD_DESC - 1))// set NextAddrs + (phw->txcmd_desc + i)->Dword9 = set_desc(phw->txcmd_ring_addr); + else + (phw->txcmd_desc + i)->Dword9 = set_desc(phw->txcmd_ring_addr + (i+1) * sizeof(struct tx_desc)); + } + + priv->pshare->amsdu_timer_head = priv->pshare->amsdu_timer_tail = 0; + +#ifdef RX_BUFFER_GATHER + INIT_LIST_HEAD(&priv->pshare->gather_list); +#endif + } + + INIT_LIST_HEAD(&priv->wlan_acl_list); + INIT_LIST_HEAD(&priv->wlan_aclpolllist); + + pwlan_acl_poll = priv->pwlan_acl_poll; + for(i=0; i< NUM_ACL; i++) + { + INIT_LIST_HEAD(&(pwlan_acl_poll->aclnode[i].list)); + list_add_tail(&(pwlan_acl_poll->aclnode[i].list), &priv->wlan_aclpolllist); + } + + // copy acl from mib to link list + for (i=0; i<priv->pmib->dot11StationConfigEntry.dot11AclNum; i++) + { + struct list_head *pnewlist; + struct wlan_acl_node *paclnode; + + pnewlist = priv->wlan_aclpolllist.next; + list_del_init(pnewlist); + + paclnode = list_entry(pnewlist, struct wlan_acl_node, list); + memcpy((void *)paclnode->addr, priv->pmib->dot11StationConfigEntry.dot11AclAddr[i], 6); + paclnode->mode = (unsigned char)priv->pmib->dot11StationConfigEntry.dot11AclMode; + + list_add_tail(pnewlist, &priv->wlan_acl_list); + } + + for(i=0; i<NUM_STAT; i++) + INIT_LIST_HEAD(&(priv->stat_hash[i])); + +#ifdef CONFIG_RTK_MESH + /* + * CAUTION !! These statement meshX(virtual interface) ONLY, Maybe modify.... + */ +#ifdef _MESH_ACL_ENABLE_ // copy acl from mib to link list (below code copy above ACL code) + INIT_LIST_HEAD(&priv->mesh_acl_list); + INIT_LIST_HEAD(&priv->mesh_aclpolllist); + + pmesh_acl_poll = priv->pmesh_acl_poll; + for(i=0; i< NUM_MESH_ACL; i++) + { + INIT_LIST_HEAD(&(pmesh_acl_poll->meshaclnode[i].list)); + list_add_tail(&(pmesh_acl_poll->meshaclnode[i].list), &priv->mesh_aclpolllist); + } + + for (i=0; i<priv->pmib->dot1180211sInfo.mesh_acl_num; i++) + { + struct list_head *pnewlist; + struct wlan_acl_node *paclnode; + + pnewlist = priv->mesh_aclpolllist.next; + list_del_init(pnewlist); + + paclnode = list_entry(pnewlist, struct wlan_acl_node, list); + memcpy((void *)paclnode->addr, priv->pmib->dot1180211sInfo.mesh_acl_addr[i], MACADDRLEN); + paclnode->mode = (unsigned char)priv->pmib->dot1180211sInfo.mesh_acl_mode; + + list_add_tail(pnewlist, &priv->mesh_acl_list); + } +#endif + +#ifdef MESH_BOOTSEQ_AUTH + INIT_LIST_HEAD(&(priv->mesh_auth_hdr)); +#endif + + INIT_LIST_HEAD(&(priv->mesh_unEstablish_hdr)); + INIT_LIST_HEAD(&(priv->mesh_mp_hdr)); + + priv->mesh_profile[0].used = FALSE; // Configure by WEB in the future, Maybe delete, Preservation before delete +#endif + + INIT_LIST_HEAD(&(priv->asoc_list)); + INIT_LIST_HEAD(&(priv->auth_list)); + INIT_LIST_HEAD(&(priv->sleep_list)); + INIT_LIST_HEAD(&(priv->defrag_list)); + INIT_LIST_HEAD(&(priv->wakeup_list)); + INIT_LIST_HEAD(&(priv->rx_datalist)); + INIT_LIST_HEAD(&(priv->rx_mgtlist)); + INIT_LIST_HEAD(&(priv->rx_ctrllist)); + INIT_LIST_HEAD(&(priv->addRAtid_list)); // to avoid add RAtid fail + INIT_LIST_HEAD(&(priv->addrssi_list)); + +#ifdef A4_STA + INIT_LIST_HEAD(&(priv->a4_sta_list)); +#endif + +#ifdef CHECK_BEACON_HANGUP + if (priv->reset_hangup) + priv->pshare->beacon_wait_cnt = 1; + else + priv->pshare->beacon_wait_cnt = 2; +#endif + +#ifdef CHECK_HANGUP + if (priv->reset_hangup) { + get_available_channel(priv); + validate_oper_rate(priv); + get_oper_rate(priv); + DOT11_InitQueue(priv->pevent_queue); + return 0; + } +#endif + + // construct operation and basic rates set + { + // validate region domain + if ((priv->pmib->dot11StationConfigEntry.dot11RegDomain < DOMAIN_FCC) || + (priv->pmib->dot11StationConfigEntry.dot11RegDomain >= DOMAIN_MAX)) { + PRINT_INFO("invalid region domain, use default value [DOMAIN_FCC]!\n"); + priv->pmib->dot11StationConfigEntry.dot11RegDomain = DOMAIN_FCC; + } + + // validate band + if (priv->pmib->dot11BssType.net_work_type == 0) { + PRINT_INFO("operation band is not set, use G+B as default!\n"); + priv->pmib->dot11BssType.net_work_type = WIRELESS_11B | WIRELESS_11G; + } + if ((OPMODE & WIFI_AP_STATE) && (priv->pmib->dot11BssType.net_work_type & (WIRELESS_11B | WIRELESS_11G))) { + if (priv->pmib->dot11BssType.net_work_type & WIRELESS_11A) { + priv->pmib->dot11BssType.net_work_type &= (WIRELESS_11B | WIRELESS_11G); + PRINT_INFO("operation band not appropriate, use G/B as default!\n"); + } + } + + if (should_forbid_Nmode(priv)) + priv->pmib->dot11BssType.net_work_type &= ~WIRELESS_11N; + + // validate channel number + if (get_available_channel(priv) == FAIL) { + PRINT_INFO("can't get operation channels, abort!\n"); + return 1; + } + if (priv->pmib->dot11RFEntry.dot11channel != 0) { + for (i=0; i<priv->available_chnl_num; i++) + if (priv->pmib->dot11RFEntry.dot11channel == priv->available_chnl[i]) + break; + if (i == priv->available_chnl_num) { + priv->pmib->dot11RFEntry.dot11channel = priv->available_chnl[0]; + + PRINT_INFO("invalid channel number, use default value [%d]!\n", + priv->pmib->dot11RFEntry.dot11channel); + } + priv->auto_channel = 0; +#ifdef SIMPLE_CH_UNI_PROTOCOL + SET_PSEUDO_RANDOM_NUMBER(priv->mesh_ChannelPrecedence); +#endif + } + else { +#ifdef SIMPLE_CH_UNI_PROTOCOL + if(GET_MIB(priv)->dot1180211sInfo.mesh_enable) + priv->auto_channel = 16; + else +#endif + { + if (OPMODE & WIFI_AP_STATE) + priv->auto_channel = 1; + else + priv->auto_channel = 2; + priv->pmib->dot11RFEntry.dot11channel = priv->available_chnl[0]; + } + } + priv->auto_channel_backup = priv->auto_channel; + + // validate hi and low channel + if (priv->pmib->dot11RFEntry.dot11ch_low != 0) { + for (i=0; i<priv->available_chnl_num; i++) + if (priv->pmib->dot11RFEntry.dot11ch_low == priv->available_chnl[i]) + break; + if (i == priv->available_chnl_num) { + priv->pmib->dot11RFEntry.dot11ch_low = priv->available_chnl[0]; + + PRINT_INFO("invalid low channel number, use default value [%d]!\n", + priv->pmib->dot11RFEntry.dot11ch_low); + } + } + if (priv->pmib->dot11RFEntry.dot11ch_hi != 0) { + for (i=0; i<priv->available_chnl_num; i++) + if (priv->pmib->dot11RFEntry.dot11ch_hi == priv->available_chnl[i]) + break; + if (i == priv->available_chnl_num) { + priv->pmib->dot11RFEntry.dot11ch_hi = priv->available_chnl[priv->available_chnl_num-1]; + + PRINT_INFO("invalid hi channel number, use default value [%d]!\n", + priv->pmib->dot11RFEntry.dot11ch_hi); + } + } + +// Mark the code to auto disable N mode in WEP encrypt +#if 0 + if ((priv->pmib->dot11BssType.net_work_type & WIRELESS_11N) && + (priv->pmib->dot1180211AuthEntry.dot11PrivacyAlgrthm == _WEP_40_PRIVACY_ || + priv->pmib->dot1180211AuthEntry.dot11PrivacyAlgrthm == _WEP_104_PRIVACY_)) + priv->pmib->dot11BssType.net_work_type &= ~WIRELESS_11N; +#endif +//------------------------------ david+2008-01-11 + + // support cck only in channel 14 + if ((priv->pmib->dot11RFEntry.disable_ch14_ofdm) && + (priv->pmib->dot11RFEntry.dot11channel == 14)) { + priv->pmib->dot11BssType.net_work_type = WIRELESS_11B; + PRINT_INFO("support cck only in channel 14!\n"); + } + + // validate and get support and basic rates + validate_oper_rate(priv); + get_oper_rate(priv); + + if (priv->pmib->dot11nConfigEntry.dot11nUse40M && + (!(priv->pmib->dot11BssType.net_work_type & WIRELESS_11N))) { + PRINT_INFO("enable 40M but not in N mode! back to 20M\n"); + priv->pmib->dot11nConfigEntry.dot11nUse40M = 0; + } + + // check deny band + if ((priv->pmib->dot11BssType.net_work_type & (~priv->pmib->dot11StationConfigEntry.legacySTADeny)) == 0) { + PRINT_INFO("legacySTADeny %d not suitable! set to 0\n", priv->pmib->dot11StationConfigEntry.legacySTADeny); + priv->pmib->dot11StationConfigEntry.legacySTADeny = 0; + } + } + + if (!(priv->pmib->dot11BssType.net_work_type & WIRELESS_11N)) { + if (AMSDU_ENABLE) + AMSDU_ENABLE = 0; + if (AMPDU_ENABLE) + AMPDU_ENABLE = 0; +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { + priv->pshare->is_40m_bw = 0; + priv->pshare->offset_2nd_chan = HT_2NDCH_OFFSET_DONTCARE; + } + } + else { +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { + priv->pshare->is_40m_bw = priv->pmib->dot11nConfigEntry.dot11nUse40M; + if (priv->pshare->is_40m_bw == 0) + priv->pshare->offset_2nd_chan = HT_2NDCH_OFFSET_DONTCARE; + else { +#ifdef CONFIG_RTL_92D_SUPPORT + if ((priv->pmib->dot11RFEntry.phyBandSelect & PHY_BAND_5G)&&(GET_CHIP_VER(priv)==VERSION_8192D)) { + priv->pshare->offset_2nd_chan = priv->pmib->dot11nConfigEntry.dot11n2ndChOffset; + } else +#endif + { + if ((priv->pmib->dot11RFEntry.dot11channel < 5) && + (priv->pmib->dot11nConfigEntry.dot11n2ndChOffset == HT_2NDCH_OFFSET_BELOW)) + priv->pshare->offset_2nd_chan = HT_2NDCH_OFFSET_ABOVE; + else if ((priv->pmib->dot11RFEntry.dot11channel > 9) && + (priv->pmib->dot11nConfigEntry.dot11n2ndChOffset == HT_2NDCH_OFFSET_ABOVE)) + priv->pshare->offset_2nd_chan = HT_2NDCH_OFFSET_BELOW; + else + priv->pshare->offset_2nd_chan = priv->pmib->dot11nConfigEntry.dot11n2ndChOffset; + } + } + } + + // force wmm enabled if n mode + // so hostapd should always set wmm_enabled=1 if n mode. + QOS_ENABLE = 1; + } + + // set wep key length + if (priv->pmib->dot1180211AuthEntry.dot11PrivacyAlgrthm == _WEP_40_PRIVACY_) + priv->pmib->dot1180211AuthEntry.dot11PrivacyKeyLen = 8; + else if (priv->pmib->dot1180211AuthEntry.dot11PrivacyAlgrthm == _WEP_104_PRIVACY_) + priv->pmib->dot1180211AuthEntry.dot11PrivacyKeyLen = 16; + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { + init_crc32_table(); // for sw encryption + } + +#ifdef WIFI_WMM + if (QOS_ENABLE) { + if ((OPMODE & WIFI_AP_STATE) +#ifdef CLIENT_MODE + || (OPMODE & WIFI_ADHOC_STATE) +#endif + ) { + GET_EDCA_PARA_UPDATE = 0; + //BK + GET_STA_AC_BK_PARA.AIFSN = 7; + GET_STA_AC_BK_PARA.TXOPlimit = 0; + GET_STA_AC_BK_PARA.ACM = 0; + GET_STA_AC_BK_PARA.ECWmin = 4; + GET_STA_AC_BK_PARA.ECWmax = 10; + //BE + GET_STA_AC_BE_PARA.AIFSN = 3; + GET_STA_AC_BE_PARA.TXOPlimit = 0; + GET_STA_AC_BE_PARA.ACM = 0; + GET_STA_AC_BE_PARA.ECWmin = 4; + GET_STA_AC_BE_PARA.ECWmax = 10; + //VI + GET_STA_AC_VI_PARA.AIFSN = 2; + if ((priv->pmib->dot11BssType.net_work_type & WIRELESS_11G) || + (priv->pmib->dot11BssType.net_work_type & WIRELESS_11A)) + GET_STA_AC_VI_PARA.TXOPlimit = 94; // 3.008ms + else + GET_STA_AC_VI_PARA.TXOPlimit = 188; // 6.016ms + GET_STA_AC_VI_PARA.ACM = 0; + GET_STA_AC_VI_PARA.ECWmin = 3; + GET_STA_AC_VI_PARA.ECWmax = 4; + //VO + GET_STA_AC_VO_PARA.AIFSN = 2; + if ((priv->pmib->dot11BssType.net_work_type & WIRELESS_11G) || + (priv->pmib->dot11BssType.net_work_type & WIRELESS_11A)) + GET_STA_AC_VO_PARA.TXOPlimit = 47; // 1.504ms + else + GET_STA_AC_VO_PARA.TXOPlimit = 102; // 3.264ms + GET_STA_AC_VO_PARA.ACM = 0; + GET_STA_AC_VO_PARA.ECWmin = 2; + GET_STA_AC_VO_PARA.ECWmax = 3; + + //init WMM Para ie in beacon + init_WMM_Para_Element(priv, priv->pmib->dot11QosEntry.WMM_PARA_IE); + } +#ifdef CLIENT_MODE + else if (OPMODE & WIFI_STATION_STATE) + init_WMM_Para_Element(priv, priv->pmib->dot11QosEntry.WMM_IE); // WMM STA +#endif + } +#endif + + i = priv->pmib->dot11ErpInfo.ctsToSelf; + memset(&priv->pmib->dot11ErpInfo, '\0', sizeof(struct erp_mib)); // reset ERP mib + priv->pmib->dot11ErpInfo.ctsToSelf = i; + + if ( (priv->pmib->dot11BssType.net_work_type & WIRELESS_11G) || + (priv->pmib->dot11BssType.net_work_type & WIRELESS_11A) ) + priv->pmib->dot11ErpInfo.shortSlot = 1; + else + priv->pmib->dot11ErpInfo.shortSlot = 0; + + if (OPMODE & WIFI_AP_STATE) { + memcpy(priv->pmib->dot11StationConfigEntry.dot11Bssid, + priv->pmib->dot11OperationEntry.hwaddr, 6); + //priv->oper_band = priv->pmib->dot11BssType.net_work_type; +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { + if (!priv->pshare->ra40MLowerMinus && (priv->pshare->rf_ft_var.raGoUp40MLower > 5) && + (priv->pshare->rf_ft_var.raGoDown40MLower > 5)) { + priv->pshare->rf_ft_var.raGoUp40MLower-=5; + priv->pshare->rf_ft_var.raGoDown40MLower-=5; + priv->pshare->ra40MLowerMinus++; + } +#ifdef HIGH_POWER_EXT_PA + if (!priv->pshare->raThdHP_Minus) { + + if( priv->pshare->rf_ft_var.use_ext_pa ) { + if(priv->pshare->rf_ft_var.raGoDownUpper > RSSI_DIFF_PA) { + priv->pshare->rf_ft_var.raGoDownUpper -= RSSI_DIFF_PA; + priv->pshare->rf_ft_var.raGoUpUpper -= RSSI_DIFF_PA; + } + if(priv->pshare->rf_ft_var.raGoDown20MLower > RSSI_DIFF_PA) { + priv->pshare->rf_ft_var.raGoDown20MLower -= RSSI_DIFF_PA; + priv->pshare->rf_ft_var.raGoUp20MLower -= RSSI_DIFF_PA; + } + if(priv->pshare->rf_ft_var.raGoDown40MLower > RSSI_DIFF_PA) { + priv->pshare->rf_ft_var.raGoDown40MLower -= RSSI_DIFF_PA; + priv->pshare->rf_ft_var.raGoUp40MLower -= RSSI_DIFF_PA; + } + } + ++priv->pshare->raThdHP_Minus; + } +#endif + } + } +#ifdef CLIENT_MODE + else { + if (priv->pmib->dot11StationConfigEntry.dot11DefaultSSIDLen == 0) { + priv->pmib->dot11StationConfigEntry.dot11DefaultSSIDLen = 11; + memcpy(priv->pmib->dot11StationConfigEntry.dot11DefaultSSID, "defaultSSID", 11); + } + memset(priv->pmib->dot11StationConfigEntry.dot11Bssid, 0, 6); + priv->join_res = STATE_Sta_No_Bss; + +// Add mac clone address manually ---------- +#ifdef RTK_BR_EXT + if (priv->pmib->ethBrExtInfo.macclone_enable == 2) { + extern void mac_clone(struct rtl8192cd_priv *priv, unsigned char *addr); + mac_clone(priv, priv->pmib->ethBrExtInfo.nat25_dmzMac); + priv->macclone_completed = 1; + } +#endif +//------------------------- david+2007-5-31 + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { + if (priv->pshare->ra40MLowerMinus) { + priv->pshare->rf_ft_var.raGoUp40MLower+=5; + priv->pshare->rf_ft_var.raGoDown40MLower+=5; + priv->pshare->ra40MLowerMinus = 0; + } + } + } +#endif + + // initialize event queue + DOT11_InitQueue(priv->pevent_queue); +#ifdef CONFIG_RTL_WAPI_SUPPORT + DOT11_InitQueue(priv->wapiEvent_queue); +#ifdef MBSSID + if (IS_ROOT_INTERFACE(priv)&&priv->pmib->miscEntry.vap_enable) { + for (i=0; i<RTL8192CD_NUM_VWLAN; i++) + DOT11_InitQueue(priv->pvap_priv[i]->wapiEvent_queue); + } +#endif + +#endif + +#ifdef CONFIG_RTK_MESH + if(GET_MIB(priv)->dot1180211sInfo.mesh_enable == 1) // plus add 0217, not mesh mode should not do below function + { + DOT11_InitQueue2(priv->pathsel_queue, MAXQUEUESIZE2, MAXDATALEN2); +#ifdef _11s_TEST_MODE_ + DOT11_InitQueue2(priv->receiver_queue, MAXQUEUESIZE2, MAXDATALEN2); +#endif + //modify by Joule for SECURITY + i = priv->pmib->dot11sKeysTable.dot11Privacy; + memset(&priv->pmib->dot11sKeysTable, '\0', sizeof(struct Dot11KeyMappingsEntry)); // reset key + priv->pmib->dot11sKeysTable.dot11Privacy = i; + } // +#endif + +#ifdef __DRAYTEK_OS__ + if (priv->pmib->dot1180211AuthEntry.dot11PrivacyAlgrthm != _TKIP_PRIVACY_ && + priv->pmib->dot1180211AuthEntry.dot11PrivacyAlgrthm != _CCMP_PRIVACY_ && + priv->pmib->dot1180211AuthEntry.dot11PrivacyAlgrthm != _WEP_WPA_MIXED_PRIVACY_) { +#ifdef UNIVERSAL_REPEATER + if (IS_ROOT_INTERFACE(priv)) +#endif + { + priv->pmib->dot11RsnIE.rsnielen = 0; // reset RSN IE length + memset(&priv->pmib->dot11GroupKeysTable, '\0', sizeof(struct Dot11KeyMappingsEntry)); // reset group key +#ifdef UNIVERSAL_REPEATER + if (GET_VXD_PRIV(priv)) + GET_VXD_PRIV(priv)->pmib->dot11RsnIE.rsnielen = 0; +#endif + } + } +#endif + + i = RC_ENTRY_NUM; + for (;;) { + if (priv->pmib->reorderCtrlEntry.ReorderCtrlWinSz >= i) { + priv->pmib->reorderCtrlEntry.ReorderCtrlWinSz = i; + break; + } + else if (i > 8) + i = i / 2; + else { + priv->pmib->reorderCtrlEntry.ReorderCtrlWinSz = 8; + break; + } + } + + // Realtek proprietary IE + memcpy(&(priv->pshare->rtk_ie_buf[0]), Realtek_OUI, 3); + priv->pshare->rtk_ie_buf[3] = 2; + priv->pshare->rtk_ie_buf[4] = 1; + priv->pshare->rtk_ie_buf[5] = 0; + priv->pshare->rtk_ie_buf[5] |= RTK_CAP_IE_WLAN_88C92C | RTK_CAP_IE_WLAN_8192SE; +#ifdef CLIENT_MODE + if (OPMODE & WIFI_STATION_STATE) + priv->pshare->rtk_ie_buf[5] |= RTK_CAP_IE_AP_CLIENT; +#endif + priv->pshare->rtk_ie_len = 6; + + +#ifdef WIFI_HAPD + if ((priv->pmib->dot1180211AuthEntry.dot11EnablePSK == 0) + && (priv->pmib->dot118021xAuthEntry.dot118021xAlgrthm) + && (priv->pmib->dot1180211AuthEntry.dot11WPACipher || priv->pmib->dot1180211AuthEntry.dot11WPA2Cipher)) + rsn_init(priv); +#endif + +#if defined(INCLUDE_WPA_PSK) || defined(WIFI_HAPD) + if (priv->pmib->dot1180211AuthEntry.dot11EnablePSK) + psk_init(priv); + +#ifdef WDS +#ifdef UNIVERSAL_REPEATER + if (IS_ROOT_INTERFACE(priv)) +#endif + { +#ifdef LAZY_WDS + if (priv->pmib->dot11WdsInfo.wdsEnabled == WDS_LAZY_ENABLE) { + priv->pmib->dot11WdsInfo.wdsNum = 0; + memset(priv->pmib->dot11WdsInfo.entry, '\0', sizeof(struct wdsEntry)*NUM_WDS); + } +#endif + if ((priv->pmib->dot11WdsInfo.wdsEnabled) && (priv->pmib->dot11WdsInfo.wdsNum > 0) && + ((priv->pmib->dot11WdsInfo.wdsPrivacy == _TKIP_PRIVACY_) || + (priv->pmib->dot11WdsInfo.wdsPrivacy == _CCMP_PRIVACY_))) + wds_psk_init(priv); + } +#endif +#endif + +#ifdef WIFI_WPAS + + { + WPAS_ASSOCIATION_INFO Assoc_Info; + memset((void *)&Assoc_Info, 0, sizeof(struct _WPAS_ASSOCIATION_INFO)); + Assoc_Info.ReqIELen = priv->pmib->dot11RsnIE.rsnie[1]+ 2; + memcpy(Assoc_Info.ReqIE, priv->pmib->dot11RsnIE.rsnie, Assoc_Info.ReqIELen); + + //event_indicate_wpas(priv, NULL, WPAS_ASSOC_INFO, (UINT8 *)&Assoc_Info); + } + +#endif + + +#ifdef GBWC + priv->GBWC_tx_queue.head = 0; + priv->GBWC_tx_queue.tail = 0; + priv->GBWC_rx_queue.head = 0; + priv->GBWC_rx_queue.tail = 0; + priv->GBWC_tx_count = 0; + priv->GBWC_rx_count = 0; + priv->GBWC_consuming_Q = 0; +#endif + + priv->release_mcast = 0; + +#ifdef USB_PKT_RATE_CTRL_SUPPORT //mark_test + priv->change_toggle = 0; + priv->pre_pkt_cnt = 0; + priv->pkt_nsec_diff = 0; + priv->poll_usb_cnt = 0; + priv->auto_rate_mask = 0; +#endif + +#ifdef STA_EXT + priv->pshare->fw_free_space = FW_NUM_STAT - 2; // One for MAGANEMENT_AID, one for other STAs +#endif + +#ifdef CONFIG_RTK_VLAN_SUPPORT + if (priv->pmib->vlan.global_vlan) + priv->pmib->dot11OperationEntry.disable_brsc = 1; +#endif + + +#ifdef CONFIG_RTL_WAPI_SUPPORT + if (priv->pmib->wapiInfo.wapiType!=wapiDisable) + { + /* set NMK */ + GenerateRandomData(priv->wapiNMK, WAPI_KEY_LEN); + priv->wapiMCastKeyId = 0; + priv->wapiMCastKeyUpdate = 0; + priv->wapiWaiTxSeq = 0; + wapiInit(priv); + } + +#ifdef MBSSID + if (IS_ROOT_INTERFACE(priv)&&priv->pmib->miscEntry.vap_enable) { + for (i=0; i<RTL8192CD_NUM_VWLAN; i++) + if (priv->pvap_priv[i]->pmib->wapiInfo.wapiType!=wapiDisable) { + /* set NMK */ + GenerateRandomData(priv->pvap_priv[i]->wapiNMK, WAPI_KEY_LEN); + priv->pvap_priv[i]->wapiMCastKeyId = 0; + priv->pvap_priv[i]->wapiMCastKeyUpdate = 0; + priv->wapiWaiTxSeq = 0; + wapiInit(priv->pvap_priv[i]); + } + } +#endif +#endif + +#ifdef MBSSID + // if vap enabled, set beacon int to 100 at minimun when guest ssid num <= 4 + // if vap enabled, set beacon int to 200 at minimun when guest ssid num > 4 + { + int ssid_num = 1, minbcn_period; + priv->bcn_period_bak = priv->pmib->dot11StationConfigEntry.dot11BeaconPeriod; + + if ((OPMODE & WIFI_AP_STATE) && GET_ROOT(priv)->pmib->miscEntry.vap_enable) + { + for (i=0; i<RTL8192CD_NUM_VWLAN; i++) + { + if (GET_ROOT(priv)->pvap_priv[i] && IS_DRV_OPEN(GET_ROOT(priv)->pvap_priv[i])) + { + ssid_num++; + } + } + } + + if (ssid_num >= 5) + minbcn_period = 200; + else + minbcn_period = 100; + + // if vap enabled, set beacon int to 100 at minimun + if ((OPMODE & WIFI_AP_STATE) && GET_ROOT(priv)->pmib->miscEntry.vap_enable + && priv->pmib->dot11StationConfigEntry.dot11BeaconPeriod < minbcn_period) + priv->pmib->dot11StationConfigEntry.dot11BeaconPeriod = minbcn_period; + + if ((OPMODE & WIFI_AP_STATE) && GET_ROOT(priv)->pmib->miscEntry.vap_enable) + { + for (i=0; i<RTL8192CD_NUM_VWLAN; i++) + { + if (GET_ROOT(priv)->pvap_priv[i]) + { + GET_ROOT(priv)->pvap_priv[i]->pmib->dot11StationConfigEntry.dot11BeaconPeriod = priv->pmib->dot11StationConfigEntry.dot11BeaconPeriod; + GET_ROOT(priv)->pvap_priv[i]->update_bcn_period = 1; + } + } + + GET_ROOT(priv)->pmib->dot11StationConfigEntry.dot11BeaconPeriod = priv->pmib->dot11StationConfigEntry.dot11BeaconPeriod; + GET_ROOT(priv)->update_bcn_period = 1; + + } + } +#endif + +#ifdef DOT11D +{ + extern COUNTRY_IE_ELEMENT countryIEArray[]; + int found = 0; + char *CStringPtr = priv->pmib->dot11dCountry.dot11CountryString ; + + if ((OPMODE & WIFI_AP_STATE) && COUNTRY_CODE_ENABLED) { + for (i=0; i<COUNTRYNUMBER; i++) { + if (!memcmp(CStringPtr, countryIEArray[i].countryA2, 2)) { + priv->pshare->countryTabIdx = i; + found = 1; + break; + } + } + + if (found == 0) { + priv->pshare->countryTabIdx = 1; + printk("can't found country code(%s),use default region\n",CStringPtr); + } + + i = priv->pshare->countryTabIdx; + + if (countryIEArray[i].A_Band_Region == 0) { + priv->pshare->countryBandUsed=0; /*2.4G*/ + } else { + /*5G*/ + if (priv->pmib->dot11BssType.net_work_type & WIRELESS_11A) { + priv->pshare->countryBandUsed = 1; /*5G*/ + //printk("2.4G and 5G both enabled, select 5G \n"); + } else { + priv->pshare->countryBandUsed = 0; /*2.4G*/ + //printk("2.4G and 5G both enabled, select 2.4G \n"); + } + } + } +} +#endif + +#ifdef RTL_MANUAL_EDCA + for (i=0; i<8; i++) { + if ((priv->pmib->dot11QosEntry.TID_mapping[i] < 1) || (priv->pmib->dot11QosEntry.TID_mapping[i] > 4)) + priv->pmib->dot11QosEntry.TID_mapping[i] = 2; + } +#endif + +#ifdef TXREPORT + priv->pshare->sta_query_idx=-1; + + // Init StaDetectInfo to detect disappearing STA. Added by Annie, 2010-08-10. + priv->pmib->staDetectInfo.txRprDetectPeriod = 1; +#endif + +#ifdef INCLUDE_WPS +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + wps_init(priv); +#endif + + return 0; +} + + +static int rtl8192cd_stop_sw(struct rtl8192cd_priv *priv) +{ + struct rtl8192cd_hw *phw; + unsigned long flags; + int i; + + // we hope all this can be done in critical section + SAVE_INT_AND_CLI(flags); + +#ifdef INCLUDE_WPS + priv->pshare->WSC_CONT_S.wait_reinit = 1 ; +#endif + + if (timer_pending(&priv->frag_to_filter)) + del_timer_sync(&priv->frag_to_filter); + +#ifdef DETECT_STA_EXISTANCE + // Added by Annie for Retry Limit Recovery Timer, 2010-08-10. + if (timer_pending(&priv->pshare->rl_recover_timer)) + del_timer_sync (&priv->pshare->rl_recover_timer); +#endif + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { + if (timer_pending(&priv->expire_timer)) + del_timer_sync(&priv->expire_timer); +#ifdef SW_ANT_SWITCH + if (timer_pending(&priv->pshare->swAntennaSwitchTimer)) + del_timer_sync(&priv->pshare->swAntennaSwitchTimer); +#endif + if (timer_pending(&priv->pshare->rc_sys_timer)) + del_timer_sync(&priv->pshare->rc_sys_timer); +#if 0 + if (timer_pending(&priv->pshare->phw->tpt_timer)) + del_timer_sync(&priv->pshare->phw->tpt_timer); +#endif +#ifdef CONFIG_RTL_WAPI_SUPPORT + if (priv->pmib->wapiInfo.wapiType!=wapiDisable +#ifdef CHECK_HANGUP + &&(!priv->reset_hangup) +#endif + ) + { + wapiExit(priv); + } + + +#ifdef MBSSID + if( priv->pmib->miscEntry.vap_enable) { + for (i=0; i<RTL8192CD_NUM_VWLAN; i++){ + if (priv->pvap_priv[i]->pmib->wapiInfo.wapiType!=wapiDisable +#ifdef CHECK_HANGUP + &&(!priv->pvap_priv[i]->reset_hangup) +#endif + ) { + wapiExit(priv->pvap_priv[i]); + } + } + } +#endif +#endif + +#ifdef PCIE_POWER_SAVING + if (timer_pending(&priv->ps_timer)) + del_timer_sync(&priv->ps_timer); +#endif +#ifdef CONFIG_RTK_MESH + /* + * CAUTION !! These statement meshX(virtual interface) ONLY, Maybe modify.... + */ + if (timer_pending(&priv->mesh_peer_link_timer)) + del_timer_sync(&priv->mesh_peer_link_timer); + +#ifdef MESH_BOOTSEQ_AUTH + if (timer_pending(&priv->mesh_auth_timer)) + del_timer_sync(&priv->mesh_auth_timer); +#endif +#endif + } + if (timer_pending(&priv->ss_timer)) + del_timer_sync(&priv->ss_timer); + if (timer_pending(&priv->MIC_check_timer)) + del_timer_sync(&priv->MIC_check_timer); + if (timer_pending(&priv->assoc_reject_timer)) + del_timer_sync(&priv->assoc_reject_timer); + // to avoid add RAtid fail + if (timer_pending(&priv->add_RATid_timer)) + del_timer_sync(&priv->add_RATid_timer); + if (timer_pending(&priv->add_rssi_timer)) + del_timer_sync(&priv->add_rssi_timer); + if (timer_pending(&priv->add_ps_timer)) + del_timer_sync(&priv->add_ps_timer); + + +#if defined(CONFIG_RTL_92D_SUPPORT) && defined(CONFIG_RTL_NOISE_CONTROL) + if (GET_CHIP_VER(priv) == VERSION_8192D){ + if (timer_pending(&priv->dnc_timer)) + del_timer_sync(&priv->dnc_timer); + } +#endif + +#ifdef SMART_CONCURRENT_92D + if (GET_CHIP_VER(priv) == VERSION_8192D){ + if (timer_pending(&priv->smcc_prb_timer)) + del_timer_sync(&priv->smcc_prb_timer); + } +#endif + +#ifdef CLIENT_MODE + if (timer_pending(&priv->reauth_timer)) + del_timer_sync(&priv->reauth_timer); + if (timer_pending(&priv->reassoc_timer)) + del_timer_sync(&priv->reassoc_timer); + if (timer_pending(&priv->idle_timer)) + del_timer_sync(&priv->idle_timer); +#endif + +#ifdef GBWC + if (timer_pending(&priv->GBWC_timer)) + del_timer_sync(&priv->GBWC_timer); + while (CIRC_CNT(priv->GBWC_tx_queue.head, priv->GBWC_tx_queue.tail, NUM_TXPKT_QUEUE)) + { + struct sk_buff *pskb = priv->GBWC_tx_queue.pSkb[priv->GBWC_tx_queue.tail]; + rtl_kfree_skb(priv, pskb, _SKB_TX_); + priv->GBWC_tx_queue.tail++; + priv->GBWC_tx_queue.tail = priv->GBWC_tx_queue.tail & (NUM_TXPKT_QUEUE - 1); + } + while (CIRC_CNT(priv->GBWC_rx_queue.head, priv->GBWC_rx_queue.tail, NUM_TXPKT_QUEUE)) + { + struct sk_buff *pskb = priv->GBWC_rx_queue.pSkb[priv->GBWC_rx_queue.tail]; + rtl_kfree_skb(priv, pskb, _SKB_RX_); + priv->GBWC_rx_queue.tail++; + priv->GBWC_rx_queue.tail = priv->GBWC_rx_queue.tail & (NUM_TXPKT_QUEUE - 1); + } +#endif + +#ifdef INCLUDE_WPA_PSK + if (timer_pending(&priv->wpa_global_info->GKRekeyTimer)) + del_timer_sync(&priv->wpa_global_info->GKRekeyTimer); +#endif + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { +#ifdef DFS + if (timer_pending(&priv->DFS_timer)) + del_timer_sync(&priv->DFS_timer); + + if (timer_pending(&priv->ch_avail_chk_timer)) + del_timer_sync(&priv->ch_avail_chk_timer); + + /* + * when we disable the DFS function dynamically, we also remove the channel + * from NOP_chnl while the driver is rebooting + */ + if (timer_pending(&priv->ch52_timer) && + (priv->pmib->dot11DFSEntry.disable_DFS)) { + del_timer_sync(&priv->ch52_timer); + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 52); + } + + if (timer_pending(&priv->ch56_timer) && + (priv->pmib->dot11DFSEntry.disable_DFS)) { + del_timer_sync(&priv->ch56_timer); + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 56); + } + + if (timer_pending(&priv->ch60_timer) && + (priv->pmib->dot11DFSEntry.disable_DFS)) { + del_timer_sync(&priv->ch60_timer); + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 60); + } + + if (timer_pending(&priv->ch64_timer) && + (priv->pmib->dot11DFSEntry.disable_DFS)) { + del_timer_sync(&priv->ch64_timer); + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 64); + } + + if (timer_pending(&priv->ch100_timer) && + (priv->pmib->dot11DFSEntry.disable_DFS)) { + del_timer_sync(&priv->ch100_timer); + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 100); + } + + if (timer_pending(&priv->ch104_timer) && + (priv->pmib->dot11DFSEntry.disable_DFS)) { + del_timer_sync(&priv->ch104_timer); + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 104); + } + + if (timer_pending(&priv->ch108_timer) && + (priv->pmib->dot11DFSEntry.disable_DFS)) { + del_timer_sync(&priv->ch108_timer); + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 108); + } + + if (timer_pending(&priv->ch112_timer) && + (priv->pmib->dot11DFSEntry.disable_DFS)) { + del_timer_sync(&priv->ch112_timer); + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 112); + } + + if (timer_pending(&priv->ch116_timer) && + (priv->pmib->dot11DFSEntry.disable_DFS)) { + del_timer_sync(&priv->ch116_timer); + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 116); + } + + if (timer_pending(&priv->ch120_timer) && + (priv->pmib->dot11DFSEntry.disable_DFS)) { + del_timer_sync(&priv->ch120_timer); + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 120); + } + + if (timer_pending(&priv->ch124_timer) && + (priv->pmib->dot11DFSEntry.disable_DFS)) { + del_timer_sync(&priv->ch124_timer); + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 124); + } + + if (timer_pending(&priv->ch128_timer) && + (priv->pmib->dot11DFSEntry.disable_DFS)) { + del_timer_sync(&priv->ch128_timer); + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 128); + } + + if (timer_pending(&priv->ch132_timer) && + (priv->pmib->dot11DFSEntry.disable_DFS)) { + del_timer_sync(&priv->ch128_timer); + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 128); + } + + if (timer_pending(&priv->ch136_timer) && + (priv->pmib->dot11DFSEntry.disable_DFS)) { + del_timer_sync(&priv->ch136_timer); + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 136); + } + + if (timer_pending(&priv->ch140_timer) && + (priv->pmib->dot11DFSEntry.disable_DFS)) { + del_timer_sync(&priv->ch140_timer); + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 140); + } + + /* + * For JAPAN in adhoc mode + */ + if (((priv->pmib->dot11StationConfigEntry.dot11RegDomain == DOMAIN_MKK) || + (priv->pmib->dot11StationConfigEntry.dot11RegDomain == DOMAIN_MKK3)) && + (OPMODE & WIFI_ADHOC_STATE)) { + if (priv->pmib->dot11BssType.net_work_type & WIRELESS_11A) { + if (!timer_pending(&priv->ch52_timer)) + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 52); + if (!timer_pending(&priv->ch56_timer)) + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 56); + if (!timer_pending(&priv->ch60_timer)) + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 60); + if (!timer_pending(&priv->ch64_timer)) + RemoveChannel(priv->NOP_chnl, &priv->NOP_chnl_num, 64); + } + } +#endif // DFS + + // for SW LED + if ((LED_TYPE >= LEDTYPE_SW_LINK_TXRX) && (LED_TYPE < LEDTYPE_SW_MAX)) + disable_sw_LED(priv); + +#ifdef __KERNEL__ +#ifdef DFS + /* prevent killing tasklet issue in interrupt */ + if (!priv->pmib->dot11DFSEntry.DFS_detected) +#endif + { +#ifdef CHECK_HANGUP + if (!priv->reset_hangup) +#endif + { + tasklet_kill(&priv->pshare->rx_tasklet); + tasklet_kill(&priv->pshare->tx_tasklet); + tasklet_kill(&priv->pshare->oneSec_tasklet); + } + } +#endif // __KERNEL__ + + phw = GET_HW(priv); + +#ifdef DELAY_REFILL_RX_BUF + priv->pshare->phw->cur_rx_refill = priv->pshare->phw->cur_rx = 0; // avoid refill to rx ring +#endif + for (i=0; i<NUM_RX_DESC; i++) + { + if (phw->rx_infoL[i].pbuf != NULL) { +#ifdef CONFIG_NET_PCI + // if pci bios, then pci_unmap_single and dev_kfree_skb + if (IS_PCIBIOS_TYPE) + pci_unmap_single(priv->pshare->pdev, phw->rx_infoL[i].paddr, (RX_BUF_LEN - sizeof(struct rx_frinfo)), PCI_DMA_FROMDEVICE); +#endif + rtl_kfree_skb(priv, (struct sk_buff*)(phw->rx_infoL[i].pbuf), _SKB_RX_); + } + } + + // free the skb buffer in Low and Hi queue + DEBUG_INFO("free tx Q0 head %d tail %d\n", phw->txhead0, phw->txtail0); + DEBUG_INFO("free tx Q1 head %d tail %d\n", phw->txhead1, phw->txtail1); + DEBUG_INFO("free tx Q2 head %d tail %d\n", phw->txhead2, phw->txtail2); + DEBUG_INFO("free tx Q3 head %d tail %d\n", phw->txhead3, phw->txtail3); + DEBUG_INFO("free tx Q4 head %d tail %d\n", phw->txhead4, phw->txtail4); + DEBUG_INFO("free tx Q5 head %d tail %d\n", phw->txhead5, phw->txtail5); + + for (i=0; i<NUM_TX_DESC; i++) + { + // free tx queue skb + struct tx_desc_info *tx_info; + int j; + int head, tail; + + for (j=0; j<=HIGH_QUEUE; j++) { + head = get_txhead(phw, j); + tail = get_txtail(phw, j); +// if (i <tail || i >= head) + if( (tail < head) ? (i <tail || i >= head) :(i <tail && i >= head)) + continue; + + tx_info = get_txdesc_info(priv->pshare->pdesc_info, j); + +#if defined(RESERVE_TXDESC_FOR_EACH_IF) && (defined(UNIVERSAL_REPEATER) || defined(MBSSID)) //Family add + if (tx_info[i].priv) + tx_info[i].priv->use_txdesc_cnt[j]--; +#endif + + if (tx_info[i].pframe && + (tx_info[i].type == _SKB_FRAME_TYPE_)) { +#ifdef CONFIG_NET_PCI + if (IS_PCIBIOS_TYPE) + pci_unmap_single(priv->pshare->pdev, tx_info[i].paddr, (tx_info[i].len), PCI_DMA_TODEVICE); +#endif + rtl_kfree_skb(priv, tx_info[i].pframe, _SKB_TX_); + DEBUG_INFO("free skb in queue %d\n", j); + } + } + } // TX descriptor Free + +#ifdef CONFIG_NET_PCI + // unmap beacon buffer + if (IS_PCIBIOS_TYPE) { + pci_unmap_single(priv->pshare->pdev, get_desc(phw->tx_descB->Dword8), + 128*sizeof(unsigned int), PCI_DMA_TODEVICE); + } +#endif + +#ifdef RX_BUFFER_GATHER + flush_rx_list(priv); +#endif + } // if (IS_ROOT_INTERFACE(priv)) + + for (i=0; i<NUM_STAT; i++) + { + if (priv->pshare->aidarray[i]) { +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (priv != priv->pshare->aidarray[i]->priv) + continue; + else +#endif + { + if (priv->pshare->aidarray[i]->used == TRUE) + if (free_stainfo(priv, &(priv->pshare->aidarray[i]->station)) == FALSE) + DEBUG_ERR("free station %d fails\n", i); + +#if defined(WIFI_WMM) && defined(WMM_APSD) +#ifdef PRIV_STA_BUF + free_sta_que(priv, priv->pshare->aidarray[i]->station.VO_dz_queue); + free_sta_que(priv, priv->pshare->aidarray[i]->station.VI_dz_queue); + free_sta_que(priv, priv->pshare->aidarray[i]->station.BE_dz_queue); + free_sta_que(priv, priv->pshare->aidarray[i]->station.BK_dz_queue); + +#else + kfree(priv->pshare->aidarray[i]->station.VO_dz_queue); + kfree(priv->pshare->aidarray[i]->station.VI_dz_queue); + kfree(priv->pshare->aidarray[i]->station.BE_dz_queue); + kfree(priv->pshare->aidarray[i]->station.BK_dz_queue); +#endif +#endif + +#if defined(WIFI_WMM) +#ifdef PRIV_STA_BUF + free_sta_mgt_que(priv, priv->pshare->aidarray[i]->station.MGT_dz_queue); +#else + kfree(priv->pshare->aidarray[i]->station.MGT_dz_queue); +#endif +#endif + +#ifdef INCLUDE_WPA_PSK +#ifdef PRIV_STA_BUF + free_wpa_buf(priv, priv->pshare->aidarray[i]->station.wpa_sta_info); +#else + kfree(priv->pshare->aidarray[i]->station.wpa_sta_info); +#endif +#endif +#ifdef RTL8192CD_VARIABLE_USED_DMEM + { + unsigned int index = (unsigned int)i; + rtl8192cd_dmem_free(AID_OBJ, &index); + } +#else +#ifdef PRIV_STA_BUF + free_sta_obj(priv, priv->pshare->aidarray[i]); +#else + kfree(priv->pshare->aidarray[i]); +#endif +#endif + priv->pshare->aidarray[i] = NULL; + } + } + } + +#ifndef __DRAYTEK_OS__ + // reset rsnie and group key from open to here, david +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { +#ifdef WIFI_SIMPLE_CONFIG + if (!priv->pmib->dot11OperationEntry.keep_rsnie) { + priv->pmib->wscEntry.beacon_ielen = 0; + priv->pmib->wscEntry.probe_rsp_ielen = 0; + priv->pmib->wscEntry.probe_req_ielen = 0; + priv->pmib->wscEntry.assoc_ielen = 0; + } + +// if (!(OPMODE & WIFI_AP_STATE)) +// priv->pmib->dot11OperationEntry.keep_rsnie = 1; +#endif + } + + if (!priv->pmib->dot11OperationEntry.keep_rsnie) { + priv->pmib->dot11RsnIE.rsnielen = 0; // reset RSN IE length + memset(&priv->pmib->dot11GroupKeysTable, '\0', sizeof(struct Dot11KeyMappingsEntry)); // reset group key +#ifdef UNIVERSAL_REPEATER + if (GET_VXD_PRIV(priv)) + GET_VXD_PRIV(priv)->pmib->dot11RsnIE.rsnielen = 0; +#endif + priv->auto_channel_backup = 0; + } + else { + // When wlan scheduling and auto-chan case, it will disable/enable + // wlan interface directly w/o re-set mib. Therefore, we need use + // "keep_rsnie" flag to keep auto-chan value + + if ( +#ifdef CHECK_HANGUP + !priv->reset_hangup && +#endif + priv->auto_channel_backup) + priv->pmib->dot11RFEntry.dot11channel = 0; + } +#endif + +#ifdef MBSSID + if (GET_ROOT(priv)->pmib->miscEntry.vap_enable) + { + if (IS_VAP_INTERFACE(priv) && !priv->pmib->dot11OperationEntry.keep_rsnie) { + priv->pmib->dot11RsnIE.rsnielen = 0; // reset RSN IE length + memset(&priv->pmib->dot11GroupKeysTable, '\0', sizeof(struct Dot11KeyMappingsEntry)); // reset group key + } + } + + priv->pmib->dot11StationConfigEntry.dot11BeaconPeriod = priv->bcn_period_bak; +#endif + +#ifdef RTK_BR_EXT + if (OPMODE & (WIFI_STATION_STATE | WIFI_ADHOC_STATE)) { +#ifdef CHECK_HANGUP + if (!priv->reset_hangup) +#endif + nat25_db_cleanup(priv); + } +#endif + +#ifdef A4_STA + if (OPMODE & WIFI_AP_STATE) { +#ifdef CHECK_HANGUP + if (!priv->reset_hangup) +#endif + a4_sta_cleanup(priv); + } +#endif + + { + int hd, tl; + struct sk_buff *pskb; + + hd = priv->dz_queue.head; + tl = priv->dz_queue.tail; + while (CIRC_CNT(hd, tl, NUM_TXPKT_QUEUE)) + { + pskb = priv->dz_queue.pSkb[tl]; + rtl_kfree_skb(priv, pskb, _SKB_TX_); + tl++; + tl = tl & (NUM_TXPKT_QUEUE - 1); + } + priv->dz_queue.head = 0; + priv->dz_queue.tail = 0; + } + +#ifdef MP_TEST + if (!priv->pshare->rf_ft_var.mp_specific) +#endif + priv->pshare->ThermalValue = 0; + + RESTORE_INT(flags); + return 0; +} + + +#ifdef MBSSID +static void rtl8192cd_init_vap_mib(struct rtl8192cd_priv *priv) +{ +#if 0 + unsigned char tmpbuf[36], hwaddr[6]; + int len; + //unsigned int AclMode; + //unsigned char AclAddr[NUM_ACL][MACADDRLEN]; + //unsigned int AclNum; + struct Dot1180211AuthEntry dot1180211AuthEntry; + struct Dot118021xAuthEntry dot118021xAuthEntry; + struct Dot11DefaultKeysTable dot11DefaultKeysTable; + struct Dot11RsnIE dot11RsnIE; + + // backup mib that can differ from root interface + memcpy(hwaddr, GET_MY_HWADDR, 6); + len = SSID_LEN; + memcpy(tmpbuf, SSID, len); + //AclMode = priv->pmib->dot11StationConfigEntry.dot11AclMode; + //memcpy(AclAddr, priv->pmib->dot11StationConfigEntry.dot11AclAddr, sizeof(AclAddr)); + //AclNum = priv->pmib->dot11StationConfigEntry.dot11AclNum; + memcpy(&dot1180211AuthEntry, &priv->pmib->dot1180211AuthEntry, sizeof(struct Dot1180211AuthEntry)); + memcpy(&dot118021xAuthEntry, &priv->pmib->dot118021xAuthEntry, sizeof(struct Dot118021xAuthEntry)); + memcpy(&dot11DefaultKeysTable, &priv->pmib->dot11DefaultKeysTable, sizeof(struct Dot11DefaultKeysTable)); + memcpy(&dot11RsnIE, &priv->pmib->dot11RsnIE, sizeof(struct Dot11RsnIE)); + + // copy mib from root interface + memcpy(priv->pmib, GET_ROOT_PRIV(priv)->pmib, sizeof(struct wifi_mib)); + + // restore the different part + memcpy(GET_MY_HWADDR, hwaddr, 6); + SSID_LEN = len; + memcpy(SSID, tmpbuf, len); + SSID2SCAN_LEN = len; + memcpy(SSID2SCAN, SSID, len); + //priv->pmib->dot11StationConfigEntry.dot11AclMode = AclMode; + //memcpy(priv->pmib->dot11StationConfigEntry.dot11AclAddr, AclAddr, sizeof(AclAddr)); + //priv->pmib->dot11StationConfigEntry.dot11AclNum = AclNum; + memcpy(&priv->pmib->dot1180211AuthEntry, &dot1180211AuthEntry, sizeof(struct Dot1180211AuthEntry)); + memcpy(&priv->pmib->dot118021xAuthEntry, &dot118021xAuthEntry, sizeof(struct Dot118021xAuthEntry)); + memcpy(&priv->pmib->dot11DefaultKeysTable, &dot11DefaultKeysTable, sizeof(struct Dot11DefaultKeysTable)); + memcpy(&priv->pmib->dot11RsnIE, &dot11RsnIE, sizeof(struct Dot11RsnIE)); +#endif + + // copy mib_rf from root interface + memcpy(&priv->pmib->dot11RFEntry, &GET_ROOT_PRIV(priv)->pmib->dot11RFEntry, sizeof(struct Dot11RFEntry)); + + // special mib that need to set +#ifdef WIFI_WMM + //QOS_ENABLE = 0; +#ifdef WMM_APSD + APSD_ENABLE = 0; +#endif +#endif + +#ifdef WDS + // always disable wds in vap + priv->pmib->dot11WdsInfo.wdsEnabled = 0; + priv->pmib->dot11WdsInfo.wdsPure = 0; +#endif +#ifdef CONFIG_RTK_MESH + // in current release, mesh can be only run upon wlan0, so we disable the following flag in vap + priv->pmib->dot1180211sInfo.mesh_enable = 0; +#endif +} + + +static void rtl8192cd_init_mbssid(struct rtl8192cd_priv *priv) +{ + int i, j; + unsigned int camData[2]; + unsigned char *macAddr = GET_MY_HWADDR; + + if (IS_ROOT_INTERFACE(priv)) + { + //camData[0] = 0x00800000 | (macAddr[5] << 8) | macAddr[4]; + camData[0] = MBIDCAM_POLL | MBIDWRITE_EN | MBIDCAM_VALID | (macAddr[5] << 8) | macAddr[4]; + camData[1] = (macAddr[3] << 24) | (macAddr[2] << 16) | (macAddr[1] << 8) | macAddr[0]; +// for (j=0; j<2; j++) { + for (j=1; j>=0; j--) { + //RTL_W32((_MBIDCAMCONTENT_+4)-4*j, camData[j]); + RTL_W32((MBIDCAMCFG+4)-4*j, camData[j]); + } + //RTL_W8(_MBIDCAMCFG_, BIT(7) | BIT(6)); + + // clear the rest area of CAM + //camData[0] = 0; + camData[1] = 0; + for (i=1; i<8; i++) { + camData[0] = MBIDCAM_POLL | MBIDWRITE_EN | (i&MBIDCAM_ADDR_Mask)<<MBIDCAM_ADDR_SHIFT; +// for (j=0; j<2; j++) { + for (j=1; j>=0; j--) { + RTL_W32((MBIDCAMCFG+4)-4*j, camData[j]); + } +// RTL_W8(_MBIDCAMCFG_, BIT(7) | BIT(6) | (unsigned char)i); + } + + // set MBIDCTRL & MBID_BCN_SPACE by cmd +// set_fw_reg(priv, 0xf1000101, 0, 0); + RTL_W32(MBSSID_BCN_SPACE, + (priv->pmib->dot11StationConfigEntry.dot11BeaconPeriod & BCN_SPACE2_Mask)<<BCN_SPACE2_SHIFT + |(priv->pmib->dot11StationConfigEntry.dot11BeaconPeriod & BCN_SPACE1_Mask)<<BCN_SPACE1_SHIFT); + + RTL_W8(BCN_CTRL, 0); + RTL_W8(0x553, 1); + + if (IS_TEST_CHIP(priv)) + RTL_W8(BCN_CTRL, EN_BCN_FUNCTION |EN_MBSSID| DIS_SUB_STATE | DIS_TSF_UPDATE|EN_TXBCN_RPT); + else + RTL_W8(BCN_CTRL, EN_BCN_FUNCTION | DIS_SUB_STATE_N | DIS_TSF_UPDATE_N|EN_TXBCN_RPT); + + RTL_W32(RCR, RTL_R32(RCR) | RCR_MBID_EN); // MBSSID enable +/* +#ifdef CLIENT_MODE + if ((OPMODE & WIFI_STATION_STATE) || (OPMODE & WIFI_ADHOC_STATE)) + RTL_W32(RCR, RTL_R32(RCR) | RCR_CBSSID); +#endif +*/ + } + else if (IS_VAP_INTERFACE(priv)) + { +// priv->vap_init_seq = (RTL_R8(_MBIDCTRL_) & (BIT(4) | BIT(5) | BIT(6))) >> 4; +// priv->vap_init_seq++; +// set_fw_reg(priv, 0xf1000001 | ((priv->vap_init_seq + 1)&0xffff)<<8, 0, 0); + + priv->vap_init_seq = RTL_R8(MBID_NUM) & MBID_BCN_NUM_Mask; + priv->vap_init_seq++; + camData[0] = MBIDCAM_POLL | MBIDWRITE_EN | MBIDCAM_VALID | + (priv->vap_init_seq&MBIDCAM_ADDR_Mask)<<MBIDCAM_ADDR_SHIFT | + (macAddr[5] << 8) | macAddr[4]; + camData[1] = (macAddr[3] << 24) | (macAddr[2] << 16) | (macAddr[1] << 8) | macAddr[0]; + for (j=1; j>=0; j--) { + RTL_W32((MBIDCAMCFG+4)-4*j, camData[j]); + } +// RTL_W8(_MBIDCAMCFG_, BIT(7) | BIT(6) | ((unsigned char)priv->vap_init_seq & 0x1f)); + RTL_W32(MBSSID_BCN_SPACE, + ((priv->pmib->dot11StationConfigEntry.dot11BeaconPeriod- + ((priv->pmib->dot11StationConfigEntry.dot11BeaconPeriod/(priv->vap_init_seq+1))*priv->vap_init_seq)) + & BCN_SPACE2_Mask)<<BCN_SPACE2_SHIFT + |((priv->pmib->dot11StationConfigEntry.dot11BeaconPeriod/(priv->vap_init_seq+1)) & BCN_SPACE1_Mask) + <<BCN_SPACE1_SHIFT); + RTL_W8(BCN_CTRL, 0); + RTL_W8(0x553, 1); + + if (IS_TEST_CHIP(priv)) + RTL_W8(BCN_CTRL, EN_BCN_FUNCTION |EN_MBSSID| DIS_SUB_STATE | DIS_TSF_UPDATE|EN_TXBCN_RPT); + else + RTL_W8(BCN_CTRL, EN_BCN_FUNCTION | DIS_SUB_STATE_N | DIS_TSF_UPDATE_N|EN_TXBCN_RPT); + + + RTL_W8(MBID_NUM, priv->vap_init_seq & MBID_BCN_NUM_Mask); + RTL_W32(RCR, RTL_R32(RCR) & ~RCR_MBID_EN); + RTL_W32(RCR, RTL_R32(RCR) | RCR_MBID_EN); // MBSSID enable + } +} + + +static void rtl8192cd_stop_mbssid(struct rtl8192cd_priv *priv) +{ + int i, j; + unsigned int camData[2]; + camData[1] = 0; + + if (IS_ROOT_INTERFACE(priv)) + { + // clear the rest area of CAM + for (i=0; i<8; i++) { + camData[0] = MBIDCAM_POLL | MBIDWRITE_EN | (i&MBIDCAM_ADDR_Mask)<<MBIDCAM_ADDR_SHIFT; + for (j=1; j>=0; j--) { + RTL_W32((MBIDCAMCFG+4)-4*j, camData[j]); + } +// RTL_W8(_MBIDCAMCFG_, BIT(7) | BIT(6) | (unsigned char)i); + } + +// set_fw_reg(priv, 0xf1000001, 0, 0); + RTL_W32(RCR, RTL_R32(RCR) & ~RCR_MBID_EN); // MBSSID disable + RTL_W32(MBSSID_BCN_SPACE, + (priv->pmib->dot11StationConfigEntry.dot11BeaconPeriod & BCN_SPACE1_Mask)<<BCN_SPACE1_SHIFT); + + RTL_W8(BCN_CTRL, 0); + RTL_W8(0x553, 1); + if (IS_TEST_CHIP(priv)) + RTL_W8(BCN_CTRL, EN_BCN_FUNCTION | DIS_SUB_STATE | DIS_TSF_UPDATE| EN_TXBCN_RPT); + else + RTL_W8(BCN_CTRL, EN_BCN_FUNCTION | DIS_SUB_STATE_N | DIS_TSF_UPDATE_N| EN_TXBCN_RPT); + + } + else if (IS_VAP_INTERFACE(priv) && (priv->vap_init_seq >= 0)) + { +// set_fw_reg(priv, 0xf1000001 | (((RTL_R8(_MBIDCTRL_) & (BIT(4) | BIT(5) | BIT(6))) >> 4)&0xffff)<<8, 0, 0); + camData[0] = MBIDCAM_POLL | MBIDWRITE_EN | + (priv->vap_init_seq&MBIDCAM_ADDR_Mask)<<MBIDCAM_ADDR_SHIFT; + for (j=1; j>=0; j--) { + RTL_W32((MBIDCAMCFG+4)-4*j, camData[j]); + } +// RTL_W8(_MBIDCAMCFG_, BIT(7) | BIT(6) | ((unsigned char)priv->vap_init_seq & 0x1f)); + + if (RTL_R8(MBID_NUM) & MBID_BCN_NUM_Mask) { + RTL_W8(MBID_NUM, (RTL_R8(MBID_NUM)-1) & MBID_BCN_NUM_Mask); + RTL_W32(MBSSID_BCN_SPACE, + ((priv->pmib->dot11StationConfigEntry.dot11BeaconPeriod- + ((priv->pmib->dot11StationConfigEntry.dot11BeaconPeriod/(RTL_R8(MBID_NUM)+1))*RTL_R8(MBID_NUM))) + & BCN_SPACE2_Mask)<<BCN_SPACE2_SHIFT + |((priv->pmib->dot11StationConfigEntry.dot11BeaconPeriod/(RTL_R8(MBID_NUM)+1)) & BCN_SPACE1_Mask) + <<BCN_SPACE1_SHIFT); + RTL_W8(BCN_CTRL, 0); + RTL_W8(0x553, 1); + if (IS_TEST_CHIP(priv)) + RTL_W8(BCN_CTRL, EN_BCN_FUNCTION | DIS_SUB_STATE | DIS_TSF_UPDATE| EN_TXBCN_RPT); + else + RTL_W8(BCN_CTRL, EN_BCN_FUNCTION | DIS_SUB_STATE_N | DIS_TSF_UPDATE_N| EN_TXBCN_RPT); + + } + RTL_W32(RCR, RTL_R32(RCR) & ~RCR_MBID_EN); + RTL_W32(RCR, RTL_R32(RCR) | RCR_MBID_EN); + priv->vap_init_seq = -1; + } +} +#endif + + +#ifdef WDS +#ifdef LAZY_WDS +void delete_wds_entry(struct rtl8192cd_priv *priv, struct stat_info *pstat) +{ + int i; + + for (i=0; i<NUM_WDS; i++) { + if (!memcmp(priv->pmib->dot11WdsInfo.entry[i].macAddr, pstat->hwaddr, MACADDRLEN)) { + memcpy(priv->pmib->dot11WdsInfo.entry[i].macAddr, + NULL_MAC_ADDR, MACADDRLEN); + free_stainfo(priv, pstat); + priv->pmib->dot11WdsInfo.wdsNum--; + break; + } + } +} +#endif + +struct stat_info *add_wds_entry(struct rtl8192cd_priv *priv, int idx, unsigned char *mac) +{ + struct stat_info *pstat; + DOT11_SET_KEY Set_Key; +#ifdef LAZY_WDS + int i; + + if (mac != NULL) { + for (i=0; i<NUM_WDS; i++) { + if (!memcmp(priv->pmib->dot11WdsInfo.entry[i].macAddr, + NULL_MAC_ADDR, MACADDRLEN)) { + memcpy( priv->pmib->dot11WdsInfo.entry[i].macAddr, mac, MACADDRLEN); + idx = i; + priv->pmib->dot11WdsInfo.wdsNum++; + break; + } + } + if (i == NUM_WDS) { + DEBUG_ERR("WDS table is full!!!\n"); + return NULL; + } + } +#endif + + pstat = alloc_stainfo(priv, priv->pmib->dot11WdsInfo.entry[idx].macAddr, -1); + if (pstat == NULL) { + DEBUG_ERR("alloc_stainfo() fail!\n"); + return NULL; + } + + // use self supported rate for wds + memcpy(pstat->bssrateset, AP_BSSRATE, AP_BSSRATE_LEN); + pstat->bssratelen = AP_BSSRATE_LEN; + + if (priv->pmib->dot11WdsInfo.entry[idx].txRate) { + if (!(priv->pmib->dot11BssType.net_work_type & WIRELESS_11N)) + priv->pmib->dot11WdsInfo.entry[idx].txRate &= 0x0000fff; // mask HT rates + + if (!(priv->pmib->dot11BssType.net_work_type & WIRELESS_11A) && + !(priv->pmib->dot11BssType.net_work_type & WIRELESS_11G)) + priv->pmib->dot11WdsInfo.entry[idx].txRate &= 0xffff00f; // mask OFDM rates + + if (!(priv->pmib->dot11BssType.net_work_type & WIRELESS_11B)) + priv->pmib->dot11WdsInfo.entry[idx].txRate &= 0xffffff0; // mask CCK rates + } + + pstat->state = WIFI_WDS; + +#ifdef LAZY_WDS + if (priv->pmib->dot11WdsInfo.wdsEnabled == WDS_LAZY_ENABLE) + pstat->state |= WIFI_WDS_LAZY; + + if ((priv->pmib->dot11WdsInfo.wdsEnabled == WDS_LAZY_ENABLE) && + (priv->pmib->dot11WdsInfo.wdsPrivacy == _TKIP_PRIVACY_ || + priv->pmib->dot11WdsInfo.wdsPrivacy == _CCMP_PRIVACY_)) + wds_psk_set(priv, idx, NULL); +#endif + + if (priv->pmib->dot11WdsInfo.wdsPrivacy == _WEP_40_PRIVACY_ || + priv->pmib->dot11WdsInfo.wdsPrivacy == _WEP_104_PRIVACY_ ) { +#ifndef CONFIG_RTL8186_KB + memcpy(Set_Key.MACAddr, priv->pmib->dot11WdsInfo.entry[idx].macAddr, 6); + Set_Key.KeyType = DOT11_KeyType_Pairwise; + Set_Key.EncType = priv->pmib->dot11WdsInfo.wdsPrivacy; + Set_Key.KeyIndex = priv->pmib->dot11WdsInfo.wdsKeyId; + DOT11_Process_Set_Key(priv->dev, NULL, &Set_Key, priv->pmib->dot11WdsInfo.wdsWepKey); +#endif + } + else if ((priv->pmib->dot11WdsInfo.wdsPrivacy == _TKIP_PRIVACY_ || + priv->pmib->dot11WdsInfo.wdsPrivacy == _CCMP_PRIVACY_) && + (priv->pmib->dot11WdsInfo.wdsMappingKeyLen[idx]&0x80000000) ) { + priv->pmib->dot11WdsInfo.wdsMappingKeyLen[idx] &= ~0x80000000; + memcpy(Set_Key.MACAddr, priv->pmib->dot11WdsInfo.entry[idx].macAddr, 6); + Set_Key.KeyType = DOT11_KeyType_Pairwise; + Set_Key.EncType = priv->pmib->dot11WdsInfo.wdsPrivacy; + Set_Key.KeyIndex = priv->pmib->dot11WdsInfo.wdsKeyId; + DOT11_Process_Set_Key(priv->dev, NULL, &Set_Key, priv->pmib->dot11WdsInfo.wdsMapingKey[idx]); + } + + add_update_RATid(priv, pstat); + pstat->wds_idx = idx; + assign_tx_rate(priv, pstat, NULL); + assign_aggre_mthod(priv, pstat); + assign_aggre_size(priv, pstat); + + list_add_tail(&pstat->asoc_list, &priv->asoc_list); + + return pstat; +} + +static void create_wds_tbl(struct rtl8192cd_priv *priv) +{ + int i; + struct stat_info *pstat; + +#ifdef FAST_RECOVERY + if (priv->reset_hangup) + return; +#endif + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (!IS_ROOT_INTERFACE(priv)) + return; +#endif + + if (priv->pmib->dot11WdsInfo.wdsEnabled) { + for (i=0; i<priv->pmib->dot11WdsInfo.wdsNum; i++) { + pstat = add_wds_entry(priv, i, NULL); + if (pstat == NULL) + break; + } + } +} +#endif // WDS + + +void validate_fixed_tx_rate(struct rtl8192cd_priv *priv) +{ + if (!priv->pmib->dot11StationConfigEntry.autoRate) { + if (!(priv->pmib->dot11BssType.net_work_type & WIRELESS_11N)) + priv->pmib->dot11StationConfigEntry.fixedTxRate &= 0x0000fff; // mask HT rates + + if((priv->pmib->dot11BssType.net_work_type & WIRELESS_11N) && + ((get_rf_mimo_mode(priv) == MIMO_1T2R) || (get_rf_mimo_mode(priv) == MIMO_1T1R))) + priv->pmib->dot11StationConfigEntry.fixedTxRate &= 0x00fffff; // mask MCS8 - MCS15 + + if (!(priv->pmib->dot11BssType.net_work_type & WIRELESS_11A) && + !(priv->pmib->dot11BssType.net_work_type & WIRELESS_11G)) + priv->pmib->dot11StationConfigEntry.fixedTxRate &= 0xffff00f; // mask OFDM rates + + if (!(priv->pmib->dot11BssType.net_work_type & WIRELESS_11B)) + priv->pmib->dot11StationConfigEntry.fixedTxRate &= 0xffffff0; // mask CCK rates + + if (priv->pmib->dot11StationConfigEntry.fixedTxRate==0) { + priv->pmib->dot11StationConfigEntry.autoRate=1; + PRINT_INFO("invalid fixed tx rate, use auto rate!\n"); + } + else + priv->pshare->current_tx_rate = get_rate_from_bit_value(priv->pmib->dot11StationConfigEntry.fixedTxRate); + } +} + + +#if defined(RESERVE_TXDESC_FOR_EACH_IF) && (defined(UNIVERSAL_REPEATER) || defined(MBSSID)) +void recalc_txdesc_limit(struct rtl8192cd_priv *priv) +{ + struct rtl8192cd_priv *root_priv = NULL; + int i, num, total_if = 0; + + if (IS_ROOT_INTERFACE(priv)) + root_priv = priv; + else + root_priv = GET_ROOT_PRIV(priv); + + if (IS_DRV_OPEN(root_priv)) + total_if++; + +#ifdef UNIVERSAL_REPEATER + if (IS_DRV_OPEN(root_priv->pvxd_priv)) + total_if++; +#endif + +#ifdef MBSSID + for (i=0; i<RTL8192CD_NUM_VWLAN; i++) { + if (IS_DRV_OPEN(root_priv->pvap_priv[i])) + total_if++; + } +#endif + + if (total_if <= 1) { + root_priv->pshare->num_txdesc_cnt = NUM_TX_DESC - 2; // 2 for space... + root_priv->pshare->num_txdesc_upper_limit = NUM_TX_DESC - 2; + root_priv->pshare->num_txdesc_lower_limit = 0; + return; + } + + num = (NUM_TX_DESC * IF_TXDESC_UPPER_LIMIT) / 100; + root_priv->pshare->num_txdesc_upper_limit = num; + + num = ((NUM_TX_DESC - 2) - num) / (total_if - 1); + root_priv->pshare->num_txdesc_lower_limit = num; + + num = root_priv->pshare->num_txdesc_upper_limit + + root_priv->pshare->num_txdesc_lower_limit * (total_if - 1); + root_priv->pshare->num_txdesc_cnt = num; +} +#endif + + +int rtl8192cd_open(struct net_device *dev) +{ + struct rtl8192cd_priv *priv; // recuresively used, can't be static + int rc; +#ifdef MBSSID + int i; +#endif + unsigned long x; +#if defined(CONFIG_RTL865X_WTDOG) || (defined(CONFIG_RTL_WTDOG) && defined(CONFIG_RTL_92D_SUPPORT)) +#if !(defined(CONFIG_RTL8196B) || defined(CONFIG_RTL_819X)) + unsigned long wtval; +#endif +#endif +#ifdef CHECK_HANGUP + int is_reset; +#endif + int init_hw_cnt = 0; + + DBFENTER; + +#ifdef NETDEV_NO_PRIV + priv = ((struct rtl8192cd_priv *)netdev_priv(dev))->wlan_priv; +#else + priv = dev->priv; +#endif + + #if defined(CONFIG_RTL_CUSTOM_PASSTHRU) + /*to open custom pass through pseudo device*/ + if(dev==wlan_device[passThruWanIdx].priv->pWlanDev) + { + /*root device is opened*/ + if(priv->drv_state&DRV_STATE_OPEN) + { + netif_start_queue(dev); + return 0; + } + + } + #endif + +#if 0 +//#ifdef PCIE_POWER_SAVING + if (((REG32(CLK_MANAGE) & BIT(11)) == 0) +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + && IS_ROOT_INTERFACE(priv) +#endif + ) { + extern void setBaseAddressRegister(void); + REG32(CLK_MANAGE) |= BIT(11); + delay_ms(10); + PCIE_reset_procedure(0, 0, 1, priv->pshare->ioaddr); + setBaseAddressRegister(); + } +#endif + +#ifdef CHECK_HANGUP + is_reset = priv->reset_hangup; +#endif + +// init mib from cfg file, we only need to load cfg file once - chris 2010/02 +#ifdef CONFIG_RTL_COMAPI_CFGFILE +#ifdef WDS + if (dev->base_addr) //root +#endif + { + //printk(">>>>>>>>>>>>>>>>>>>>>>>>>>>>>dev %s set_default\n", dev->name); + //memset(priv->pmib, 0, sizeof(struct wifi_mib)); + //set_mib_default(priv); + + CfgFileProc(dev); + } +#endif + +// register iw_handler - chris 2010/02 +#ifdef CONFIG_RTL_COMAPI_WLTOOLS + dev->wireless_handlers = (struct iw_handler_def *) &rtl8192cd_iw_handler_def; +#endif + + memcpy((void *)dev->dev_addr, priv->pmib->dot11OperationEntry.hwaddr, 6); + +#ifdef WDS + if (dev->base_addr == 0) + { +#ifdef BR_SHORTCUT + extern struct net_device *cached_wds_dev; + cached_wds_dev = NULL; +#endif + + netif_start_queue(dev); + return 0; + } +#endif + +#ifdef CONFIG_RTK_MESH + if (dev->base_addr == 1) { + netif_start_queue(dev); + return 0; + } +#endif // CONFIG_RTK_MESH + +#ifdef PCIE_POWER_SAVING + if((priv->pwr_state == L1) || (priv->pwr_state == L2)) { + PCIeWakeUp(priv, (POWER_DOWN_T0<<3)); + } +#endif + + // stop h/w in the very beginning +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { + if ( CheckNoResetHwExceptionCase(priv) ) { + rtl8192cd_stop_hw(priv); + } + } + +#ifdef UNIVERSAL_REPEATER + // If vxd interface, see if some mandatory mib is set. If ok, backup these + // mib, and copy all mib from root interface. Then, restore the backup mib + // to current. + + if (IS_VXD_INTERFACE(priv)) { + DEBUG_INFO("Open request from vxd\n"); + if (!IS_DRV_OPEN(GET_ROOT_PRIV(priv))) { + printk("Open vxd error! Root interface should be opened in advanced.\n"); + return 0; + } + + if (!(priv->drv_state & DRV_STATE_VXD_INIT)) { +// Mark following code. MIB copy will be executed through ioctl ------------- +#if 0 + unsigned char tmpbuf[36]; + int len, encyption, is_1x, mac_clone, nat25; + struct Dot11RsnIE rsnie; + + len = SSID_LEN; + memcpy(tmpbuf, SSID, len); + encyption = priv->pmib->dot1180211AuthEntry.dot11PrivacyAlgrthm; + is_1x = IEEE8021X_FUN; + mac_clone = priv->pmib->ethBrExtInfo.macclone_enable; + nat25 = priv->pmib->ethBrExtInfo.nat25_disable; + memcpy((char *)&rsnie, (char *)&priv->pmib->dot11RsnIE, sizeof(rsnie)); + + memcpy(priv->pmib, GET_ROOT_PRIV(priv)->pmib, sizeof(struct wifi_mib)); + + SSID_LEN = len; + memcpy(SSID, tmpbuf, len); + SSID2SCAN_LEN = len; + memcpy(SSID2SCAN, SSID, len); + priv->pmib->dot1180211AuthEntry.dot11PrivacyAlgrthm = encyption; + IEEE8021X_FUN = is_1x; + priv->pmib->ethBrExtInfo.macclone_enable = mac_clone; + priv->pmib->ethBrExtInfo.nat25_disable = nat25; + memcpy((char *)&priv->pmib->dot11RsnIE, (char *)&rsnie, sizeof(rsnie)); +#ifdef WDS + // always disable wds in vxd + priv->pmib->dot11WdsInfo.wdsEnabled = 0; + priv->pmib->dot11WdsInfo.wdsPure = 0; +#endif + + // if root interface is AP mode, set infra-client in vxd + // if root interfeace is a infra-client, set AP in vxd + if (OPMODE & WIFI_AP_STATE) { + OPMODE = WIFI_STATION_STATE; +#if defined(WIFI_WMM) && defined(WMM_APSD) + APSD_ENABLE = 0; +#endif + DEBUG_INFO("Set vxd as an infra-client\n"); + } + else if (OPMODE & WIFI_STATION_STATE) { + OPMODE = WIFI_AP_STATE; + priv->auto_channel = 0; + DEBUG_INFO("Set vxd as an AP\n"); + } + else { + DEBUG_ERR("Invalid opmode for vxd!\n"); + return 0; + } +#endif +//---------------------------------------------------------- david+2008-03-17 + + // correct RSN IE will be set later for WPA/WPA2 +#ifdef CHECK_HANGUP + if (!is_reset) +#endif + memset(&priv->pmib->dot11RsnIE, 0, sizeof(struct Dot11RsnIE)); + +#ifdef WDS + // always disable wds in vxd + priv->pmib->dot11WdsInfo.wdsEnabled = 0; + priv->pmib->dot11WdsInfo.wdsPure = 0; +#endif + +#ifdef CONFIG_RTK_MESH + // always disable mesh in vxd (for dev) + GET_MIB(priv)->dot1180211sInfo.mesh_enable = 0; +#endif // CONFIG_RTK_MESH + priv->drv_state |= DRV_STATE_VXD_INIT; // indicate the mib of vxd driver has been initialized + } + } +#endif // UNIVERSAL_REPEATER + +#ifdef CHECK_HANGUP + if (!is_reset) +#endif + { + if (OPMODE & WIFI_AP_STATE) + OPMODE = WIFI_AP_STATE; +#ifdef CLIENT_MODE + else if (OPMODE & WIFI_STATION_STATE) { + OPMODE = WIFI_STATION_STATE; +#if defined(WIFI_WMM) && defined(WMM_APSD) + APSD_ENABLE = 0; +#endif + } + else if (OPMODE & WIFI_ADHOC_STATE) { + OPMODE = WIFI_ADHOC_STATE; +#ifdef WIFI_WMM +#ifdef WMM_APSD + APSD_ENABLE = 0; +#endif +#endif + } +#endif + else { + printk("Undefined state... using AP mode as default\n"); + OPMODE = WIFI_AP_STATE; + } + } + +#ifdef MBSSID + if (GET_ROOT(priv)->pmib->miscEntry.vap_enable) + { + if (IS_VAP_INTERFACE(priv)) { + if (!IS_DRV_OPEN(GET_ROOT_PRIV(priv))) { + printk("Open vap error! Root interface should be opened in advanced.\n"); + return -1; + } + + if ((GET_ROOT_PRIV(priv)->pmib->dot11OperationEntry.opmode & WIFI_AP_STATE) == 0) { + printk("Fail to open VAP under non-AP mode!\n"); + return -1; + } + else { + rtl8192cd_init_vap_mib(priv); + } + } + } +#endif + +#if defined(BR_SHORTCUT) +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if(IS_ROOT_INTERFACE(priv)) +#endif + clear_shortcut_cache(); +#endif + + rc = rtl8192cd_init_sw(priv); + if (rc) + return rc; + +#if defined(CONFIG_RTL8196B_TR) || defined(CONFIG_RTL8196C_EC) + if (!priv->auto_channel) { + LOG_START_MSG(); + } +#endif +//#ifdef CONFIG_RTL865X_AC +#if defined(CONFIG_RTL865X_AC) || defined(CONFIG_RTL865X_KLD) || defined(CONFIG_RTL8196B_KLD) || defined(CONFIG_RTL8196C_KLD) + if (!priv->auto_channel) { + LOG_START_MSG(); + } +#endif + + validate_fixed_tx_rate(priv); + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { +#ifdef __KERNEL__ +#ifdef CHECK_HANGUP + if (!is_reset) +#endif + { +#if !defined(CONFIG_RTL8671) + #if defined(__LINUX_2_6__) + #if defined(CONFIG_RTL_92D_DMDP) + rc = request_irq(dev->irq, rtl8192cd_interrupt, IRQF_SHARED|IRQF_DISABLED, dev->name, dev); + #else + rc = request_irq(dev->irq, rtl8192cd_interrupt, IRQF_DISABLED, dev->name, dev); + #endif + #else + rc = request_irq(dev->irq, rtl8192cd_interrupt, SA_SHIRQ, dev->name, dev); + #endif +#else + rc = request_irq(dev->irq, rtl8192cd_interrupt, SA_INTERRUPT, dev->name, dev); +#endif + +#if defined(PCIE_POWER_SAVING) && defined(GPIO_WAKEPIN) + rc |= request_irq_for_wakeup_pin(dev); +#endif + + if (rc) { + DEBUG_ERR("some issue in request_irq, rc=%d\n", rc); + } + } +#endif + + SAVE_INT_AND_CLI(x); + SMP_LOCK(x); + +#ifdef CONFIG_RTL865X_WTDOG +#if !(defined(CONFIG_RTL8196B) || defined(CONFIG_RTL_819X)) + wtval = *((volatile unsigned long *)0xB800311C); + *((volatile unsigned long *)0xB800311C) = 0xA5f00000; +#endif +#endif + +do_hw_init: + +#ifdef EN_EFUSE + { + int i, readEfuse=0; + for(i=0;i<MAX_2G_CHANNEL_NUM;i++) { + if( priv->pmib->dot11RFEntry.pwrlevelCCK_A[i]==0 + || priv->pmib->dot11RFEntry.pwrlevelHT40_1S_A[i] ==0) { + readEfuse = 1; + break; + } + } + if(readEfuse) + ReadTxPowerInfoFromHWPG(priv); + + if(priv->pmib->dot11RFEntry.ther==0) { + ReadThermalMeterFromEfuse(priv); + +#ifdef CONFIG_RTL_92D_SUPPORT + if (GET_CHIP_VER(priv) == VERSION_8192D) { + ReadCrystalCalibrationFromEfuse(priv); + ReadDeltaValFromEfuse(priv); + ReadTRSWPAPEFromEfuse(priv); + } +#endif + } + } +#endif + rc = rtl8192cd_init_hw_PCI(priv); + //delay_ms(200); // TODO: need refinement, for 98 watchdog time out + + // write IDR0, IDR4 here + { + unsigned long reg = 0; + reg = *(unsigned long *)(dev->dev_addr); +// RTL_W32(IDR0, (cpu_to_le32(reg))); + RTL_W32(MACID, (cpu_to_le32(reg))); + reg = *(unsigned short *)((unsigned long)dev->dev_addr + 4); +// RTL_W32(IDR4, (cpu_to_le32(reg))); + RTL_W16(MACID+4, (cpu_to_le16(reg))); +#ifdef CONFIG_RTL_92D_SUPPORT + RTL_W32(MACID1 , RTL_R32(MACID)); + RTL_W16(MACID1+4, RTL_R16(MACID+4)); +#endif + } + + if (rc && ++init_hw_cnt < 5) { +#if defined(CONFIG_RTL865X_WTDOG) || (defined(CONFIG_RTL_WTDOG) && defined(CONFIG_RTL_92D_SUPPORT)) + if (GET_CHIP_VER(priv)==VERSION_8192D) { + *((volatile unsigned long *)0xB800311C) |= 1 << 23; + } +#endif + goto do_hw_init; + } + +#if defined(CONFIG_RTL865X_WTDOG) || (defined(CONFIG_RTL_WTDOG) && defined(CONFIG_RTL_92D_SUPPORT)) +if (GET_CHIP_VER(priv)==VERSION_8192D) { +#if !(defined(CONFIG_RTL8196B) || defined(CONFIG_RTL_819X)) + *((volatile unsigned long *)0xB800311C) |= 1 << 23; + *((volatile unsigned long *)0xB800311C) = wtval; +#endif +} +#endif + + + + RESTORE_INT(x); + SMP_UNLOCK(x); + + if (rc) { + DEBUG_ERR("init hw failed!\n"); +#ifdef CONFIG_RTL_92D_SUPPORT + if (GET_CHIP_VER(priv)==VERSION_8192D){ + RTL_W8(RSV_MAC0_CTRL, RTL_R8(RSV_MAC0_CTRL)&(~MAC0_EN)); + RTL_W8(RSV_MAC1_CTRL, RTL_R8(RSV_MAC1_CTRL)&(~MAC1_EN)); + } +#endif //CONFIG_RTL_92D_SUPPORT + force_stop_wlan_hw(); +#ifdef CONFIG_RTL_92D_SUPPORT + if (GET_CHIP_VER(priv)!=VERSION_8192D) /* do not trigger wtdog if 92D open fail */ +#endif + { +#ifdef LINUX_2_6_21_ + local_irq_disable(); +#else + cli(); +#endif +#ifndef NOT_RTK_BSP + *(volatile unsigned long *)(0xB800311c) = 0; /* enable watchdog reset now */ +#endif + for(;;) + ; + } + return rc; + } + } +#if 0 // defined(UNIVERSAL_REPEATER) || defined(MBSSID) + else { + if (get_rf_mimo_mode(priv) == MIMO_1T1R) + GET_MIB(priv)->dot11nConfigEntry.dot11nSupportedMCS &= 0x00ff; + } +#endif + +#ifdef MBSSID + if ((OPMODE & WIFI_AP_STATE) && (GET_ROOT(priv)->pmib->miscEntry.vap_enable)) { + rtl8192cd_init_mbssid(priv); + + if (IS_VAP_INTERFACE(priv)) { + // set BcnDmaInt & BcnOk of different VAP in IMR + priv->pshare->InterruptMask |= (HIMR_BCNDOK1 << (priv->vap_init_seq-1)) | + (HIMR_BCNDMA1 << (priv->vap_init_seq-1)); + RTL_W32(HIMR, priv->pshare->InterruptMask); + + if (GET_ROOT_PRIV(priv)->auto_channel == 0) { + priv->pmib->dot11RFEntry.dot11channel = GET_ROOT_PRIV(priv)->pmib->dot11RFEntry.dot11channel; + priv->ht_cap_len = 0; // re-construct HT IE + + init_beacon(priv); + } + } + } +#endif + +// new added to reset keep_rsnie flag + if (priv->pmib->dot11OperationEntry.keep_rsnie) + priv->pmib->dot11OperationEntry.keep_rsnie = 0; +//------------------- david+2006-06-30 + + priv->drv_state |= DRV_STATE_OPEN; // set driver as has been opened, david + +#if defined(RESERVE_TXDESC_FOR_EACH_IF) && (defined(UNIVERSAL_REPEATER) || defined(MBSSID)) + recalc_txdesc_limit(priv); +#endif + + //memcpy((void *)dev->dev_addr, priv->pmib->dot11OperationEntry.hwaddr, 6); + + // below is for site_survey timer + init_timer(&priv->ss_timer); + priv->ss_timer.data = (unsigned long) priv; + priv->ss_timer.function = rtl8192cd_ss_timer; + +#ifdef CONFIG_RTL_COMAPI_WLTOOLS + init_waitqueue_head(&priv->ss_wait); +#endif + +#ifdef CLIENT_MODE + init_timer(&priv->reauth_timer); + priv->reauth_timer.data = (unsigned long) priv; + priv->reauth_timer.function = rtl8192cd_reauth_timer; + + init_timer(&priv->reassoc_timer); + priv->reassoc_timer.data = (unsigned long) priv; + priv->reassoc_timer.function = rtl8192cd_reassoc_timer; + + init_timer(&priv->idle_timer); + priv->idle_timer.data = (unsigned long) priv; + priv->idle_timer.function = rtl8192cd_idle_timer; +#ifdef DFS + init_timer(&priv->dfs_cntdwn_timer); + priv->dfs_cntdwn_timer.data = (unsigned long) priv; + priv->dfs_cntdwn_timer.function = rtl8192cd_dfs_cntdwn_timer; +#endif +#endif + + priv->frag_to = 0; + init_timer(&priv->frag_to_filter); + priv->frag_to_filter.expires = jiffies + FRAG_TO; + priv->frag_to_filter.data = (unsigned long) priv; + priv->frag_to_filter.function = rtl8192cd_frag_timer; + mod_timer(&priv->frag_to_filter, jiffies + FRAG_TO); + +#ifdef DETECT_STA_EXISTANCE + // Added by Annie for Retry Limit Recovery Timer, 2010-08-10. + init_timer(&priv->pshare->rl_recover_timer); + priv->pshare->rl_recover_timer.data = (unsigned long) priv; + priv->pshare->rl_recover_timer.function = RetryLimitRecovery; + priv->pshare->bRLShortened = FALSE; +#endif + + priv->auth_to = AUTH_TO / HZ; + priv->assoc_to = ASSOC_TO / HZ; + +#ifdef PCIE_POWER_SAVING_DEBUG + priv->expire_to = 60; +#else + priv->expire_to = (EXPIRETIME > 100)? (EXPIRETIME / 100) : 86400; /*10ms to 1s*/ +#endif + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { +#ifdef PCIE_POWER_SAVING + init_timer(&priv->ps_timer); + priv->ps_timer.expires = jiffies + POWER_DOWN_T0; + priv->ps_timer.data = (unsigned long) priv; + priv->ps_timer.function = PCIe_power_save_timer; + mod_timer(&priv->ps_timer, jiffies + POWER_DOWN_T0); +// priv->ps_ctrl = 0x11; +#endif + init_timer(&priv->expire_timer); + priv->expire_timer.expires = jiffies + EXPIRE_TO; + priv->expire_timer.data = (unsigned long) priv; + priv->expire_timer.function = rtl8192cd_1sec_timer; +#ifdef SW_ANT_SWITCH + init_timer(&priv->pshare->swAntennaSwitchTimer); + priv->pshare->swAntennaSwitchTimer.data = (unsigned long) priv; + priv->pshare->swAntennaSwitchTimer.function = dm_SW_AntennaSwitchCallback; +#endif + mod_timer(&priv->expire_timer, jiffies + EXPIRE_TO); + + init_timer(&priv->pshare->rc_sys_timer); + priv->pshare->rc_sys_timer.data = (unsigned long) priv; + priv->pshare->rc_sys_timer.function = reorder_ctrl_timeout; + +#if 0 + init_timer(&priv->pshare->phw->tpt_timer); + priv->pshare->phw->tpt_timer.data = (unsigned long)priv; + priv->pshare->phw->tpt_timer.function = rtl8192cd_tpt_timer; +#endif + } + + // for MIC check + init_timer(&priv->MIC_check_timer); + priv->MIC_check_timer.data = (unsigned long) priv; + priv->MIC_check_timer.function = DOT11_Process_MIC_Timerup; + init_timer(&priv->assoc_reject_timer); + priv->assoc_reject_timer.data = (unsigned long) priv; + priv->assoc_reject_timer.function = DOT11_Process_Reject_Assoc_Timerup; + priv->MIC_timer_on = FALSE; + priv->assoc_reject_on = FALSE; + +#ifdef GBWC + init_timer(&priv->GBWC_timer); + priv->GBWC_timer.data = (unsigned long) priv; + priv->GBWC_timer.function = rtl8192cd_GBWC_timer; + mod_timer(&priv->GBWC_timer, jiffies + GBWC_TO); +#endif + + // to avoid add RAtid fail + init_timer(&priv->add_RATid_timer); + priv->add_RATid_timer.data = (unsigned long) priv; + priv->add_RATid_timer.function = add_RATid_timer; + + init_timer(&priv->add_rssi_timer); + priv->add_rssi_timer.data = (unsigned long) priv; + priv->add_rssi_timer.function = add_rssi_timer; + + init_timer(&priv->add_ps_timer); + priv->add_ps_timer.data = (unsigned long) priv; + priv->add_ps_timer.function = add_ps_timer; + +#if defined(CONFIG_RTL_92D_SUPPORT) && defined(CONFIG_RTL_NOISE_CONTROL) + if (GET_CHIP_VER(priv) == VERSION_8192D){ + init_timer(&priv->dnc_timer); + priv->dnc_timer.data = (unsigned long) priv; + priv->dnc_timer.function = dnc_timer; + } +#endif + +#ifdef SMART_CONCURRENT_92D + if (GET_CHIP_VER(priv) == VERSION_8192D){ + init_timer(&priv->smcc_prb_timer); + priv->smcc_prb_timer.data = (unsigned long) priv; + priv->smcc_prb_timer.function = smcc_prb_timer; + } + +#endif + +#ifdef CONFIG_RTK_MESH + /* + * CAUTION !! These statement meshX(virtual interface) ONLY, Maybe modify.... + * These statment is initial information, (If "ZERO" no need set it, because all cleared to ZERO) + */ + init_timer(&priv->mesh_peer_link_timer); + priv->mesh_peer_link_timer.data = (unsigned long) priv; + priv->mesh_peer_link_timer.function = mesh_peer_link_timer; + +#ifdef MESH_BOOTSEQ_AUTH + init_timer(&priv->mesh_auth_timer); + priv->mesh_auth_timer.data = (unsigned long) priv; + priv->mesh_auth_timer.function = mesh_auth_timer; +#endif + + priv->mesh_Version = 1; +#ifdef MESH_ESTABLISH_RSSI_THRESHOLD + priv->mesh_fake_mib.establish_rssi_threshold = DEFAULT_ESTABLISH_RSSI_THRESHOLD; +#endif + +#ifdef MESH_USE_METRICOP + // in next version, the fake_mib related values will be actually recorded in MIB + priv->mesh_fake_mib.metricID = 1; // 0: very old version, 1: version before 2009/3/10, 2: 11s + priv->mesh_fake_mib.isPure11s = 0; + priv->mesh_fake_mib.intervalMetricAuto = RTL_SECONDS_TO_JIFFIES(60); // 1 Mins + priv->mesh_fake_mib.spec11kv.defPktTO = RTL_SECONDS_TO_JIFFIES(2); // 2 * 100 = 2 secs + priv->mesh_fake_mib.spec11kv.defPktLen = 1024; // bt=8196 bits + priv->mesh_fake_mib.spec11kv.defPktCnt = 2; + priv->mesh_fake_mib.spec11kv.defPktPri = 5; + + // once driver starts up, toMeshMetricAuto will be updated + // (I think it might be put in "init one" to match our original concept...) + priv->toMeshMetricAuto = jiffies + priv->mesh_fake_mib.intervalMetricAuto; +#endif // MESH_USE_METRICOP + + mesh_set_PeerLink_CAP(priv, GET_MIB(priv)->dot1180211sInfo.mesh_max_neightbor); + + priv->mesh_PeerCAP_flags = 0x80; // Bit15(Operation as MP) shall be "1" + priv->mesh_HeaderFlags = 0; // NO Address Extension + + // The following info can be saved by FLASH in the future + priv->mesh_profile[0].used = TRUE; + priv->mesh_profile[0].PathSelectMetricID.value = 0; + priv->mesh_profile[0].PathSelectMetricID.OUI[0] = 0x00; + priv->mesh_profile[0].PathSelectMetricID.OUI[1] = 0x0f; + priv->mesh_profile[0].PathSelectMetricID.OUI[2] = 0xac; + priv->mesh_profile[0].PathSelectProtocolID.value = 0; + priv->mesh_profile[0].PathSelectProtocolID.OUI[0] = 0x00; + priv->mesh_profile[0].PathSelectProtocolID.OUI[1] = 0x0f; + priv->mesh_profile[0].PathSelectProtocolID.OUI[2] = 0xac; +#endif + + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { + // for HW/SW LED + if ((LED_TYPE >= LEDTYPE_HW_TX_RX) && (LED_TYPE <= LEDTYPE_HW_LINKACT_INFRA)) + enable_hw_LED(priv, LED_TYPE); + else if ((LED_TYPE >= LEDTYPE_SW_LINK_TXRX) && (LED_TYPE < LEDTYPE_SW_MAX)) + enable_sw_LED(priv, 1); +#ifdef SW_ANT_SWITCH + dm_SW_AntennaSwitchInit(priv); // SW Ant Switch use LED pin to control TRX Antenna +#endif +#if defined(HW_ANT_SWITCH) + dm_HW_AntennaSwitchInit(priv); +#endif + +#ifdef DFS + if (!priv->pmib->dot11DFSEntry.disable_DFS && + (OPMODE & WIFI_AP_STATE) && + (((priv->pmib->dot11RFEntry.dot11channel >= 52) && + (priv->pmib->dot11RFEntry.dot11channel <= 64)) || + ((priv->pmib->dot11RFEntry.dot11channel >= 100) && + (priv->pmib->dot11RFEntry.dot11channel <= 140)))) { + init_timer(&priv->ch_avail_chk_timer); + priv->ch_avail_chk_timer.data = (unsigned long) priv; + priv->ch_avail_chk_timer.function = rtl8192cd_ch_avail_chk_timer; + mod_timer(&priv->ch_avail_chk_timer, jiffies + CH_AVAIL_CHK_TO); + + init_timer(&priv->DFS_timer); + priv->DFS_timer.data = (unsigned long) priv; + priv->DFS_timer.function = rtl8192cd_DFS_timer; + + /* DFS activated after 5 sec; prevent switching channel due to DFS false alarm */ + mod_timer(&priv->DFS_timer, jiffies + RTL_SECONDS_TO_JIFFIES(5)); + + DFS_SetReg(priv); + } +#endif + +#ifdef SUPPORT_SNMP_MIB + mib_init(priv); +#endif + + } + +#if defined(BR_SHORTCUT) && defined(CLIENT_MODE) + if (OPMODE & WIFI_STATION_STATE) { + extern struct net_device *cached_sta_dev; + cached_sta_dev = NULL; + } +#endif + +#if defined(BR_SHORTCUT) && defined(RTL_CACHED_BR_STA) + { + extern unsigned char cached_br_sta_mac[MACADDRLEN]; + extern struct net_device *cached_br_sta_dev; + memset(cached_br_sta_mac, 0, MACADDRLEN); + cached_br_sta_dev = NULL; + } +#endif + + //if (OPMODE & WIFI_AP_STATE) //in case of station mode, queue will start only after assoc. + netif_start_queue(dev); // start queue always + +#ifdef WDS + create_wds_tbl(priv); +#endif + +#ifdef CHECK_HANGUP + if (priv->reset_hangup) + priv->reset_hangup = 0; +#endif + +#if defined(INCLUDE_WPA_PSK) && defined(CLIENT_MODE) + if (OPMODE & WIFI_ADHOC_STATE) + if (priv->pmib->dot1180211AuthEntry.dot11EnablePSK) + ToDrv_SetGTK(priv); +#endif + +#ifdef MBSSID + if (IS_ROOT_INTERFACE(priv)) +#endif + if ((OPMODE & WIFI_AP_STATE) && priv->auto_channel) { + if (((priv->pmib->dot1180211AuthEntry.dot11PrivacyAlgrthm != _TKIP_PRIVACY_) && + (priv->pmib->dot1180211AuthEntry.dot11PrivacyAlgrthm != _CCMP_PRIVACY_) && + (priv->pmib->dot1180211AuthEntry.dot11PrivacyAlgrthm != _WEP_WPA_MIXED_PRIVACY_)) || + (priv->pmib->dot11RsnIE.rsnielen > 0)) { + priv->ss_ssidlen = 0; + DEBUG_INFO("start_clnt_ss, trigger by %s, ss_ssidlen=0\n", (char *)__FUNCTION__); + start_clnt_ss(priv); + } + } + +#ifdef CLIENT_MODE + if (OPMODE & (WIFI_STATION_STATE | WIFI_ADHOC_STATE)) { +#ifdef RTK_BR_EXT + if (priv->dev->br_port) { +#ifdef __LINUX_2_6__ + memcpy(priv->br_mac, priv->dev->br_port->br->dev->dev_addr, MACADDRLEN); +#else + memcpy(priv->br_mac, priv->dev->br_port->br->dev.dev_addr, MACADDRLEN); +#endif + } +#endif + + if (!IEEE8021X_FUN || (IEEE8021X_FUN && (priv->pmib->dot11RsnIE.rsnielen > 0))) { +#ifdef CHECK_HANGUP + if (!is_reset || priv->join_res == STATE_Sta_No_Bss || + priv->join_res == STATE_Sta_Roaming_Scan || priv->join_res == 0) +#endif + { +#ifdef CHECK_HANGUP + if (is_reset) + OPMODE &= ~WIFI_SITE_MONITOR; +#endif + start_clnt_lookup(priv, 1); + } + } + } +#endif + +#ifdef UNIVERSAL_REPEATER + if (IS_ROOT_INTERFACE(priv) && + netif_running(GET_VXD_PRIV(priv)->dev)) { + SAVE_INT_AND_CLI(x); + rtl8192cd_open(GET_VXD_PRIV(priv)->dev); + RESTORE_INT(x); + } + if (IS_VXD_INTERFACE(priv) && + (GET_ROOT_PRIV(priv)->pmib->dot11OperationEntry.opmode&WIFI_STATION_STATE) && + (GET_ROOT_PRIV(priv)->pmib->dot11OperationEntry.opmode&WIFI_ASOC_STATE) && +#ifdef RTK_BR_EXT + !(GET_ROOT_PRIV(priv)->pmib->ethBrExtInfo.macclone_enable && !priv->macclone_completed) && +#endif + !(priv->drv_state & DRV_STATE_VXD_AP_STARTED) ) + enable_vxd_ap(priv); +#endif + +#ifdef MBSSID + if (IS_ROOT_INTERFACE(priv)) { + if (priv->pmib->miscEntry.vap_enable) { + for (i=0; i<RTL8192CD_NUM_VWLAN; i++) { + if (netif_running(priv->pvap_priv[i]->dev)) + rtl8192cd_open(priv->pvap_priv[i]->dev); + } + } + } +#endif + +#ifdef PCIE_POWER_SAVING +#ifdef CHECK_HANGUP + if(!is_reset) +#endif +#ifdef MBSSID + if (IS_ROOT_INTERFACE(priv)) +#endif + { +#ifdef CONFIG_RTL_8198 + if (1) +#else + if (REG32(REVR) == RTL8196C_REVISION_B) +#endif + init_pcie_power_saving(priv); + else + priv->pshare->rf_ft_var.power_save = 0; + } +#endif + + DBFEXIT; + return 0; +} + + +int rtl8192cd_set_hwaddr(struct net_device *dev, void *addr) +{ + unsigned long flags; +#ifdef NETDEV_NO_PRIV + struct rtl8192cd_priv *priv = ((struct rtl8192cd_priv *)netdev_priv(dev))->wlan_priv; +#else + struct rtl8192cd_priv *priv = (struct rtl8192cd_priv *)dev->priv; +#endif + unsigned long reg; + unsigned char *p; +#ifdef WDS + int i; +#endif + +#ifdef __KERNEL__ + p = ((struct sockaddr *)addr)->sa_data; +#else + p = (unsigned char *)addr; +#endif + + SAVE_INT_AND_CLI(flags); + SMP_LOCK(flags); + + memcpy(priv->dev->dev_addr, p, 6); + memcpy(GET_MY_HWADDR, p, 6); + +#ifdef MBSSID + if (GET_ROOT(priv)->pmib->miscEntry.vap_enable) + { + if (IS_VAP_INTERFACE(priv)) { + RESTORE_INT(flags); + SMP_UNLOCK(flags); + return 0; + } + } +#endif + +#ifdef WDS + for (i=0; i<NUM_WDS; i++) + if (priv->wds_dev[i]) + memcpy(priv->wds_dev[i]->dev_addr, p, 6); +#endif +#ifdef CONFIG_RTK_MESH + if(NUM_MESH>0) + if (priv->mesh_dev) + memcpy(priv->mesh_dev->dev_addr, p, 6); +#endif + +#ifdef UNIVERSAL_REPEATER + if (IS_ROOT_INTERFACE(priv)) { + if (GET_VXD_PRIV(priv)) { + memcpy(GET_VXD_PRIV(priv)->dev->dev_addr, p, 6); + memcpy(GET_VXD_PRIV(priv)->pmib->dot11OperationEntry.hwaddr, p, 6); + } + } + else if (IS_VXD_INTERFACE(priv)) { + memcpy(GET_ROOT_PRIV(priv)->dev->dev_addr, p, 6); + memcpy(GET_ROOT_PRIV(priv)->pmib->dot11OperationEntry.hwaddr, p, 6); + } +#endif + reg = *(unsigned long *)(dev->dev_addr); + RTL_W32(MACID, (cpu_to_le32(reg))); + reg = *(unsigned short *)((unsigned long)dev->dev_addr + 4); + RTL_W16(MACID+4, (cpu_to_le16(reg))); +#ifdef CONFIG_RTL_92D_SUPPORT + RTL_W32(MACID1 , RTL_R32(MACID)); + RTL_W16(MACID1+4, RTL_R16(MACID+4)); +#endif + +#ifdef MBSSID + if (GET_ROOT(priv)->pmib->miscEntry.vap_enable) + { + if (OPMODE & WIFI_AP_STATE) + rtl8192cd_init_mbssid(priv); + } +#endif + + RESTORE_INT(flags); + SMP_UNLOCK(flags); + +#ifdef CLIENT_MODE + if (!(OPMODE & WIFI_AP_STATE) && netif_running(priv->dev)) { + int link_status = chklink_wkstaQ(priv); + if (link_status) + start_clnt_join(priv); + } +#endif + + return 0; +} + + +int rtl8192cd_close(struct net_device *dev) +{ +#ifdef NETDEV_NO_PRIV + struct rtl8192cd_priv *priv = ((struct rtl8192cd_priv *)netdev_priv(dev))->wlan_priv; +#else + struct rtl8192cd_priv *priv = dev->priv; +#endif +#ifdef UNIVERSAL_REPEATER + struct rtl8192cd_priv *priv_vxd; +#endif + + unsigned long flags; + + DBFENTER; + + if (!(priv->drv_state & DRV_STATE_OPEN) +#ifdef WDS + && dev->base_addr +#endif +) { + DBFEXIT; + return 0; + } + + SAVE_INT_AND_CLI(flags); + SMP_LOCK(flags); + +#ifdef WDS + if(dev->base_addr) +#endif + priv->drv_state &= ~DRV_STATE_OPEN; // set driver as has been closed, david + +#ifdef PCIE_POWER_SAVING + if((priv->pwr_state == L1) || (priv->pwr_state == L2)) { + PCIeWakeUp(priv, (POWER_DOWN_T0<<3)); + } +#endif + +#if defined(RESERVE_TXDESC_FOR_EACH_IF) && (defined(UNIVERSAL_REPEATER) || defined(MBSSID)) + recalc_txdesc_limit(priv); +#endif + +#ifdef UNIVERSAL_REPEATER + if (IS_ROOT_INTERFACE(priv)) { + priv_vxd = GET_VXD_PRIV(priv); + + // if vxd interface is opened, close it first + if (IS_DRV_OPEN(priv_vxd)) + rtl8192cd_close(priv_vxd->dev); + } + else { +#ifdef MBSSID +/* + if (GET_ROOT(priv)->pmib->miscEntry.vap_enable) +*/ + if (priv->vap_id < 0) +#endif + disable_vxd_ap(priv); + } +#endif + + netif_stop_queue(dev); + +#ifdef WDS + if (dev->base_addr == 0) + { + RESTORE_INT(flags); + SMP_UNLOCK(flags); + DBFEXIT; + return 0; + } +#endif + +#ifdef CONFIG_RTK_MESH + if (dev->base_addr == 1) + { + RESTORE_INT(flags); + SMP_UNLOCK(flags); + return 0; + } +#endif // CONFIG_RTK_MESH + +#ifdef CHECK_HANGUP + if (!priv->reset_hangup) +#endif + if (OPMODE & WIFI_AP_STATE) { + int i; + for(i=0; i<NUM_STAT; i++) + { + if (priv->pshare->aidarray[i] && (priv->pshare->aidarray[i]->used == TRUE) +#ifdef WDS + && !(priv->pshare->aidarray[i]->station.state & WIFI_WDS) +#endif + ) { +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (priv != priv->pshare->aidarray[i]->priv) + continue; +#endif + issue_deauth(priv, priv->pshare->aidarray[i]->station.hwaddr, _RSON_DEAUTH_STA_LEAVING_); + } + } + + delay_ms(10); + } +#ifdef CLIENT_MODE /* WPS2DOTX */ + else if((OPMODE & (WIFI_STATION_STATE|WIFI_AUTH_SUCCESS|WIFI_ASOC_STATE))==(WIFI_STATION_STATE|WIFI_AUTH_SUCCESS|WIFI_ASOC_STATE)){ + //printMac(BSSID); + issue_disassoc(priv,BSSID,_RSON_DEAUTH_STA_LEAVING_); + delay_ms(30);//make sure before issue_disassoc then TX be close + OPMODE &= ~(WIFI_AUTH_SUCCESS|WIFI_ASOC_STATE) ; + } +#endif /* WPS2DOTX */ + +#ifdef MBSSID + if (IS_ROOT_INTERFACE(priv)) { + int i; + if (priv->pmib->miscEntry.vap_enable) { + for (i=0; i<RTL8192CD_NUM_VWLAN; i++) { + if (IS_DRV_OPEN(priv->pvap_priv[i])) + rtl8192cd_close(priv->pvap_priv[i]->dev); + } + } + } + + if (GET_ROOT(priv)->pmib->miscEntry.vap_enable) + rtl8192cd_stop_mbssid(priv); +#endif + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { +#ifdef RTK_QUE + free_rtk_queue(priv, &priv->pshare->skb_queue); +#else + free_skb_queue(priv, &priv->pshare->skb_queue); +#endif + + rtl8192cd_stop_hw(priv); + +#ifdef __KERNEL__ +#ifdef CHECK_HANGUP + if (!priv->reset_hangup) +#endif + { + free_irq(dev->irq, dev); +#ifdef PCIE_POWER_SAVING +#ifdef GPIO_WAKEPIN +#ifdef CONFIG_RTL_8198 + REG32(0xb8003000) &= ~ BIT(16); // GIMR +#else + if (REG32(REVR) == RTL8196C_REVISION_B) + REG32(0xb8003000) &= ~ BIT(9); // GIMR +#endif +#if defined(__LINUX_2_6__) + free_irq(BSP_GPIO_ABCD_IRQ, dev); +#else + free_irq(1, dev); +#endif +#endif +#endif + } +#endif + + + +#ifdef UNIVERSAL_REPEATER + GET_VXD_PRIV(priv)->drv_state &= ~DRV_STATE_VXD_INIT; +#endif + } + + rtl8192cd_stop_sw(priv); + +#ifdef ENABLE_RTL_SKB_STATS + DEBUG_INFO("skb_tx_cnt =%d\n", rtl_atomic_read(&priv->rtl_tx_skb_cnt)); + DEBUG_INFO("skb_rx_cnt =%d\n", rtl_atomic_read(&priv->rtl_rx_skb_cnt)); +#endif + +#if defined(CONFIG_RTL_92D_DMDP) && defined(CHECK_HANGUP) +#ifdef MBSSID + if (IS_ROOT_INTERFACE(priv)) +#endif + if ((GET_CHIP_VER(priv) == VERSION_8192D) + && (priv->pshare->rf_ft_var.peerReinit) + && (priv->pmib->dot11RFEntry.macPhyMode == DUALMAC_DUALPHY)) + reset_dmdp_peer(priv); +#endif + + RESTORE_INT(flags); + SMP_UNLOCK(flags); + + +#if 0 +//#ifdef PCIE_POWER_SAVING + if(!IS_UMC_A_CUT_88C(priv)) +#ifdef CHECK_HANGUP + if (!priv->reset_hangup) +#endif +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + HostPCIe_Close(); +#endif + + DBFEXIT; + return 0; +} + + +#ifdef CONFIG_RTL_KERNEL_MIPS16_WLAN +__NOMIPS16 +#endif +static void MDL_DEVINIT set_mib_default(struct rtl8192cd_priv *priv) +{ + unsigned char *p; +#ifdef __KERNEL__ + struct sockaddr addr; + p = addr.sa_data; +#else + unsigned char tmpbuf[10]; + p = (unsigned char *)tmpbuf; +#endif + + priv->pmib->mib_version = MIB_VERSION; + set_mib_default_tbl(priv); + + // others that are not types of byte and int + strcpy((char *)priv->pmib->dot11StationConfigEntry.dot11DesiredSSID, "RTL8186-default"); + priv->pmib->dot11StationConfigEntry.dot11DesiredSSIDLen = strlen("RTL8186-default"); + memcpy(p, "\x00\xe0\x4c\x81\x86\x86", MACADDRLEN); + +#ifdef DOT11D + // set countryCode for 11d + strcpy(priv->pmib->dot11dCountry.dot11CountryString, "US"); +#endif + +#ifdef __KERNEL__ + rtl8192cd_set_hwaddr(priv->dev, (void *)&addr); +#else + rtl8192cd_set_hwaddr(priv->dev, (void *)p); +#endif + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + { +#ifdef DFS + init_timer(&priv->ch52_timer); + priv->ch52_timer.data = (unsigned long) priv; + priv->ch52_timer.function = rtl8192cd_ch52_timer; + + init_timer(&priv->ch56_timer); + priv->ch56_timer.data = (unsigned long) priv; + priv->ch56_timer.function = rtl8192cd_ch56_timer; + + init_timer(&priv->ch60_timer); + priv->ch60_timer.data = (unsigned long) priv; + priv->ch60_timer.function = rtl8192cd_ch60_timer; + + init_timer(&priv->ch64_timer); + priv->ch64_timer.data = (unsigned long) priv; + priv->ch64_timer.function = rtl8192cd_ch64_timer; + + init_timer(&priv->ch100_timer); + priv->ch100_timer.data = (unsigned long) priv; + priv->ch100_timer.function = rtl8192cd_ch100_timer; + + init_timer(&priv->ch104_timer); + priv->ch104_timer.data = (unsigned long) priv; + priv->ch104_timer.function = rtl8192cd_ch104_timer; + + init_timer(&priv->ch108_timer); + priv->ch108_timer.data = (unsigned long) priv; + priv->ch108_timer.function = rtl8192cd_ch108_timer; + + init_timer(&priv->ch112_timer); + priv->ch112_timer.data = (unsigned long) priv; + priv->ch112_timer.function = rtl8192cd_ch112_timer; + + init_timer(&priv->ch116_timer); + priv->ch116_timer.data = (unsigned long) priv; + priv->ch116_timer.function = rtl8192cd_ch116_timer; + + init_timer(&priv->ch120_timer); + priv->ch120_timer.data = (unsigned long) priv; + priv->ch120_timer.function = rtl8192cd_ch120_timer; + + init_timer(&priv->ch124_timer); + priv->ch124_timer.data = (unsigned long) priv; + priv->ch124_timer.function = rtl8192cd_ch124_timer; + + init_timer(&priv->ch128_timer); + priv->ch128_timer.data = (unsigned long) priv; + priv->ch128_timer.function = rtl8192cd_ch128_timer; + + init_timer(&priv->ch132_timer); + priv->ch132_timer.data = (unsigned long) priv; + priv->ch132_timer.function = rtl8192cd_ch132_timer; + + init_timer(&priv->ch136_timer); + priv->ch136_timer.data = (unsigned long) priv; + priv->ch136_timer.function = rtl8192cd_ch136_timer; + + init_timer(&priv->ch140_timer); + priv->ch140_timer.data = (unsigned long) priv; + priv->ch140_timer.function = rtl8192cd_ch140_timer; +#endif + + if (((priv->pshare->type>>TYPE_SHIFT) & TYPE_MASK) == TYPE_EMBEDDED) { + // not implement yet + } else { +#ifdef IO_MAPPING + priv->pshare->io_mapping = 1; +#endif + } + } +} + +#if defined(__LINUX_2_6__) && !defined(CONFIG_COMPAT_NET_DEV_OPS) +static const struct net_device_ops rtl8192cd_netdev_ops = { + .ndo_open = rtl8192cd_open, + .ndo_stop = rtl8192cd_close, + .ndo_set_mac_address = rtl8192cd_set_hwaddr, + .ndo_set_multicast_list = rtl8192cd_set_rx_mode, + .ndo_get_stats = rtl8192cd_get_stats, + .ndo_do_ioctl = rtl8192cd_ioctl, + .ndo_start_xmit = rtl8192cd_start_xmit, +}; +#endif + + +#ifdef CONFIG_RTL_KERNEL_MIPS16_WLAN +__NOMIPS16 +#endif +static int MDL_DEVINIT rtl8192cd_init_one(struct pci_dev *pdev, + const struct pci_device_id *ent, struct _device_info_ *wdev, int vap_idx) +{ +#if defined(NOT_RTK_BSP) + unsigned long page_align_phy=0; +#endif + struct net_device *dev; + struct rtl8192cd_priv *priv; + void *regs; + struct wifi_mib *pmib; + DOT11_QUEUE *pevent_queue; +#ifdef CONFIG_RTL_WAPI_SUPPORT + DOT11_QUEUE *wapiEvent_queue; + #if defined(MBSSID) + DOT11_QUEUE *wapiVapEvent_queue; + #endif + +#endif + struct rtl8192cd_hw *phw; + struct rtl8192cd_tx_desc_info *ptxdesc; + struct wlan_hdr_poll *pwlan_hdr_poll; + struct wlanllc_hdr_poll *pwlanllc_hdr_poll; + struct wlanbuf_poll *pwlanbuf_poll; + struct wlanicv_poll *pwlanicv_poll; + struct wlanmic_poll *pwlanmic_poll; + struct wlan_acl_poll *pwlan_acl_poll; + DOT11_EAP_PACKET *Eap_packet; +#if defined(INCLUDE_WPA_PSK) || defined(WIFI_HAPD) + WPA_GLOBAL_INFO *wpa_global_info; +#endif +#ifdef CONFIG_RTK_MESH +#ifdef _MESH_ACL_ENABLE_ + struct mesh_acl_poll *pmesh_acl_poll = NULL; +#endif +#ifdef _11s_TEST_MODE_ + struct Galileo_poll *pgalileo_poll = NULL; +#endif +#ifndef WDS + char baseDevName[8]; +#endif + int mesh_num; + + struct hash_table *proxy_table; +#ifdef PU_STANDARD + //pepsi + struct hash_table *proxyupdate_table; +#endif + struct mpp_tb *pann_mpp_tb; + struct hash_table *mesh_rreq_retry_queue; + // add by chuangch 2007.09.13 + struct hash_table *pathsel_table; + + DOT11_QUEUE2 *pathsel_queue ; +#ifdef _11s_TEST_MODE_ + DOT11_QUEUE2 *receiver_queue = NULL; +#endif +#endif // CONFIG_RTK_MESH + + + +#ifdef CONFIG_NET_PCI + u8 cache_size; + u16 pci_command; +#ifndef USE_IO_OPS + u32 pciaddr; +#endif + u32 pmem_len; +#endif + +#ifdef WDS + int wds_num; + char baseDevName[8]; +#endif +#if defined(CONFIG_RTK_MESH) || defined(WDS) + int i; +#endif + + unsigned char *page_ptr; + struct priv_shared_info *pshare; // david + + int rc=0; + + priv = NULL; + regs = NULL; + pmib = NULL; + pevent_queue = NULL; +#ifdef CONFIG_RTL_WAPI_SUPPORT + wapiEvent_queue = NULL; + #if defined(MBSSID) + wapiVapEvent_queue = NULL; + #endif + +#endif + phw = NULL; + ptxdesc = NULL; + pwlan_hdr_poll = NULL; + pwlanllc_hdr_poll = NULL; + pwlanbuf_poll = NULL; + pwlanicv_poll = NULL; + pwlanmic_poll = NULL; + pwlan_acl_poll = NULL; + Eap_packet = NULL; +#if defined(INCLUDE_WPA_PSK) || defined(WIFI_HAPD) + wpa_global_info = NULL; +#endif + + + + +#ifdef CONFIG_NET_PCI + cache_size=0; +#ifndef USE_IO_OPS + pciaddr=0; +#endif + pmem_len=0; +#endif + + + pshare = NULL; // david + + printk("=====>>INSIDE rtl8192cd_init_one <<=====\n"); + dev = alloc_etherdev(sizeof(struct rtl8192cd_priv)); + if (!dev) { + printk(KERN_ERR "alloc_etherdev() error!\n"); + return -ENOMEM; + } + + // now, allocating memory for pmib +#ifdef RTL8192CD_VARIABLE_USED_DMEM + pmib = (struct wifi_mib *)rtl8192cd_dmem_alloc(PMIB, NULL); +#else + pmib = (struct wifi_mib *)kmalloc((sizeof(struct wifi_mib)), GFP_ATOMIC); +#endif + if (!pmib) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for wifi_mib (size %d)\n", sizeof(struct wifi_mib)); + goto err_out_free; + } + memset(pmib, 0, sizeof(struct wifi_mib)); + + pevent_queue = (DOT11_QUEUE *)kmalloc((sizeof(DOT11_QUEUE)), GFP_ATOMIC); + if (!pevent_queue) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for DOT11_QUEUE (size %d)\n", sizeof(DOT11_QUEUE)); + goto err_out_free; + } + memset((void *)pevent_queue, 0, sizeof(DOT11_QUEUE)); +#ifdef CONFIG_RTL_WAPI_SUPPORT + wapiEvent_queue = (DOT11_QUEUE *)kmalloc((sizeof(DOT11_QUEUE)), GFP_ATOMIC); + if (!wapiEvent_queue) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for DOT11_QUEUE (size %d)\n", sizeof(DOT11_QUEUE)); + goto err_out_free; + } + memset((void *)wapiEvent_queue, 0, sizeof(DOT11_QUEUE)); +#ifdef MBSSID + wapiVapEvent_queue = (DOT11_QUEUE *)kmalloc((sizeof(DOT11_QUEUE)*RTL8192CD_NUM_VWLAN), GFP_ATOMIC); + if (!wapiVapEvent_queue) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for DOT11_QUEUE (size %d)\n", sizeof(DOT11_QUEUE)*RTL8192CD_NUM_VWLAN); + goto err_out_free; + } + memset((void *)wapiVapEvent_queue, 0, sizeof(DOT11_QUEUE)*RTL8192CD_NUM_VWLAN); + #endif + +#endif + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (wdev->priv == NULL) // root interface +#endif + { +#ifdef PRIV_STA_BUF +#ifdef CONCURRENT_MODE + phw = &hw_info[wlandev_idx]; +#else + phw = &hw_info; +#endif +#else + phw = (struct rtl8192cd_hw *)kmalloc((sizeof(struct rtl8192cd_hw)), GFP_ATOMIC); + if (!phw) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for rtl8192cd_hw (size %d)\n", sizeof(struct rtl8192cd_hw)); + goto err_out_free; + } +#endif + memset((void *)phw, 0, sizeof(struct rtl8192cd_hw)); + ptxdesc = &phw->tx_info; + +#ifdef PRIV_STA_BUF +#ifdef CONCURRENT_MODE + pshare = &shared_info[wlandev_idx]; +#else + pshare = &shared_info; +#endif +#else + pshare = (struct priv_shared_info *)kmalloc(sizeof(struct priv_shared_info), GFP_ATOMIC); + if (!pshare) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for priv_shared_info (size %d)\n", sizeof(struct priv_shared_info)); + goto err_out_free; + } +#endif + memset((void *)pshare, 0, sizeof(struct priv_shared_info)); + +#ifdef PRIV_STA_BUF +#ifdef CONCURRENT_MODE + pwlan_hdr_poll = &hdr_pool[wlandev_idx]; +#else + pwlan_hdr_poll = &hdr_pool; +#endif +#else + pwlan_hdr_poll = (struct wlan_hdr_poll *) + kmalloc((sizeof(struct wlan_hdr_poll)), GFP_ATOMIC); + if (!pwlan_hdr_poll) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for wlan_hdr_poll (size %d)\n", sizeof(struct wlan_hdr_poll)); + goto err_out_free; + } +#endif + +#ifdef PRIV_STA_BUF +#ifdef CONCURRENT_MODE + pwlanllc_hdr_poll = &llc_pool[wlandev_idx]; +#else + pwlanllc_hdr_poll = &llc_pool; +#endif +#else + pwlanllc_hdr_poll = (struct wlanllc_hdr_poll *) + kmalloc((sizeof(struct wlanllc_hdr_poll)), GFP_ATOMIC); + if (!pwlanllc_hdr_poll) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for wlanllc_hdr_poll (size %d)\n", sizeof(struct wlanllc_hdr_poll)); + goto err_out_free; + } +#endif + +#ifdef PRIV_STA_BUF +#ifdef CONCURRENT_MODE + pwlanbuf_poll = &buf_pool[wlandev_idx]; +#else + pwlanbuf_poll = &buf_pool; +#endif +#else + pwlanbuf_poll = (struct wlanbuf_poll *) + kmalloc((sizeof(struct wlanbuf_poll)), GFP_ATOMIC); + if (!pwlanbuf_poll) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for wlanbuf_poll (size %d)\n", sizeof(struct wlanbuf_poll)); + goto err_out_free; + } +#endif + +#ifdef PRIV_STA_BUF +#ifdef CONCURRENT_MODE + pwlanicv_poll = &icv_pool[wlandev_idx]; +#else + pwlanicv_poll = &icv_pool; +#endif +#else + pwlanicv_poll = (struct wlanicv_poll *) + kmalloc((sizeof(struct wlanicv_poll)), GFP_ATOMIC); + if (!pwlanicv_poll) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for wlanicv_poll (size %d)\n", sizeof(struct wlanicv_poll)); + goto err_out_free; + } +#endif + +#ifdef PRIV_STA_BUF +#ifdef CONCURRENT_MODE + pwlanmic_poll = &mic_pool[wlandev_idx]; + wlandev_idx++; +#else + pwlanmic_poll = &mic_pool; +#endif +#else + pwlanmic_poll = (struct wlanmic_poll *) + kmalloc((sizeof(struct wlanmic_poll)), GFP_ATOMIC); + if (!pwlanmic_poll) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for wlanmic_poll (size %d)\n", sizeof(struct wlanmic_poll)); + goto err_out_free; + } +#endif + } + + pwlan_acl_poll = (struct wlan_acl_poll *) + kmalloc((sizeof(struct wlan_acl_poll)), GFP_ATOMIC); + if (!pwlan_acl_poll) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for wlan_acl_poll (size %d)\n", sizeof(struct wlan_acl_poll)); + goto err_out_free; + } + +#if defined(CONFIG_RTK_MESH) && defined(_MESH_ACL_ENABLE_) // below code copy above ACL code + pmesh_acl_poll = (struct mesh_acl_poll *) + kmalloc((sizeof(struct mesh_acl_poll)), GFP_ATOMIC); + if (!pmesh_acl_poll) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for Mesh wlan_acl_poll (size %d)\n", sizeof(struct mesh_acl_poll)); + goto err_out_free; + } +#endif + + Eap_packet = (DOT11_EAP_PACKET *) + kmalloc((sizeof(DOT11_EAP_PACKET)), GFP_ATOMIC); + if (!Eap_packet) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for Eap_packet (size %d)\n", sizeof(DOT11_EAP_PACKET)); + goto err_out_free; + } + memset((void *)Eap_packet, 0, sizeof(DOT11_EAP_PACKET)); + +#if defined(INCLUDE_WPA_PSK) || defined(WIFI_HAPD) + wpa_global_info = (WPA_GLOBAL_INFO *) + kmalloc((sizeof(WPA_GLOBAL_INFO)), GFP_ATOMIC); + if (!wpa_global_info) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for wpa_global_info (size %d)\n", sizeof(WPA_GLOBAL_INFO)); + goto err_out_free; + } + memset((void *)wpa_global_info, 0, sizeof(WPA_GLOBAL_INFO)); +#endif + +#ifndef __DRAYTEK_OS__ +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (wdev->priv) { +#ifdef UNIVERSAL_REPEATER + if (vap_idx < 0) + sprintf(dev->name, "%s-vxd", wdev->priv->dev->name); +#endif +#ifdef MBSSID + if (vap_idx >= 0) +#ifdef CONFIG_RTL8672 + sprintf(dev->name, "%s-vap%d", wdev->priv->dev->name, vap_idx); +#else + sprintf(dev->name, "%s-va%d", wdev->priv->dev->name, vap_idx); +#endif +#endif + } + else +#endif + strcpy(dev->name, "wlan%d"); +#endif + +#ifdef LINUX_2_6_24_ + /*SET_MODULE_OWNER is obsolete from 2.6.24*/ +#else + SET_MODULE_OWNER(dev); +#endif + +#ifdef NETDEV_NO_PRIV + priv = (struct rtl8192cd_priv *)netdev_priv(dev); + priv->wlan_priv = priv; +#else + priv = dev->priv; +#endif + + priv->pmib = pmib; +#if 0/*defined(CONFIG_RTL_WAPI_SUPPORT)*/ + /* only for test */ + priv->pmib->wapiInfo.wapiType = wapiDisable; + priv->pmib->wapiInfo.wapiUpdateMCastKeyType = wapi_disable_update; + priv->pmib->wapiInfo.wapiUpdateUCastKeyTimeout = WAPI_KEY_UPDATE_PERIOD; + priv->pmib->wapiInfo.wapiUpdateUCastKeyPktNum = WAPI_KEY_UPDATE_PKTCNT; +#endif + priv->pevent_queue = pevent_queue; +#ifdef CONFIG_RTL_WAPI_SUPPORT + priv->wapiEvent_queue= wapiEvent_queue; +#endif + priv->pwlan_acl_poll = pwlan_acl_poll; + priv->Eap_packet = Eap_packet; +#if defined(INCLUDE_WPA_PSK) || defined(WIFI_HAPD) + priv->wpa_global_info = wpa_global_info; +#endif +#ifdef MBSSID + priv->vap_id = -1; +#endif +#if defined(CONFIG_RTK_MESH) && defined(_MESH_ACL_ENABLE_) // below code copy above ACL code + priv->pmesh_acl_poll = pmesh_acl_poll; +#endif + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (wdev->priv) { + priv->pshare = wdev->priv->pshare; + GET_ROOT_PRIV(priv) = wdev->priv; +#ifdef UNIVERSAL_REPEATER + if (vap_idx < 0) // create for vxd + GET_VXD_PRIV(wdev->priv) = priv; +#endif +#ifdef MBSSID + if (vap_idx >= 0) { // create for vap + GET_ROOT_PRIV(priv)->pvap_priv[vap_idx] = priv; + priv->vap_id = vap_idx; + priv->vap_init_seq = -1; + + #if 0 /*defined(CONFIG_RTL_WAPI_SUPPORT)*/ + priv->pmib->wapiInfo.wapiType = wapiDisable; + priv->pmib->wapiInfo.wapiUpdateMCastKeyType = wapi_disable_update; + priv->pmib->wapiInfo.wapiUpdateUCastKeyTimeout = WAPI_KEY_UPDATE_PERIOD; + priv->pmib->wapiInfo.wapiUpdateUCastKeyPktNum = WAPI_KEY_UPDATE_PKTCNT; + priv->wapiEvent_queue= &wapiVapEvent_queue[vap_idx]; + printk("dev[%s]:wapiType[%d] UCastKeyType[%d] psk[%s] len[%d]\n", + priv->dev->name, priv->pmib->wapiInfo.wapiType, + priv->pmib->wapiInfo.wapiUpdateMCastKeyType, + priv->pmib->wapiInfo.wapiPsk.octet, + priv->pmib->wapiInfo.wapiPsk.len); + #endif + + + } +#endif + } + else +#endif + { + priv->pshare = pshare; // david + priv->pshare->phw = phw; +#ifdef CONCURRENT_MODE + priv->pshare->wlandev_idx = wlan_index; +#endif + priv->pshare->pdesc_info = ptxdesc; + priv->pshare->pwlan_hdr_poll = pwlan_hdr_poll; + priv->pshare->pwlanllc_hdr_poll = pwlanllc_hdr_poll; + priv->pshare->pwlanbuf_poll = pwlanbuf_poll; + priv->pshare->pwlanicv_poll = pwlanicv_poll; + priv->pshare->pwlanmic_poll = pwlanmic_poll; + wdev->priv = priv; +#ifdef __KERNEL__ + spin_lock_init(&priv->pshare->lock); +#endif + +#ifdef CONFIG_RTK_MESH + spin_lock_init(&priv->pshare->lock_queue); + spin_lock_init(&priv->pshare->lock_Rreq); +#endif + +#ifdef SMP_SYNC + spin_lock_init(&priv->pshare->lock_xmit); + spin_lock_init(&priv->pshare->lock_skb); + spin_lock_init(&priv->pshare->lock_buf); +#endif + +#ifdef CONFIG_NET_PCI + if (((wdev->type >> TYPE_SHIFT) & TYPE_MASK) == TYPE_PCI_BIOS) + priv->pshare->pdev = pdev; +#endif + priv->pshare->type = wdev->type; +#ifdef USE_RTL8186_SDK +#if defined(CONFIG_RTL8196B) || defined(CONFIG_RTL_819X) + priv->pshare->have_hw_mic = 1; +#else + priv->pshare->have_hw_mic = 0; +#endif +#else + priv->pshare->have_hw_mic = 0; +#endif +// priv->pshare->is_giga_exist = is_giga_board(); + } + + priv->dev = dev; +#ifndef __DRAYTEK_OS__ +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (GET_ROOT_PRIV(priv)) { // is a vxd or vap + dev->base_addr = GET_ROOT_PRIV(priv)->dev->base_addr; + goto register_driver; + } +#endif +#ifdef CONFIG_NET_PCI + if (((wdev->type >> TYPE_SHIFT) & TYPE_MASK) == TYPE_PCI_BIOS) + { + rc = pci_enable_device(pdev); + if (rc) + goto err_out_free; +#ifndef USE_IO_OPS + rc = pci_request_regions(pdev, DRV_NAME); +#endif + if (rc) + goto err_out_disable; + + if (pdev->irq < 2) { + rc = -EIO; +#ifdef __LINUX_2_6__ + printk(KERN_ERR "invalid irq (%d) for pci dev\n", pdev->irq); +#else + printk(KERN_ERR "invalid irq (%d) for pci dev %s\n", pdev->irq, pdev->slot_name); +#endif + goto err_out_res; + } +#ifdef USE_IO_OPS + { + unsigned long pio_start, pio_len, pio_flags; + + pio_start = (unsigned long)pci_resource_start(pdev, 0); + pio_len = (unsigned long)pci_resource_len(pdev, 0); + pio_flags = (unsigned long)pci_resource_flags(pdev, 0); +//// pio_start = (unsigned long)pci_resource_start(pdev, 2); +//// pio_len = (unsigned long)pci_resource_len(pdev, 2); +//// pio_flags = (unsigned long)pci_resource_flags(pdev, 2); + + + if (!(pio_flags & IORESOURCE_IO)) { + rc = -EIO; +#ifdef __LINUX_2_6__ + printk(KERN_ERR "pci: region #0 not a PIO resource, aborting\n"); +#else + printk(KERN_ERR "%s: region #0 not a PIO resource, aborting\n", pdev->slot_name); +#endif + goto err_out_res; + } + + if (!request_region(pio_start, pio_len, DRV_NAME)) { + rc = -EIO; + printk(KERN_ERR "request_region failed!\n"); + goto err_out_res; + } + + if (pio_len < RTL8192CD_REGS_SIZE) { + rc = -EIO; +#ifdef __LINUX_2_6__ + printk(KERN_ERR "PIO resource (%lx) too small on pci dev\n", pio_len); +#else + printk(KERN_ERR "PIO resource (%lx) too small on pci dev %s\n", pio_len, pdev->slot_name); +#endif + goto err_out_res; + } + + dev->base_addr = pio_start; + priv->pshare->ioaddr = pio_start; // device I/O address + } +#else +#ifdef IO_MAPPING + pciaddr = pci_resource_start(pdev, 0); +//// pciaddr = pci_resource_start(pdev, 2); +#else +// pciaddr = pci_resource_start(pdev, 1); + pciaddr = pci_resource_start(pdev, 2); +#endif + if (!pciaddr) { + rc = -EIO; +#ifdef __LINUX_2_6__ + printk(KERN_ERR "no MMIO resource for pci dev"); +#else + printk(KERN_ERR "no MMIO resource for pci dev %s\n", pdev->slot_name); +#endif + goto err_out_res; + } + +// if ((pmem_len = pci_resource_len(pdev, 1)) < RTL8192CD_REGS_SIZE) { + if ((pmem_len = pci_resource_len(pdev, 2)) < RTL8192CD_REGS_SIZE) { + rc = -EIO; +#ifdef __LINUX_2_6__ + printk(KERN_ERR "MMIO resource () too small on pci dev\n"); +#else + printk(KERN_ERR "MMIO resource (%lx) too small on pci dev %s\n", (unsigned long)pmem_len, pdev->slot_name); +#endif + goto err_out_res; + } + + regs = ioremap_nocache(pciaddr, pmem_len); + if (!regs) { + rc = -EIO; +#ifdef __LINUX_2_6__ + printk(KERN_ERR "Cannot map PCI MMIO () on pci dev \n"); +#else + printk(KERN_ERR "Cannot map PCI MMIO (%lx@%lx) on pci dev %s\n", (unsigned long)pmem_len, (long)pciaddr, pdev->slot_name); +#endif + goto err_out_res; + } + + dev->base_addr = (unsigned long)regs; + priv->pshare->ioaddr = (UINT)regs; + check_chipID_MIMO(priv); +#endif // USE_IO_OPS +#ifdef CONFIG_RTL_92D_SUPPORT + u32 vendor_deivce_id, config_base; + + vendor_deivce_id = dev->base_addr; + if(dev->base_addr==0xba000000) + { + vendor_deivce_id= *((volatile unsigned long *)(0xb8b30000)); + } + else if(dev->base_addr==0xb9000000) + { + vendor_deivce_id= *((volatile unsigned long *)(0xb8b10000)); + } + printk(" vendor_deivce_id=%x\n", vendor_deivce_id); + if (vendor_deivce_id == ((unsigned long)((0x8193<<16)|PCI_VENDOR_ID_REALTEK))) + priv->pshare->version_id = VERSION_8192D; + else + priv->pshare->version_id =0x1234; +#endif + + } + else +#endif + { + regs = (void *)wdev->base_addr; + dev->base_addr = (unsigned long)wdev->base_addr; + priv->pshare->ioaddr = (UINT)regs; + + if (((priv->pshare->type>>TYPE_SHIFT) & TYPE_MASK) == TYPE_PCI_DIRECT) + { + int i ; +#ifdef CONFIG_RTL_92D_SUPPORT + u32 vendor_deivce_id, config_base, base_addr; + config_base = wdev->conf_addr; + base_addr = wdev->base_addr; +#endif + + _DEBUG_INFO("INIT PCI config space directly\n"); +#if !defined(CONFIG_NET_PCI) && (defined(CONFIG_RTL8196B) || defined(CONFIG_RTL_819X)) + #if defined(CONFIG_RTL8196C) || defined(CONFIG_RTL_8196C) || defined(CONFIG_RTL_819X) + if (BSP_PCIE0_D_CFG0 == wdev->conf_addr) + { + if ((PCIE_reset_procedure(0, 0, 1, wdev->conf_addr))== FAIL) + { + rc = -ENODEV; + goto err_out_free; + } + } + else if (BSP_PCIE1_D_CFG0 == wdev->conf_addr) + { + if ((PCIE_reset_procedure(1, 0, 1, wdev->conf_addr))== FAIL) + { + rc = -ENODEV; + goto err_out_free; + } + } + else + { + goto err_out_free; + } + #else + if (rtl8196b_pci_reset(wdev->conf_addr) == FAIL) + goto err_out_free; + #endif + //REG32(0xb8000040)=0xf; // port 5 support !!!!! +#endif + + + { + u32 vendor_deivce_id, config_base; + config_base = wdev->conf_addr; + vendor_deivce_id = *((volatile unsigned long *)(config_base+0)); + printk("vendor_deivce_id=%x\n", vendor_deivce_id); + if ( +#if defined(CONFIG_RTL_92D_SUPPORT) + (vendor_deivce_id != ((unsigned long)((0x8193<<16)|PCI_VENDOR_ID_REALTEK))) && +#endif + (vendor_deivce_id != ((unsigned long)((0x8191<<16)|PCI_VENDOR_ID_REALTEK))) && + (vendor_deivce_id != ((unsigned long)((0x8171<<16)|PCI_VENDOR_ID_REALTEK))) && + (vendor_deivce_id != ((unsigned long)((0x8178<<16)|PCI_VENDOR_ID_REALTEK))) && + (vendor_deivce_id != ((unsigned long)((0x8174<<16)|PCI_VENDOR_ID_REALTEK))) && + (vendor_deivce_id != ((unsigned long)((0x8176<<16)|PCI_VENDOR_ID_REALTEK)))) + { + _DEBUG_ERR("vendor_deivce_id=%x not match\n", vendor_deivce_id); + rc = -EIO; + goto err_out_free; + } +#ifdef CONFIG_RTL_92D_SUPPORT + if (vendor_deivce_id == ((unsigned long)((0x8193<<16)|PCI_VENDOR_ID_REALTEK))) + priv->pshare->version_id = VERSION_8192D; + else + priv->pshare->version_id = 0; +#endif + } + + *((volatile unsigned long *)PCI_CONFIG_BASE1) = virt_to_bus((void *)dev->base_addr); + //DEBUG_INFO("...config_base1 = 0x%08lx\n", *((volatile unsigned long *)PCI_CONFIG_BASE1)); + for(i=0; i<1000000; i++); + *((volatile unsigned char *)PCI_CONFIG_COMMAND) = 0x07; + //DEBUG_INFO("...command = 0x%08lx\n", *((volatile unsigned long *)PCI_CONFIG_COMMAND)); + for(i=0; i<1000000; i++); + *((volatile unsigned short *)PCI_CONFIG_LATENCY) = 0x2000; + for(i=0; i<1000000; i++); + //DEBUG_INFO("...latency = 0x%08lx\n", *((volatile unsigned long *)PCI_CONFIG_LATENCY)); + +#if defined(CONFIG_RTL8196C_REVISION_B) || defined(CONFIG_RTL_8196C) + #if defined(CONFIG_NET_PCI) + #define REVR 0xB8000000 + #define RTL8196C_REVISION_A 0x80000001 + #define RTL8196C_REVISION_B 0x80000002 + #endif + if (REG32(REVR) == RTL8196C_REVISION_B) { + REG32(0xb9000354)=0xc940; //Card PCIE PHY initial parameter for rtl8196c revision B + REG32(0xb9000358)=0x24; + for(i=0; i<1000000; i++); + REG32(0xb9000354)=0x4270; + REG32(0xb9000358)=0x25; + for(i=0; i<1000000; i++); + REG32(0xb9000354)=0x019E; //Card PCIE PHY initial parameter for rtl8196c revision B + REG32(0xb9000358)=0x23; + } +#endif +#if defined(CONFIG_RTL8198_REVISION_B) || defined(CONFIG_RTL_8198) +#if !defined(CONFIG_RTL_92D_SUPPORT) + #if defined(CONFIG_NET_PCI) + #define BSP_REVR 0xB8000000 + #define BSP_RTL8198_REVISION_A 0xC0000000 + #define BSP_RTL8198_REVISION_B 0xC0000001 + #endif + #if !defined(CONFIG_RTL_DUAL_PCIESLOT_BIWLAN) + if (REG32(BSP_REVR) >= BSP_RTL8198_REVISION_B) { + REG32(0xb9000354)=0xc940; //Card PCIE PHY initial parameter for rtl8196c revision B + REG32(0xb9000358)=0x24; + for(i=0; i<1000000; i++); + REG32(0xb9000354)=0x4270; + REG32(0xb9000358)=0x25; + for(i=0; i<1000000; i++); + REG32(0xb9000354)=0x019E; //Card PCIE PHY initial parameter for rtl8196c revision B + REG32(0xb9000358)=0x23; + } + #endif +#endif +#endif + +#ifndef NOT_RTK_BSP + REG32(0xB800311C) |= 1 << 23; +#endif + check_chipID_MIMO(priv); + //Exception Case + if ( CheckNoResetHwExceptionCase(priv) ) { + rtl8192cd_stop_hw(priv); + } + } + } +/* ==========>> maybe later +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (IS_ROOT_INTERFACE(priv)) +#endif + rtl8192cd_ePhyInit(priv); +*/ + + +#ifdef CONFIG_NET_PCI + if (((wdev->type >> TYPE_SHIFT) & TYPE_MASK) == TYPE_PCI_BIOS) { + dev->irq = pdev->irq; + pci_set_drvdata(pdev, dev); + } + else +#endif + { + dev->irq = wdev->irq; + } + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) +register_driver: +#endif + +#if !defined(__LINUX_2_6__) || defined(CONFIG_COMPAT_NET_DEV_OPS) + dev->open = rtl8192cd_open; + dev->stop = rtl8192cd_close; + dev->set_multicast_list = rtl8192cd_set_rx_mode; + dev->hard_start_xmit = rtl8192cd_start_xmit; + dev->get_stats = rtl8192cd_get_stats; + dev->do_ioctl = rtl8192cd_ioctl; + dev->set_mac_address = rtl8192cd_set_hwaddr; +#else + dev->netdev_ops = &rtl8192cd_netdev_ops; +#endif + +#ifdef CONFIG_RTL8672 + dev->priv_flags = IFF_DOMAIN_WLAN; +#endif + + rc = register_netdev(dev); + if (rc) + goto err_out_iomap; + +#ifdef CONFIG_RTL8672 + // Added by Mason Yu + // MBSSID Port Mapping + wlanDev[wlanDevNum].dev_pointer = dev; + wlanDev[wlanDevNum].dev_ifgrp_member = 0; + wlanDevNum++; +#endif + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (GET_ROOT_PRIV(priv) == NULL) // is root interface +#endif + DEBUG_INFO("Init %s, base_addr=%08x, irq=%d\n", + dev->name, (UINT)dev->base_addr, dev->irq); + +#else // __DRAYTEK_OS__ + regs = (void *)wdev->base_addr; + dev->base_addr = (unsigned long)wdev->base_addr; + priv->pshare->ioaddr = (UINT)regs; + +#ifdef UNIVERSAL_REPEATER + if (GET_ROOT_PRIV(priv) == NULL) // is root interface +#endif + DEBUG_INFO("Init %s, base_addr=%08x\n", + dev->name, (UINT)dev->base_addr); + +#endif // __DRAYTEK_OS__ + + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (GET_ROOT_PRIV(priv) == NULL) // is root interface +#endif + { +#ifdef CONFIG_NET_PCI + if (((wdev->type >> TYPE_SHIFT) & TYPE_MASK) == TYPE_PCI_BIOS) { + if (cache_size != SMP_CACHE_BYTES) { +// printk(KERN_INFO "%s: PCI cache line size set incorrectly " +// "(%i bytes) by BIOS/FW, ", dev->name, cache_size); + if (cache_size > SMP_CACHE_BYTES) + printk("expecting %i\n", SMP_CACHE_BYTES); + else { + printk("correcting to %i\n", SMP_CACHE_BYTES); + pci_write_config_byte(pdev, PCI_CACHE_LINE_SIZE, + SMP_CACHE_BYTES >> 2); + } + } + + /* enable busmastering and memory-write-invalidate */ + pci_read_config_word(pdev, PCI_COMMAND, &pci_command); + if (!(pci_command & PCI_COMMAND_INVALIDATE)) { + pci_command |= PCI_COMMAND_INVALIDATE; + pci_write_config_word(pdev, PCI_COMMAND, pci_command); + } + pci_set_master(pdev); + } +#endif + } + + +#ifdef _INCLUDE_PROC_FS_ +#ifdef __KERNEL__ + rtl8192cd_proc_init(dev); +#ifdef PERF_DUMP + { + #include <linux/proc_fs.h> + + struct proc_dir_entry *res; + res = create_proc_entry("perf_dump", 0, NULL); + if (res) { + res->read_proc = read_perf_dump; + res->write_proc = flush_perf_dump; + } + } +#endif +#endif +#endif + + // set some default value of mib + set_mib_default(priv); + +#if defined(UNIVERSAL_REPEATER) || defined(MBSSID) + if (GET_ROOT_PRIV(priv) == NULL) // is root interface +#endif + { +#ifndef __DRAYTEK_OS__ +#ifdef WDS + wds_num = (priv->pshare->type>>WDS_SHIFT) & WDS_MASK; + strcpy(baseDevName, dev->name); + + for (i=0; i<wds_num; i++) { +#ifdef NETDEV_NO_PRIV + struct rtl8192cd_wds_priv* priv_wds; + dev = alloc_etherdev(sizeof(struct rtl8192cd_wds_priv)); +#else + dev = alloc_etherdev(0); +#endif + if (!dev) { + printk(KERN_ERR "alloc_etherdev() wds error!\n"); + rc = -ENOMEM; + goto err_out_dev; + } + +#if !defined(__LINUX_2_6__) || defined(CONFIG_COMPAT_NET_DEV_OPS) + dev->open = rtl8192cd_open; + dev->stop = rtl8192cd_close; + dev->hard_start_xmit = rtl8192cd_start_xmit; + dev->get_stats = rtl8192cd_get_stats; + dev->set_mac_address = rtl8192cd_set_hwaddr; +#else + dev->netdev_ops = &rtl8192cd_netdev_ops; +#endif + +#ifdef CONFIG_RTL8672 + dev->priv_flags = IFF_DOMAIN_WLAN; +#endif + + priv->wds_dev[i] = dev; + strcpy(dev->name, baseDevName); + strcat(dev->name, "-wds%d"); + +#ifdef NETDEV_NO_PRIV + priv_wds = (struct rtl8192cd_wds_priv*)netdev_priv(dev); + priv_wds->wlan_priv = priv; +#else + dev->priv = priv; +#endif + rc = register_netdev(dev); + if (rc) { + printk(KERN_ERR "register_netdev() wds error!\n"); + goto err_out_dev; + } + } +#endif // WDS + +#ifdef CONFIG_RTK_MESH + mesh_num = (priv->pshare->type>>MESH_SHIFT) & MESH_MASK; + +#ifndef WDS + strcpy(baseDevName, dev->name); +#endif + if(mesh_num>0) { + GET_MIB(priv)->dot1180211sInfo.mesh_enable = 1; + dev = alloc_etherdev(0); // mesh allocate ethernet device BUT don't have priv memory (Because share root priv) + if (!dev) { + printk(KERN_ERR "alloc_etherdev() mesh error!\n"); + rc = -ENOMEM; + goto err_out_iomap; + } + dev->base_addr = 1; +#if !defined(__LINUX_2_6__) || defined(CONFIG_COMPAT_NET_DEV_OPS) + dev->open = rtl8192cd_open; + dev->stop = rtl8192cd_close; + dev->hard_start_xmit = rtl8192cd_start_xmit; + dev->get_stats = rtl8192cd_get_stats; + dev->set_mac_address = rtl8192cd_set_hwaddr; + dev->do_ioctl = rtl8192cd_ioctl; +#else + dev->netdev_ops = &rtl8192cd_netdev_ops; +#endif + +#ifdef CONFIG_RTL8672 + dev->priv_flags = IFF_DOMAIN_WLAN; +#endif + + priv->mesh_dev = dev; // NO priv zone dev + strcpy(dev->name, baseDevName); + strcat(dev->name, "-msh%d"); + dev->priv = priv; // mesh priv pointer to root's priv + rc = register_netdev(dev); + if (rc) { + printk(KERN_ERR "register_netdev() mesh error!\n"); + goto err_out_iomap; + } + } // end of if(mesh_num>0) + + priv->RreqEnd = 0; + priv->RreqBegin = 0; + + pann_mpp_tb = (struct mpp_tb*)kmalloc(sizeof(struct mpp_tb), GFP_ATOMIC); + if(!pann_mpp_tb) + { + rc = -ENOMEM; + printk("allocate pann_mpp_tb error!!\n"); + goto err_out_free; + } + init_mpp_pool(pann_mpp_tb); + proxy_table = (struct hash_table*)kmalloc(sizeof(struct hash_table), GFP_ATOMIC); + if(!proxy_table) + { + rc = -ENOMEM; + printk("allocate proxy_table error!!\n"); + goto err_out_free; + } + memset((void*)proxy_table, 0, sizeof(struct hash_table)); + +#ifdef PU_STANDARD + //pepsi + proxyupdate_table = (struct hash_table*)kmalloc(sizeof(struct hash_table), GFP_ATOMIC); + if(!proxyupdate_table) + { + rc = -ENOMEM; + printk("allocate proxyupdate_table error!!\n"); + goto err_out_free; + } + memset((void*)proxyupdate_table, 0, sizeof(struct hash_table)); +#endif + + pathsel_queue = (DOT11_QUEUE2 *)kmalloc((sizeof(DOT11_QUEUE2)), GFP_ATOMIC); + if (!pathsel_queue) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for PATHSELECTION_QUEUE (size %d)\n", sizeof(DOT11_QUEUE)); + goto err_out_free; + } + memset((void *)pathsel_queue, 0, sizeof (DOT11_QUEUE2)); +#ifdef _11s_TEST_MODE_ + receiver_queue = (DOT11_QUEUE2 *)kmalloc((sizeof(DOT11_QUEUE2)), GFP_ATOMIC); + if (!receiver_queue) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for receiver_queue (size %d)\n", sizeof(DOT11_QUEUE)); + goto err_out_free; + } + memset((void *)receiver_queue, 0, sizeof (DOT11_QUEUE2)); + pgalileo_poll = (struct Galileo_poll *) kmalloc((sizeof( struct Galileo_poll)), GFP_ATOMIC); + if (!pgalileo_poll) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for pgalileo_poll (size %d)\n", sizeof(struct Galileo_poll)); + goto err_out_free; + } +#endif + pathsel_table = (struct hash_table*)kmalloc(sizeof(struct hash_table), GFP_ATOMIC); + if(!pathsel_table) + { + rc = -ENOMEM; + printk("allocate pathsel_table error!!\n"); + goto err_out_free; + } + memset((void*)pathsel_table, 0, sizeof(struct hash_table)); + + mesh_rreq_retry_queue = (struct hash_table*)kmalloc(sizeof(struct hash_table), GFP_ATOMIC); + if(!mesh_rreq_retry_queue) + { + rc = -ENOMEM; + printk("allocate mesh_rreq_retry_queue error!!\n"); + goto err_out_free; + } + memset((void*)mesh_rreq_retry_queue, 0, sizeof(struct hash_table)); + + rc = init_hash_table(proxy_table, PROXY_TABLE_SIZE, MACADDRLEN, sizeof(struct proxy_table_entry), crc_hashing, search_default, insert_default, delete_default,traverse_default); + if(rc == HASH_TABLE_FAILED) + { + printk("init_hash_table \"proxy_table\" error!!\n"); + } + +#ifdef PU_STANDARD + //pepsi + rc = init_hash_table(proxyupdate_table, 8, sizeof(UINT8), sizeof(struct proxyupdate_table_entry), PU_hashing, search_default, insert_default, delete_default,traverse_default); + if(rc == HASH_TABLE_FAILED) + { + printk("init_hash_table \"proxyupdate_table\" error!!\n"); + } +#endif + rc = init_hash_table(pathsel_table, 8, MACADDRLEN, sizeof(struct path_sel_entry), crc_hashing, search_default, insert_default, delete_default,traverse_default); + if(rc == HASH_TABLE_FAILED) + { + printk("init_hash_table \"pathsel_table\" error!!\n"); + } + + rc = init_hash_table(mesh_rreq_retry_queue, DATA_SKB_BUFFER_SIZE, MACADDRLEN, sizeof(struct mesh_rreq_retry_entry), crc_hashing, search_default, insert_default, delete_default,traverse_default); + if(rc == HASH_TABLE_FAILED) + { + printk("init_hash_table \"mesh_rreq_retry_queue\" error!!\n"); + } + + for(i = 0; i < (1 << mesh_rreq_retry_queue->table_size_power); i++) + { + (((struct mesh_rreq_retry_entry*)(mesh_rreq_retry_queue->entry_array[i].data))->ptr) = (struct pkt_queue*)kmalloc(sizeof(struct pkt_queue), GFP_ATOMIC); + if (!(((struct mesh_rreq_retry_entry*)(mesh_rreq_retry_queue->entry_array[i].data))->ptr)) { + rc = -ENOMEM; + printk(KERN_ERR "Can't kmalloc for mesh_rreq_retry_entry (size %d)\n", sizeof(struct pkt_queue)); + goto err_out_free; + } + memset((void *)((((struct mesh_rreq_retry_entry*)(mesh_rreq_retry_queue->entry_array[i].data))->ptr)), 0, sizeof (struct pkt_queue)); + } + +#ifdef PU_STANDARD + priv->proxyupdate_table = proxyupdate_table; +#endif +#ifdef _11s_TEST_MODE_ + priv->receiver_queue = receiver_queue; + priv->pshare->galileo_poll = pgalileo_poll ; +#endif + priv->proxy_table = proxy_table; + priv->pathsel_queue = pathsel_queue; + priv->pann_mpp_tb = pann_mpp_tb; + priv->pathsel_table = pathsel_table; + priv->mesh_rreq_retry_queue = mesh_rreq_retry_queue; + //========================================================= +#endif // CONFIG_RTK_MESH + + +#endif // __DRAYTEK_OS__ + +#ifdef PRIV_STA_BUF +#ifdef CONCURRENT_MODE + page_ptr = (unsigned char *) + (((unsigned long)&desc_buf[priv->pshare->wlandev_idx]) + (PAGE_SIZE - (((unsigned long)&desc_buf[priv->pshare->wlandev_idx]) & (PAGE_SIZE-1)))); + phw->ring_buf_len = ((unsigned long)&desc_buf[priv->pshare->wlandev_idx]) + (sizeof(desc_buf)/NUM_WLAN_IFACE) - ((unsigned long)page_ptr); + phw->ring_dma_addr = virt_to_bus(page_ptr); + page_ptr = (unsigned char *)KSEG1ADDR(page_ptr); +#else + page_ptr = (unsigned char *) + (((unsigned long)desc_buf) + (PAGE_SIZE - (((unsigned long)desc_buf) & (PAGE_SIZE-1)))); + phw->ring_buf_len = (unsigned long)desc_buf + sizeof(desc_buf) - (unsigned long)page_ptr; + phw->ring_dma_addr = virt_to_bus(page_ptr); + page_ptr = (unsigned char *)KSEG1ADDR(page_ptr); +#endif +#else +#ifdef CONFIG_NET_PCI + if (IS_PCIBIOS_TYPE) + page_ptr = pci_alloc_consistent(priv->pshare->pdev, DESC_DMA_PAGE_SIZE, (dma_addr_t *)&phw->ring_dma_addr); + else +#endif + { +#ifdef __DRAYTEK_OS__ + page_ptr = rtl8185_malloc(DESC_DMA_PAGE_SIZE, 1); // allocate non-cache buffer +#else + page_ptr = kmalloc(DESC_DMA_PAGE_SIZE, GFP_ATOMIC); +#endif + } + + if (page_ptr == NULL) { + printk(KERN_ERR "can't allocate descriptior page, abort!\n"); + goto err_out_dev; + } + + phw->alloc_dma_buf = (unsigned long)page_ptr; +#if defined(NOT_RTK_BSP) + page_align_phy = (PAGE_SIZE - (((unsigned long)page_ptr) & (PAGE_SIZE-1))); +#endif + page_ptr = (unsigned char *) + (((unsigned long)page_ptr) + (PAGE_SIZE - (((unsigned long)page_ptr) & (PAGE_SIZE-1)))); + phw->ring_buf_len = phw->alloc_dma_buf + DESC_DMA_PAGE_SIZE - ((unsigned long)page_ptr); +#if defined(NOT_RTK_BSP) + phw->ring_dma_addr = phw->ring_dma_addr + page_align_phy; +#else + phw->ring_dma_addr = virt_to_bus(page_ptr); +#endif + +#ifdef __MIPSEB__ + page_ptr = (unsigned char *)KSEG1ADDR(page_ptr); +#endif +#endif + + DEBUG_INFO("page_ptr=%lx, size=%ld\n", (unsigned long)page_ptr, (unsigned long)DESC_DMA_PAGE_SIZE); + phw->ring_virt_addr = (unsigned long)page_ptr; + +#ifdef CONFIG_RTL8190_PRIV_SKB + init_priv_skb_buf(priv); +#endif + +#ifdef PRIV_STA_BUF + init_priv_sta_buf(priv); +#endif + +#ifdef CONFIG_RTL_92D_DMDP + if_priv[priv->pshare->wlandev_idx] = (u32)priv; +#endif + } + + INIT_LIST_HEAD(&priv->asoc_list); // init assoc_list first because webs may get sta_num even it is not open, + // and it will cause exception if it is not init, david+2008-03-05 +#ifdef EN_EFUSE + extern int ReadAdapterInfo8192CE(struct rtl8192cd_priv *priv); + ReadAdapterInfo8192CE(priv); +#endif + printk("=====>>EXIT rtl8192cd_init_one <<=====\n"); + return 0; + +err_out_dev: + + unregister_netdev(dev); + +err_out_iomap: + +#ifdef CONFIG_NET_PCI + if (((wdev->type >> TYPE_SHIFT) & TYPE_MASK) == TYPE_PCI_BIOS) { +#ifndef USE_IO_OPS + iounmap(regs); +#endif + } + +err_out_res: + + if (((wdev->type >> TYPE_SHIFT) & TYPE_MASK) == TYPE_PCI_BIOS) { +#ifdef USE_IO_OPS + release_region(dev->base_addr, pci_resource_len(pdev, 0)); +//// release_region(dev->base_addr, pci_resource_len(pdev, 2)); +#else + pci_release_regions(pdev); +#endif + } + +err_out_disable: + + if (((wdev->type >> TYPE_SHIFT) & TYPE_MASK) == TYPE_PCI_BIOS) + pci_disable_device(pdev); +#endif // CONFIG_NET_PCI + +err_out_free: + + if (pmib){ +#ifdef RTL8192CD_VARIABLE_USED_DMEM + rtl8192cd_dmem_free(PMIB, pmib); +#else + kfree(pmib); +#endif + } +#ifdef CONFIG_RTK_MESH + if(proxy_table) + { + remove_hash_table(proxy_table); + kfree(proxy_table); + } + if(mesh_rreq_retry_queue) + { + for (i=0; i< (1 << mesh_rreq_retry_queue->table_size_power); i++) + { + if(((struct mesh_rreq_retry_entry*)(mesh_rreq_retry_queue->entry_array[i].data))->ptr) + { + kfree(((struct mesh_rreq_retry_entry*)(mesh_rreq_retry_queue->entry_array[i].data))->ptr); + } + } + remove_hash_table(mesh_rreq_retry_queue); + kfree(mesh_rreq_retry_queue); + } + + // add by chuangch 2007.09.13 + if(pathsel_table) + { + remove_hash_table(pathsel_table); + kfree(pathsel_table); + } + + if(pann_mpp_tb) + kfree(pann_mpp_tb); + + if (pathsel_queue) + kfree(pathsel_queue); +#ifdef _11s_TEST_MODE_ + if (receiver_queue) + kfree(receiver_queue); +#endif +#endif // CONFIG_RTK_MESH + if (pevent_queue) + kfree(pevent_queue); +#ifdef CONFIG_RTL_WAPI_SUPPORT + if (wapiEvent_queue) + kfree(wapiEvent_queue); + #if defined(MBSSID) + if (wapiVapEvent_queue) + kfree(wapiVapEvent_queue); + #endif + +#endif +#ifndef PRIV_STA_BUF + if (phw) + kfree(phw); + if (pshare) // david + kfree(pshare); + if (pwlan_hdr_poll) + kfree(pwlan_hdr_poll); + if (pwlanllc_hdr_poll) + kfree(pwlanllc_hdr_poll); + if (pwlanbuf_poll) + kfree(pwlanbuf_poll); + if (pwlanicv_poll) + kfree(pwlanicv_poll); + if (pwlanmic_poll) + kfree(pwlanmic_poll); +#endif + if (pwlan_acl_poll) + kfree(pwlan_acl_poll); + +#if defined(CONFIG_RTK_MESH) && defined(_MESH_ACL_ENABLE_) // below code copy above ACL code + if (pmesh_acl_poll) + kfree(pmesh_acl_poll); +#endif + + if (Eap_packet) + kfree(Eap_packet); +#if defined(INCLUDE_WPA_PSK) || defined(WIFI_HAPD) + if (wpa_global_info) + kfree(wpa_global_info); +#endif + +#ifdef __LINUX_2_6__ + free_netdev(dev); +#else + kfree(dev); +#endif + wdev->priv = NULL; + printk("=====>>EXIT rtl8192cd_init_one2(%d) <<=====\n", rc); + return rc; +} + + +#ifdef CONFIG_RTL_STP +extern int rtl865x_wlanIF_Init(struct net_device *dev); +static int rtl_pseudo_dev_set_hwaddr(struct net_device *dev, void *addr) +{ + unsigned long flags; + int i; + unsigned char *p; + + p = ((struct sockaddr *)addr)->sa_data; + local_irq_save(flags); + for (i = 0; i<MACADDRLEN; ++i) { + dev->dev_addr[i] = p[i]; + } + local_irq_restore(flags); + return SUCCESS; +} + +#if defined(__LINUX_2_6__) && !defined(CONFIG_COMPAT_NET_DEV_OPS) +static const struct net_device_ops rtl8192cd_rtl_pseudodev_ops = { + .ndo_open = rtl8192cd_open, + .ndo_stop = rtl8192cd_close, + .ndo_set_mac_address = rtl_pseudo_dev_set_hwaddr, + .ndo_get_stats = rtl8192cd_get_stats, + .ndo_do_ioctl = rtl8192cd_ioctl, + .ndo_start_xmit = rtl8192cd_start_xmit, +}; +#endif + +void rtl_pseudo_dev_init(void* priv) +{ + struct net_device *dev; + +/* printk("[%s][%d] priv of %s\n", __FUNCTION__, __LINE__, ((struct rtl8192cd_priv*)priv)->dev->name);*/ + dev = alloc_etherdev(0); + if (dev == NULL) { + printk("alloc_etherdev() pseudo port5 error!\n"); + return; + } + +#if !defined(__LINUX_2_6__) || defined(CONFIG_COMPAT_NET_DEV_OPS) + dev->open = rtl8192cd_open; + dev->stop = rtl8192cd_close; + dev->hard_start_xmit = rtl8192cd_start_xmit; + dev->get_stats = rtl8192cd_get_stats; + dev->do_ioctl = rtl8192cd_ioctl; + dev->set_mac_address = rtl_pseudo_dev_set_hwaddr; +#else + dev->netdev_ops = &rtl8192cd_rtl_pseudodev_ops; +#endif + dev->priv = priv; + strcpy(dev->name, "port5"); + memcpy((char*)dev->dev_addr,"\x00\xe0\x4c\x81\x86\x86", MACADDRLEN); + if (register_netdev(dev)) { + printk(KERN_ERR "register_netdev() wds error!\n"); + } + rtl865x_wlanIF_Init(dev); +} +#endif + +#if defined(CONFIG_RTL_CUSTOM_PASSTHRU) +__DRAM_IN_865X int passThruStatusWlan; +int passThruWanIdx; +static struct proc_dir_entry *resPassThruWlan=NULL; +static char passThru_flag_wlan[10]; + +static int rtl_passthru_pseudo_dev_set_hwaddr(struct net_device *dev, void *addr) +{ + unsigned long flags; + int i; + unsigned char *p; + + p = ((struct sockaddr *)addr)->sa_data; + local_irq_save(flags); + for (i = 0; i<MACADDRLEN; ++i) { + dev->dev_addr[i] = p[i]; + } + local_irq_restore(flags); + return SUCCESS; +} + +#if defined(__LINUX_2_6__) && !defined(CONFIG_COMPAT_NET_DEV_OPS) +static const struct net_device_ops rtl8192cd_pseudodev_ops = { + .ndo_open = rtl8192cd_open, + .ndo_stop = rtl8192cd_close, + .ndo_set_mac_address = rtl_passthru_pseudo_dev_set_hwaddr, + .ndo_set_multicast_list = rtl8192cd_set_rx_mode, + .ndo_get_stats = rtl8192cd_get_stats, + .ndo_do_ioctl = rtl8192cd_ioctl, + .ndo_start_xmit = rtl8192cd_start_xmit, +}; +#endif + +void rtl_passthru_pseudo_dev_init(void *priv) +{ + struct net_device *dev; + struct rtl8192cd_priv *dp; + + dev = alloc_etherdev(sizeof(struct rtl8192cd_priv)); + if (dev == NULL) { + printk("alloc_etherdev() pseudo pwlan0 error!\n"); + return; + } + +#if !defined(__LINUX_2_6__) || defined(CONFIG_COMPAT_NET_DEV_OPS) + dev->open = rtl8192cd_open; + dev->stop = rtl8192cd_close; + dev->set_multicast_list = rtl8192cd_set_rx_mode; + dev->hard_start_xmit = rtl8192cd_start_xmit; + dev->get_stats = rtl8192cd_get_stats; + dev->do_ioctl = rtl8192cd_ioctl; + dev->set_mac_address = rtl_passthru_pseudo_dev_set_hwaddr; +#else + dev->netdev_ops = &rtl8192cd_pseudodev_ops; +#endif + +#ifndef NETDEV_NO_PRIV + dev->priv = priv; +#endif + + strcpy(dev->name, "pwlan%d"); + memcpy((char*)dev->dev_addr,"\x00\xe0\x4c\x81\x96\x96", MACADDRLEN); + if (register_netdev(dev)) { + printk(KERN_ERR "register_netdev() pwlan0 error!\n"); + } + + ((struct rtl8192cd_priv *)priv)->pWlanDev=dev; //pWlanDev point to the virtual pwlan0 +} + + +static unsigned long atoi_dec(char *s) +{ + unsigned long k = 0; + + k = 0; + while (*s != '\0' && *s >= '0' && *s <= '9') { + k = 10 * k + (*s - '0'); + s++; + } + return k; +} + +static int wlan_custom_Passthru_read_proc(char *page, char **start, off_t off, + int count, int *eof, void *data) +{ + int len; + len = sprintf(page, "%s\n", passThru_flag_wlan); + if (len <= off+count) + *eof = 1; + + *start = page + off; + len -= off; + + if (len>count) + len = count; + + if (len<0) len = 0; + + return len; +} + +static int wlan_custom_Passthru_write_proc(struct file *file, const char *buffer, + unsigned long count, void *data) +{ + int flag,i; + + if (buffer && !copy_from_user(&passThru_flag_wlan, buffer, count)) + { + flag=(int)atoi_dec(passThru_flag_wlan); + + passThruStatusWlan=flag; + + return count; + } + return -EFAULT; +} + +int rtl_wlan_customPassthru_init(void) +{ + int wlan_idx; + + for(wlan_idx=0;wlan_idx<sizeof(wlan_device)/sizeof(struct _device_info_);wlan_idx++) + { + if (wlan_device[wlan_idx].priv!=NULL) + break; + } + + if (wlan_idx==sizeof(wlan_device)/sizeof(struct _device_info_)) + return -EINVAL; + + passThruWanIdx = wlan_idx; + //initial + memset(passThru_flag_wlan,0, sizeof(passThru_flag_wlan)); + //strcpy(passThru_flag_wlan,"0"); + passThruStatusWlan=0; + + resPassThruWlan = create_proc_entry("custom_Passthru_wlan", 0, NULL); + if(resPassThruWlan) + { + resPassThruWlan->read_proc = wlan_custom_Passthru_read_proc; + resPassThruWlan->write_proc = wlan_custom_Passthru_write_proc; + } + + + rtl_passthru_pseudo_dev_init(wlan_device[passThruWanIdx].priv); + + return 0; +} + +void __exit rtl_wlan_customPassthru_exit(void) +{ + if (resPassThruWlan) { + remove_proc_entry("custom_Passthru_wlan", resPassThruWlan); + resPassThruWlan = NULL; + } +} +#endif + + + +#if defined(__DRAYTEK_OS__) && defined(WDS) +int rtl8192cd_add_wds(struct net_device *dev, struct net_device *wds_dev, unsigned char *addr) +{ + struct rtl8192cd_priv *priv = (struct rtl8192cd_priv *)dev->priv; + int wds_num=priv->pmib->dot11WdsInfo.wdsNum; + + priv->pmib->dot11WdsInfo.dev[wds_num] = wds_dev; + memcpy(priv->pmib->dot11WdsInfo.entry[wds_num].macAddr, addr, 6); + wds_dev->priv = priv; + wds_dev->base_addr = 0; + priv->pmib->dot11WdsInfo.wdsNum++; + + if (!priv->pmib->dot11WdsInfo.wdsEnabled) + priv->pmib->dot11WdsInfo.wdsEnabled = 1; + + if (netif_running(priv->dev)) + create_wds_tbl(priv); + + DEBUG_INFO("\r\nAdd WDS: %02x%02x %02x%02x%02x%02x\n", + addr[0],addr[1],addr[2],addr[3],addr[4],addr[5]); +} +#endif + + +#ifdef CONFIG_NET_PCI +static int MDL_DEVINIT rtl8192cd_init_pci(struct pci_dev *pdev, const struct pci_device_id *ent) +{ + int dev_num=0; + int ret; +#ifdef MBSSID + int i; +#endif + +#ifdef RTL8192CD_VARIABLE_USED_DMEM + /* For D-MEM allocation system's initialization : It would before ALL processes */ + rtl8192cd_dmem_init(); +#endif + + if(dev_num >= (sizeof(wlan_device)/sizeof(struct _device_info_))){ + printk("PCI device %d can't be support\n", dev_num); + return -1; + } + else { + + ret = rtl8192cd_init_one(pdev, ent, &wlan_device[dev_num++], -1); +#ifdef UNIVERSAL_REPEATER + if (ret == 0) { + ret = rtl8192cd_init_one(pdev, ent, &wlan_device[--dev_num], -1); + dev_num++; + } +#endif +#ifdef MBSSID + if (ret == 0) { + dev_num--; + for (i=0; i<RTL8192CD_NUM_VWLAN; i++) { + ret = rtl8192cd_init_one(pdev, ent, &wlan_device[dev_num], i); + if (ret != 0) { + printk("Init fail!\n"); + return ret; + } + } + dev_num++; + } +#endif + return ret; + } +} + + +static void MDL_DEVEXIT rtk_remove_one(struct pci_dev *pdev) +{ + struct net_device *dev = pci_get_drvdata(pdev); + +#ifdef NETDEV_NO_PRIV + struct rtl8192cd_priv *priv = ((struct rtl8192cd_priv *)netdev_priv(dev))->wlan_priv; +#else + struct rtl8192cd_priv *priv = dev->priv; +#endif + + if (!dev) + BUG(); + + iounmap((void *)priv->dev->base_addr); + pci_release_regions(pdev); + pci_disable_device(pdev); + pci_set_drvdata(pdev, NULL); +} + + +static struct pci_device_id MDL_DEVINITDATA rtl8192cd_pci_tbl[] = +{ +/* + { PCI_VENDOR_ID_REALTEK, 0x8190, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, 0 }, +*/ +#ifdef CONFIG_RTL_92D_SUPPORT + { PCI_VENDOR_ID_REALTEK, 0x8193, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, 0 }, +#endif + { PCI_VENDOR_ID_REALTEK, 0x8191, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, 0 }, + + { PCI_VENDOR_ID_REALTEK, 0x8171, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, 0 }, + + { PCI_VENDOR_ID_REALTEK, 0x8178, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, 0 }, + + { PCI_VENDOR_ID_REALTEK, 0x8176, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, 0 }, + + { PCI_VENDOR_ID_REALTEK, 0x8174, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, 0 }, + + { }, +}; + + +MODULE_DEVICE_TABLE(pci, rtl8192cd_pci_tbl); + +static struct pci_driver rtl8192cd_driver = { + name: DRV_NAME, + id_table: rtl8192cd_pci_tbl, + probe: rtl8192cd_init_pci, + remove: __devexit_p(rtk_remove_one), +}; +#endif // CONFIG_NET_PCI + + +#ifdef CONFIG_WIRELESS_LAN_MODULE +int GetCpuCanSuspend(void) +{ + extern int gCpuCanSuspend; + return gCpuCanSuspend; +} +#endif + + +#if defined(CONFIG_RTL8196B) +//System identification for CHIP +#define CHIP_OEM_ID 0xb8000000 +#define DDR_SELECT 0xb8000008 +#define C_CUT 2 +#define DDR_BOOT 2 +int no_ddr_patch; +#endif + + +int MDL_INIT __rtl8192cd_init(unsigned long base_addr) +{ +#ifdef CONFIG_NET_PCI + int pci_reg=0; +#endif + int rc; +#ifdef MBSSID + int i; +#endif + +#if defined(CONFIG_RTL8196B) + //System identification for CHIP + no_ddr_patch = !((REG32(CHIP_OEM_ID)<C_CUT)&(REG32(DDR_SELECT)&&DDR_BOOT)); +#endif + +#ifdef __KERNEL__ +#if defined(EAP_BY_QUEUE) && defined(USE_CHAR_DEV) +// for module, 2005-12-26 ----------- + extern struct rtl8192cd_priv* (*rtl8192cd_chr_reg_hook)(unsigned int minor, struct rtl8192cd_chr_priv *priv); + extern void (*rtl8192cd_chr_unreg_hook)(unsigned int minor); +//------------------------------------ +#endif +#ifdef CONFIG_WIRELESS_LAN_MODULE + wirelessnet_hook = GetCpuCanSuspend; +#ifdef BR_SHORTCUT + wirelessnet_hook_shortcut = get_shortcut_dev; +#endif +#ifdef PERF_DUMP + Fn_rtl8651_romeperfEnterPoint = rtl8651_romeperfEnterPoint; + Fn_rtl8651_romeperfExitPoint = rtl8651_romeperfExitPoint; +#endif +#ifdef CONFIG_RTL8190_PRIV_SKB + wirelessnet_hook_is_priv_buf = is_rtl8190_priv_buf; + wirelessnet_hook_free_priv_buf = free_rtl8190_priv_buf; +#endif +#endif // CONFIG_WIRELESS_LAN_MODULE +#endif // __KERNEL__ + +#ifndef GREEN_HILL +#ifdef CONFIG_RTL8671 + printk("%s driver version %d.%d.%d (%s)\n", DRV_NAME, DRV_VERSION_H, DRV_VERSION_L, DRV_VERSION_SUBL, DRV_RELDATE); +#else + panic_printk("%s driver version %d.%d (%s)\n", DRV_NAME, DRV_VERSION_H, DRV_VERSION_L, DRV_RELDATE); +#endif +#endif + + +#ifdef __KERNEL__ + for (wlan_index=0; wlan_index<sizeof(wlan_device)/sizeof(struct _device_info_); wlan_index++) +#else + if (wlan_index<sizeof(wlan_device)/sizeof(struct _device_info_)) +#endif + { + if (((wlan_device[wlan_index].type >> TYPE_SHIFT) & TYPE_MASK) == TYPE_PCI_BIOS) { +#ifdef CONFIG_NET_PCI + if(!pci_reg){ + pci_reg=1; +#ifdef LINUX_2_6_21_ + pci_register_driver(&rtl8192cd_driver); +#else + pci_module_init(&rtl8192cd_driver); +#endif + } +#endif + } + else { + +#ifdef __DRAYTEK_OS__ + wlan_device[wlan_index].base_addr = base_addr; + wlan_device[wlan_index].type = (TYPE_PCI_DIRECT<<TYPE_SHIFT); +#endif + rc = rtl8192cd_init_one(NULL, NULL, &wlan_device[wlan_index], -1); + + // victoryman debug + if (rc) + printk("init_one fail!!! rc=%d\n",rc); + +#ifdef UNIVERSAL_REPEATER + if (rc == 0) + rc = rtl8192cd_init_one(NULL, NULL, &wlan_device[wlan_index], -1); +#endif +#ifdef MBSSID + if (rc == 0) { + for (i=0; i<RTL8192CD_NUM_VWLAN; i++) { + rc = rtl8192cd_init_one(NULL, NULL, &wlan_device[wlan_index], i); + if (rc != 0) { + printk("Init fail! rc=%d\n", rc); + break; + } + } + } +#endif + } + +#if defined(__DRAYTEK_OS__) + if (rc != 0) + return rc; +#endif + +#ifndef __KERNEL__ + wlan_index++; +#endif + } + +#if defined(CONFIG_RTL_CUSTOM_PASSTHRU) + rtl_wlan_customPassthru_init(); +#endif + +#if 0 +//#ifdef PCIE_POWER_SAVING + HostPCIe_Close(); +#endif + + +#ifdef CONFIG_RTL_STP + rtl_pseudo_dev_init(wlan_device[0].priv); +#endif + +#ifdef _USE_DRAM_ + { + extern unsigned char *en_cipherstream; + extern unsigned char *tx_cipherstream; + extern char *rc4sbox, *rc4kbox; + extern unsigned char *pTkip_Sbox_Lower, *pTkip_Sbox_Upper; + extern unsigned char Tkip_Sbox_Lower[256], Tkip_Sbox_Upper[256]; + +#ifdef CONFIG_RTL8671 + extern void r3k_enable_DRAM(void); //6/7/04' hrchen, for 8671 DRAM init + r3k_enable_DRAM(); //6/7/04' hrchen, for 8671 DRAM init +#endif + + en_cipherstream = (unsigned char *)(DRAM_START_ADDR); + tx_cipherstream = en_cipherstream; + + rc4sbox = (char *)(DRAM_START_ADDR + 2048); + rc4kbox = (char *)(DRAM_START_ADDR + 2048 + 256); + pTkip_Sbox_Lower = (unsigned char *)(DRAM_START_ADDR + 2048 + 256*2); + pTkip_Sbox_Upper = (unsigned char *)(DRAM_START_ADDR + 2048 + 256*3); + + memcpy(pTkip_Sbox_Lower, Tkip_Sbox_Lower, 256); + memcpy(pTkip_Sbox_Upper, Tkip_Sbox_Upper, 256); + } +#endif + +#ifdef __KERNEL__ +#if defined(EAP_BY_QUEUE) && defined(USE_CHAR_DEV) +// for module, 2005-12-26 ----------- + rtl8192cd_chr_reg_hook = rtl8192cd_chr_reg; + rtl8192cd_chr_unreg_hook = rtl8192cd_chr_unreg; +//------------------------------------ + rtl8192cd_chr_init(); +#endif +#endif + +#ifdef CONFIG_RTL8671 + //turn off AP LED + { + unsigned char wlanreg = *(volatile unsigned char *)0xbd30005e; + *(volatile unsigned char *)0xbd30005e = (wlanreg | ((1<<5))); + } +#endif + +#ifdef PERF_DUMP + rtl8651_romeperfInit(); +#endif + +#ifdef USB_PKT_RATE_CTRL_SUPPORT + register_usb_hook = (register_usb_pkt_cnt_fn)(register_usb_pkt_cnt_f); +#endif + + return 0; +} + + +#ifdef __DRAYTEK_OS__ +int rtl8192cd_init(unsigned long base_addr) +{ + return __rtl8192cd_init(base_addr); +} +#else // not __DRAYTEK_OS__ +int MDL_INIT rtl8192cd_init(void) +{ +#ifdef CONFIG_RTL8671 + gpioConfig(10,2); + gpioClear(10); + delay_ms(10); + gpioSet(10); +#endif + + return __rtl8192cd_init(0); +} +#endif + + +#ifdef __KERNEL__ +static void MDL_EXIT rtl8192cd_exit (void) +{ + struct net_device *dev; + struct rtl8192cd_priv *priv; + int idx; +#if defined(WDS) || defined(MBSSID) + int i; +#endif + +#ifdef CONFIG_WIRELESS_LAN_MODULE + wirelessnet_hook = NULL; +#ifdef BR_SHORTCUT + wirelessnet_hook_shortcut = NULL; +#endif +#ifdef PERF_DUMP + Fn_rtl8651_romeperfEnterPoint = NULL; + Fn_rtl8651_romeperfExitPoint = NULL; + #endif +#ifdef CONFIG_RTL8190_PRIV_SKB + wirelessnet_hook_is_priv_buf = NULL; + wirelessnet_hook_free_priv_buf = NULL; +#endif +#endif // CONFIG_WIRELESS_LAN_MODULE + +#if defined(EAP_BY_QUEUE) && defined(USE_CHAR_DEV) +// for module, 2005-12-26 ------------ + extern struct rtl8192cd_priv* (*rtl8192cd_chr_reg_hook)(unsigned int minor, struct rtl8192cd_chr_priv *priv); + extern void (*rtl8192cd_chr_unreg_hook)(unsigned int minor); +//------------------------------------ +#endif + +#ifdef WDS + int num; + + for (idx=0; idx<sizeof(wlan_device)/sizeof(struct _device_info_); idx++) { + if (wlan_device[idx].priv) { + num = (wlan_device[idx].type >> WDS_SHIFT) & WDS_MASK; + for (i=0; i<num; i++) { + unregister_netdev(wlan_device[idx].priv->wds_dev[i]); +#ifndef NETDEV_NO_PRIV + wlan_device[idx].priv->wds_dev[i]->priv = NULL; +#endif +#ifdef __LINUX_2_6__ + free_netdev(wlan_device[idx].priv->wds_dev[i]); +#else + kfree(wlan_device[idx].priv->wds_dev[i]); +#endif + } + } + } +#endif + +#ifdef CONFIG_RTK_MESH +#ifndef WDS + int num; +#endif + + for (idx=0; idx<sizeof(wlan_device)/sizeof(struct _device_info_); idx++) { + num = (wlan_device[idx].type >> MESH_SHIFT) & MESH_MASK; + if(num > 0) { // num is always 0 or 1 in this time + // for (i=0; i<num; i++) { + if (wlan_device[idx].priv) { + wlan_device[idx].priv->mesh_dev->priv = NULL; + unregister_netdev(wlan_device[idx].priv->mesh_dev); + kfree(wlan_device[idx].priv->mesh_dev); + } + } // end of if(num > 0) + } + +#endif // CONFIG_RTK_MESH + + +#ifdef UNIVERSAL_REPEATER + for (idx=0; idx<sizeof(wlan_device)/sizeof(struct _device_info_); idx++) { + if (wlan_device[idx].priv) { + struct rtl8192cd_priv *vxd_priv = GET_VXD_PRIV(wlan_device[idx].priv); + if (vxd_priv) { + unregister_netdev(vxd_priv->dev); +#ifdef RTL8192CD_VARIABLE_USED_DMEM + rtl8192cd_dmem_free(PMIB, vxd_priv->pmib); +#else + kfree(vxd_priv->pmib); +#endif + kfree(vxd_priv->pevent_queue); +#ifdef CONFIG_RTL_WAPI_SUPPORT + kfree(vxd_priv->wapiEvent_queue); +#endif + kfree(vxd_priv->pwlan_acl_poll); + kfree(vxd_priv->Eap_packet); +#if defined(INCLUDE_WPA_PSK) || defined(WIFI_HAPD) + kfree(vxd_priv->wpa_global_info); +#endif +#ifdef __LINUX_2_6__ + free_netdev(vxd_priv->dev); +#else + kfree(vxd_priv->dev); +#endif + wlan_device[idx].priv->pvxd_priv = NULL; + } + } + } +#endif + +#ifdef MBSSID + for (idx=0; idx<sizeof(wlan_device)/sizeof(struct _device_info_); idx++) { + if (wlan_device[idx].priv) { + for (i=0; i<RTL8192CD_NUM_VWLAN; i++) { + struct rtl8192cd_priv *vap_priv = wlan_device[idx].priv->pvap_priv[i]; + if (vap_priv) { + unregister_netdev(vap_priv->dev); +#ifdef RTL8192CD_VARIABLE_USED_DMEM + rtl8192cd_dmem_free(PMIB, vap_priv->pmib); +#else + kfree(vap_priv->pmib); +#endif + kfree(vap_priv->pevent_queue); +#ifdef CONFIG_RTL_WAPI_SUPPORT + kfree(vap_priv->wapiEvent_queue); +#endif + kfree(vap_priv->pwlan_acl_poll); + kfree(vap_priv->Eap_packet); +#if defined(INCLUDE_WPA_PSK) || defined(WIFI_HAPD) + kfree(vap_priv->wpa_global_info); +#endif +#ifdef __LINUX_2_6__ + free_netdev(vap_priv->dev); +#else + kfree(vap_priv->dev); +#endif + wlan_device[idx].priv->pvap_priv[i] = NULL; + } + } + } + } +#endif + +#ifdef CONFIG_NET_PCI + pci_unregister_driver (&rtl8192cd_driver); +#endif + + for (idx=0; idx<sizeof(wlan_device)/sizeof(struct _device_info_); idx++) + { + if (wlan_device[idx].priv == NULL) + continue; + priv = wlan_device[idx].priv; + dev = priv->dev; + +#ifdef _INCLUDE_PROC_FS_ + rtl8192cd_proc_remove(dev); +#endif + + unregister_netdev(dev); + +#ifndef PRIV_STA_BUF +#ifdef CONFIG_NET_PCI + if (IS_PCIBIOS_TYPE) + pci_free_consistent(priv->pshare->pdev, DESC_DMA_PAGE_SIZE, (void *)priv->pshare->phw->ring_virt_addr, + priv->pshare->phw->alloc_dma_buf); + else +#endif + kfree((void *)priv->pshare->phw->alloc_dma_buf); +#endif + +#ifdef RTL8192CD_VARIABLE_USED_DMEM + rtl8192cd_dmem_free(PMIB, priv->pmib); +#else + kfree(priv->pmib); +#endif + kfree(priv->pevent_queue); +#ifdef CONFIG_RTL_WAPI_SUPPORT + kfree(priv->wapiEvent_queue); + #ifdef MBSSID + if (IS_ROOT_INTERFACE(priv)&&priv->pmib->miscEntry.vap_enable){ + for (i=0; i<RTL8192CD_NUM_VWLAN; i++) + kfree(priv->pvap_priv[i]->wapiEvent_queue); + } + #endif + +#endif + +#ifdef CONFIG_RTK_MESH + kfree(priv->pathsel_queue); +#ifdef _11s_TEST_MODE_ + kfree(priv->receiver_queue); + + for(i=0; i< AODV_RREQ_TABLE_SIZE; i++) + del_timer(&priv->pshare->galileo_poll->node[i].data.expire_timer); + + kfree(priv->pshare->galileo_poll); +#endif +#ifdef _MESH_ACL_ENABLE_ + kfree(priv->pmesh_acl_poll); +#endif +#endif + +#ifndef PRIV_STA_BUF + kfree(priv->pshare->phw); + kfree(priv->pshare->pwlan_hdr_poll); + kfree(priv->pshare->pwlanllc_hdr_poll); + kfree(priv->pshare->pwlanbuf_poll); + kfree(priv->pshare->pwlanicv_poll); + kfree(priv->pshare->pwlanmic_poll); +#endif + kfree(priv->pwlan_acl_poll); + kfree(priv->Eap_packet); +#if defined(INCLUDE_WPA_PSK) || defined(WIFI_HAPD) + kfree(priv->wpa_global_info); +#endif +#ifndef PRIV_STA_BUF + kfree(priv->pshare); // david +#endif +#ifdef __LINUX_2_6__ + free_netdev(dev); +#else + kfree(dev); +#endif + wlan_device[idx].priv = NULL; + } + +#if defined(EAP_BY_QUEUE) && defined(USE_CHAR_DEV) +// for module, 2005-12-26 ------------ + rtl8192cd_chr_reg_hook = NULL; + rtl8192cd_chr_unreg_hook = NULL; +//------------------------------------ + + rtl8192cd_chr_exit(); +#endif +} +#else // not __KERNEL__ +void MDL_EXIT rtl8192cd_exit(void *data) +{ + struct net_device *dev = (struct net_device *)data; + struct rtl8192cd_priv *priv = dev->priv; + int idx, i; + + for (idx=0; idx<sizeof(wlan_device)/sizeof(struct _device_info_); idx++) + if (wlan_device[idx].priv == priv) + break; + + if (idx == sizeof(wlan_device)/sizeof(struct _device_info_)) + return; // wrong argument!! + +#ifdef WDS + { + int num; + + num = (wlan_device[idx].type >> WDS_SHIFT) & WDS_MASK; + for (i=0; i<num; i++) { + wlan_device[idx].priv->pmib->dot11WdsInfo.dev[i]->priv = NULL; + unregister_netdev(wlan_device[idx].priv->pmib->dot11WdsInfo.dev[i]); + kfree(wlan_device[idx].priv->pmib->dot11WdsInfo.dev[i]); + } + } +#endif + + unregister_netdev(dev); + + kfree(priv->pmib); + +#ifdef CONFIG_RTK_MESH + + if(priv->proxy_table) + { + remove_hash_table(priv->proxy_table); + kfree(priv->proxy_table); + } + if(priv->mesh_rreq_retry_queue) + { + for (i = 0; i < (1 << priv->mesh_rreq_retry_queue->table_size_power); i++) { + if(((struct mesh_rreq_retry_entry*)(priv->mesh_rreq_retry_queue->entry_array[i].data))->ptr) + kfree(((struct mesh_rreq_retry_entry*)(priv->mesh_rreq_retry_queue->entry_array[i].data))->ptr); + } + remove_hash_table(priv->mesh_rreq_retry_queue); + kfree(priv->mesh_rreq_retry_queue); + } + + // add by chuangch 2007.09.13 + if(priv->pathsel_table) + { + remove_hash_table(priv->pathsel_table); + kfree(priv->pathsel_table); + } + + if(priv->pann_mpp_tb) + kfree(priv->pann_mpp_tb); + + kfree(priv->pathsel_queue); +#ifdef _11s_TEST_MODE_ + kfree(priv->receiver_queue); + for(i=0; i< AODV_RREQ_TABLE_SIZE; i++) + del_timer(&priv->pshare->galileo_poll->node[i].data.expire_timer); + + kfree(priv->pshare->galileo_poll); +#endif + +#ifdef _MESH_ACL_ENABLE_ + kfree(priv->pmesh_acl_poll); +#endif +#endif // CONFIG_RTK_MESH + + kfree(priv->pevent_queue); +#ifdef CONFIG_RTL_WAPI_SUPPORT +// kfree(vxd_priv->wapiEvent_queue); + kfree(priv->wapiEvent_queue); + #ifdef MBSSID + if (IS_ROOT_INTERFACE(priv)&&priv->pmib->miscEntry.vap_enable) { + for (i=0; i<RTL8192CD_NUM_VWLAN; i++) + kfree(priv->pvap_priv[i]->wapiEvent_queue); + } + #endif + +#endif +#ifndef PRIV_STA_BUF + kfree((void *)priv->pshare->phw->alloc_dma_buf); + + kfree(priv->pshare->phw); + kfree(priv->pshare->pwlan_hdr_poll); + kfree(priv->pshare->pwlanllc_hdr_poll); + kfree(priv->pshare->pwlanbuf_poll); + kfree(priv->pshare->pwlanicv_poll); + kfree(priv->pshare->pwlanmic_poll); +#endif + kfree(priv->pwlan_acl_poll); + kfree(priv->Eap_packet); +#if defined(INCLUDE_WPA_PSK) || defined(WIFI_HAPD) + kfree(priv->wpa_global_info); +#endif +#ifndef PRIV_STA_BUF + kfree(priv->pshare); // david +#endif + kfree(dev); + wlan_device[idx].priv = NULL; + + wlan_index--; +} +#endif + + +#ifdef __KERNEL__ +#ifdef USE_CHAR_DEV +struct rtl8192cd_priv *rtl8192cd_chr_reg(unsigned int minor, struct rtl8192cd_chr_priv *priv) +{ + if (wlan_device[minor].priv) + wlan_device[minor].priv->pshare->chr_priv = priv; + return wlan_device[minor].priv; +} + + +void rtl8192cd_chr_unreg(unsigned int minor) +{ + if (wlan_device[minor].priv) + wlan_device[minor].priv->pshare->chr_priv = NULL; +} +#endif + + +#ifdef RTL_WPA2_PREAUTH +void wpa2_kill_fasync(void) +{ + int wlan_index = 0; + struct _device_info_ *wdev = &wlan_device[wlan_index]; + struct rtl8192cd_priv *priv = wdev->priv; + event_indicate(priv, NULL, -1); +} + + +void wpa2_preauth_packet(struct sk_buff *pskb) +{ + // ****** NOTICE ********** + int wlan_index = 0; + struct _device_info_ *wdev = &wlan_device[wlan_index]; + // ****** NOTICE ********** + + struct rtl8192cd_priv *priv = wdev->priv; + + unsigned char szEAPOL[] = {0x02, 0x01, 0x00, 0x00}; + DOT11_EAPOL_START Eapol_Start; + + if (priv == NULL) { + PRINT_INFO("%s: priv == NULL\n", (char *)__FUNCTION__); + return; + } + +#ifndef WITHOUT_ENQUEUE + if (!memcmp(pskb->data, szEAPOL, sizeof(szEAPOL))) + { + Eapol_Start.EventId = DOT11_EVENT_EAPOLSTART_PREAUTH; + Eapol_Start.IsMoreEvent = FALSE; +#ifdef LINUX_2_6_22_ + memcpy(&Eapol_Start.MACAddr, pskb->mac_header + MACADDRLEN, WLAN_ETHHDR_LEN); +#else + memcpy(&Eapol_Start.MACAddr, pskb->mac.raw + MACADDRLEN, WLAN_ETHHDR_LEN); +#endif + DOT11_EnQueue((unsigned long)priv, priv->pevent_queue, (unsigned char*)&Eapol_Start, sizeof(DOT11_EAPOL_START)); + } + else + { + unsigned short pkt_len; + + pkt_len = WLAN_ETHHDR_LEN + pskb->len; + priv->Eap_packet->EventId = DOT11_EVENT_EAP_PACKET_PREAUTH; + priv->Eap_packet->IsMoreEvent = FALSE; + memcpy(&(priv->Eap_packet->packet_len), &pkt_len, sizeof(unsigned short)); +#ifdef LINUX_2_6_22_ + memcpy(&(priv->Eap_packet->packet[0]), pskb->mac_header, WLAN_ETHHDR_LEN); +#else + memcpy(&(priv->Eap_packet->packet[0]), pskb->mac.raw, WLAN_ETHHDR_LEN); +#endif + memcpy(&(priv->Eap_packet->packet[WLAN_ETHHDR_LEN]), pskb->data, pskb->len); + DOT11_EnQueue((unsigned long)priv, priv->pevent_queue, (unsigned char*)priv->Eap_packet, sizeof(DOT11_EAP_PACKET)); + } +#endif // WITHOUT_ENQUEUE + + event_indicate(priv, NULL, -1); + + // let dsr to free this skb +} +#endif // RTL_WPA2_PREAUTH +#endif // __KERNEL__ + +#if defined(CONFIG_RTL_ETH_PRIV_SKB_DEBUG) +extern int is_rtl865x_eth_priv_buf(unsigned char *head); +int get_nic_buf_in_wireless_tx(const char *name) +{ + int i,txCnt; + int queueCnt,idx; + struct rtl8192cd_priv *priv; + struct tx_desc_info *pdescinfoH,*pdescinfo; + struct tx_desc *pdescH, *pdesc; + struct sk_buff *skb = NULL; + struct rtl8192cd_hw *phw; + + txCnt = 0; + for (i=0; (i<sizeof(wlan_device)/sizeof(struct _device_info_)); i++) + { + //if(counted) + //break; + + if (wlan_device[i].priv && netif_running(wlan_device[i].priv->dev) && strcmp(wlan_device[i].priv->dev->name,name)==0) + { + priv = wlan_device[i].priv; + if (OPMODE & WIFI_AP_STATE) + { + for(queueCnt = 0; queueCnt <= HIGH_QUEUE;queueCnt++) + { + phw=GET_HW(priv); + pdescH = get_txdesc(phw, queueCnt); + pdescinfoH = get_txdesc_info(priv->pshare->pdesc_info,queueCnt); + for(idx = 0; idx < NUM_TX_DESC; idx++) + { + pdesc = pdescH + idx; + pdescinfo = pdescinfoH + idx; + //if (!pdesc || (get_desc(pdesc->Dword0) & TX_OWN)) + //continue; + if(pdescinfo->type == _SKB_FRAME_TYPE_ || pdescinfo->type == _RESERVED_FRAME_TYPE_) + skb = (struct sk_buff *)(pdescinfo->pframe); + else + continue; + + if(skb && is_rtl865x_eth_priv_buf(skb->head)) + txCnt++; + } + } + + //counted = 1; + } + } + } + + return txCnt; +} +#endif + + +#if defined(CONFIG_RTK_VLAN_SUPPORT) && defined(CONFIG_RTK_VLAN_FOR_CABLE_MODEM) +struct net_device* get_dev_by_vid(int vid) +{ + int i; + struct rtl8192cd_priv *priv; + + for (i=0; (i<sizeof(wlan_device)/sizeof(struct _device_info_)); i++) + { + if (wlan_device[i].priv && netif_running(wlan_device[i].priv->dev)) + { + priv = wlan_device[i].priv; + if ((OPMODE & WIFI_AP_STATE) && priv->pmib->miscEntry.vap_enable) { + int j; + for (j=0; j<RTL8192CD_NUM_VWLAN; j++) { + if (IS_DRV_OPEN(priv->pvap_priv[j])) { + if(priv->pvap_priv[j]->pmib->vlan.vlan_enable && priv->pvap_priv[j]->pmib->vlan.vlan_id == vid) + return priv->pvap_priv[j]->dev; + } + } + } + } + } + + return NULL; +} +#endif + +__MIPS16 +__IRAM_IN_865X +struct net_device *get_shortcut_dev(unsigned char *da) +{ + int i,j; + struct rtl8192cd_priv *priv; + struct stat_info *pstat; + struct net_device *dev; + +#if defined(BR_SHORTCUT) + #ifdef CONFIG_RTK_MESH //11 mesh no support shortcut now + { + extern unsigned char cached_mesh_mac[MACADDRLEN]; + extern struct net_device *cached_mesh_dev; + if (cached_mesh_dev && !memcmp(da, cached_mesh_mac, MACADDRLEN)) + return cached_mesh_dev; + } + #endif + + #ifdef WDS + { + extern unsigned char cached_wds_mac[MACADDRLEN]; + extern struct net_device *cached_wds_dev; + if (cached_wds_dev && !memcmp(da, cached_wds_mac, MACADDRLEN)) + return cached_wds_dev; + } + #endif + + #ifdef CLIENT_MODE + { + extern unsigned char cached_sta_mac[MACADDRLEN]; + extern struct net_device *cached_sta_dev; + if (cached_sta_dev && !memcmp(da, cached_sta_mac, MACADDRLEN)) + return cached_sta_dev; + } + #endif + + #if defined(RTL_CACHED_BR_STA) + { + extern unsigned char cached_br_sta_mac[MACADDRLEN]; + extern struct net_device *cached_br_sta_dev; + if ((cached_br_sta_dev!=NULL)&&!memcmp(da, cached_br_sta_mac, MACADDRLEN)) + return cached_br_sta_dev; + } + #endif +#endif + + dev = NULL; + for (i=0; (i<sizeof(wlan_device)/sizeof(struct _device_info_)) && (dev==NULL); i++) { + if (wlan_device[i].priv && netif_running(wlan_device[i].priv->dev)) { + priv = wlan_device[i].priv; + + if (priv + //2010-5-10 + #ifndef _SINUX_ + // if sinux, no linux bridge, so should don't depend on br_port if use br_shortcut (John Qian 2010/6/24) + && (priv->dev->br_port) + && !(priv->dev->br_port->br->stp_enabled) + #endif + ) { + if (!priv->pmib->dot11OperationEntry.disable_brsc) { + pstat = get_stainfo(priv, da); + if (pstat && pstat->tx_pkts > 10) { /* Make sure it must have some packets go theough bridge module before shortcut */ + #ifdef WDS + if (!(pstat->state & WIFI_WDS)) // if WDS peer + #endif + { + #ifdef CONFIG_RTK_MESH + if( isMeshPoint(pstat)) + {dev = priv->mesh_dev;} + else + #endif + {dev = priv->dev;} + break; + } + } + } + #ifdef MBSSID + else if ((OPMODE & WIFI_AP_STATE) && priv->pmib->miscEntry.vap_enable) { + for (j=0; j<RTL8192CD_NUM_VWLAN; j++) { + if ((priv->pvap_priv[j]->assoc_num > 0) && IS_DRV_OPEN(priv->pvap_priv[j]) && + !(priv->pvap_priv[j]->pmib->dot11OperationEntry.disable_brsc)) { + pstat = get_stainfo(priv->pvap_priv[j], da); + if (pstat && pstat->tx_pkts > 10) { + dev = priv->pvap_priv[j]->dev; + break; + } + } + } + } + #endif + } + } + } + +#if defined(RTL_CACHED_BR_STA) && defined(BR_SHORTCUT) + if (dev!=NULL) { + memcpy(cached_br_sta_mac, da, MACADDRLEN); + cached_br_sta_dev = dev; + } +#endif + + return dev; +} + +#if defined(BR_SHORTCUT) +void clear_shortcut_cache(void) +{ + extern unsigned char cached_eth_addr[MACADDRLEN]; + extern struct net_device *cached_dev; +#ifdef CLIENT_MODE + extern struct net_device *cached_sta_dev; + extern unsigned char cached_sta_mac[MACADDRLEN]; +#endif +#ifdef CONFIG_RTK_MESH + extern struct net_device *cached_mesh_dev; + extern unsigned char cached_mesh_mac[MACADDRLEN]; + cached_mesh_dev = NULL; + memset(cached_mesh_mac,0,MACADDRLEN); +#endif + +#ifdef WDS + extern struct net_device *cached_wds_dev; + extern unsigned char cached_wds_mac[MACADDRLEN]; + cached_wds_dev= NULL; + memset(cached_wds_mac,0,MACADDRLEN); +#endif + +#ifdef CLIENT_MODE + cached_sta_dev = NULL; + memset(cached_sta_mac,0,MACADDRLEN); +#endif + +#if defined(RTL_CACHED_BR_STA) + extern struct net_device *cached_br_sta_dev; + extern unsigned char cached_br_sta_mac[MACADDRLEN]; + cached_br_sta_dev = NULL; + memset(cached_br_sta_mac,0,MACADDRLEN); +#endif + +#if defined(CONFIG_RTL_819X) && defined(__LINUX_2_6__) + cached_dev = NULL; + memset(cached_eth_addr,0,MACADDRLEN); +#endif + +#ifdef CONFIG_RTL8672 + extern void clear_cached_eth_mac_addr(void); + clear_cached_eth_mac_addr(); +#endif +} +#endif // BR_SHORTCUT + + +void update_fwtbl_asoclst(struct rtl8192cd_priv *priv, struct stat_info *pstat) +{ + unsigned char tmpbuf[16]; + int i; + +#ifdef __KERNEL__ + struct sk_buff *skb = NULL; + struct wlan_ethhdr_t *e_hdr; + unsigned char xid_cmd[] = {0, 0, 0xaf, 0x81, 1, 2}; + + // update forwarding table of bridge module + if (priv->dev->br_port) { + skb = dev_alloc_skb(64); + if (skb != NULL) { + skb->dev = priv->dev; + skb_put(skb, 60); + e_hdr = (struct wlan_ethhdr_t *)skb->data; + memset(e_hdr, 0, 64); + memcpy(e_hdr->daddr, priv->dev->dev_addr, MACADDRLEN); + memcpy(e_hdr->saddr, pstat->hwaddr, MACADDRLEN); + e_hdr->type = 8; + memcpy(&skb->data[14], xid_cmd, sizeof(xid_cmd)); + skb->protocol = eth_type_trans(skb, priv->dev); + #if defined(__LINUX_2_6__) && defined(RX_TASKLET) + netif_receive_skb(skb); + #else + netif_rx(skb); + #endif + } + } +#endif + + // update association lists of the other WLAN interfaces + for (i=0; i<sizeof(wlan_device)/sizeof(struct _device_info_); i++) { + if (wlan_device[i].priv && (wlan_device[i].priv != priv)) { + if (wlan_device[i].priv->pmib->dot11OperationEntry.opmode & WIFI_AP_STATE) { + sprintf((char *)tmpbuf, "%02x%02x%02x%02x%02x%02x", + pstat->hwaddr[0],pstat->hwaddr[1],pstat->hwaddr[2],pstat->hwaddr[3],pstat->hwaddr[4],pstat->hwaddr[5]); + del_sta(wlan_device[i].priv, tmpbuf); + } + } + } + +#ifdef MBSSID + if (GET_ROOT(priv)->pmib->miscEntry.vap_enable) { + for (i=0; i<RTL8192CD_NUM_VWLAN; i++) { + if (priv->pvap_priv[i] && IS_DRV_OPEN(priv->pvap_priv[i]) && (priv->pvap_priv[i] != priv) && + (priv->vap_init_seq >= 0)) { + if (priv->pvap_priv[i]->pmib->dot11OperationEntry.opmode & WIFI_AP_STATE) { + sprintf(tmpbuf, "%02x%02x%02x%02x%02x%02x", + pstat->hwaddr[0],pstat->hwaddr[1],pstat->hwaddr[2],pstat->hwaddr[3],pstat->hwaddr[4],pstat->hwaddr[5]); + del_sta(priv->pvap_priv[i], tmpbuf); + } + } + } + } +#endif +} + + +// quick fix for warn reboot fail issue +void force_stop_wlan_hw(void) +{ + int i=0; + int temp; + +#if defined(CONFIG_RTL_DUAL_PCIESLOT_BIWLAN) || defined(CONFIG_RTL_DUAL_PCIESLOT_BIWLAN_D) + for (i=0; i<sizeof(wlan_device)/sizeof(struct _device_info_); i++) +#endif + { + if (wlan_device[i].priv) { + struct rtl8192cd_priv *priv = wlan_device[i].priv; +#ifdef PCIE_POWER_SAVING + if ((REG32(CLK_MANAGE) & BIT(11)) == 0) + { + extern void setBaseAddressRegister(void); + REG32(CLK_MANAGE) |= BIT(11); + delay_ms(10); + PCIE_reset_procedure(0, 0, 1, wlan_device[i].base_addr); + setBaseAddressRegister(); + } +#endif + if ( CheckNoResetHwExceptionCase(priv) ) { + rtl8192cd_stop_hw(priv); + } +#if 0 // defined(CONFIG_RTL_92D_SUPPORT) + #if defined(CONFIG_RTL_8198) + REG32(0xb8003528)=0xb000803; + mdelay(100); + #if (RTL_USED_PCIE_SLOT==1) + REG32(0xb8b21008)=1; + mdelay(100); + REG32(0xb8b21008)=0x81; + #else + REG32(0xb8b01008)=1; + mdelay(100); + REG32(0xb8b01008)=0x81; + #endif + mdelay(100); + REG32(0xb8003528)=0xb000803; + mdelay(100); + #if (RTL_USED_PCIE_SLOT==1) + temp=REG32(0xb8b30000); + #else + temp=REG32(0xb8b10000); + #endif + mdelay(100); + temp=REG32(0xb8000054); + mdelay(100); + REG32(0xb8000054)=8; + mdelay(100); + REG32(0xb8000054)=9; + mdelay(100); + REG32(0xb8000054)=0xa; + mdelay(100); + #if (RTL_USED_PCIE_SLOT==1) + temp=REG32(0xb8b30000); + #else + temp=REG32(0xb8b10000); + #endif + mdelay(100); + #endif +#endif +#if 0 +//#ifdef PCIE_POWER_SAVING + HostPCIe_Close(); +#endif + } + } +} + + +#ifdef CONFIG_RTL8671 +/*6/7/04 hrchen, invalidate the dcache with a 0->1 transition*/ + +#ifdef CONFIG_CPU_RLX4181 +int r3k_flush_dcache_range(int a, int b) +{ + int garbage_tmp; + __asm__ __volatile__( + ".set\tnoreorder\n\t" + ".set\tnoat\n\t" + "mfc0 %0, $20\n\t" + "nop\n\t" + "nop\n\t" + "andi %0, 0xFDFF\n\t" + "mtc0 %0, $20\n\t" + "nop\n\t" + "nop\n\t" + "ori %0, 0x200\n\t" + "mtc0 %0, $20\n\t" + "nop\n\t" + "nop\n\t" + ".set\tat\n\t" + ".set\treorder" + : "=r" (garbage_tmp)); + return 0; +} +#else +int r3k_flush_dcache_range(int a, int b) +{ + int garbage_tmp; + __asm__ __volatile__( + ".set\tnoreorder\n\t" + ".set\tnoat\n\t" + "mfc0 %0, $20\n\t" + "nop\n\t" + "nop\n\t" + "andi %0, 0xFFFE\n\t" + "mtc0 %0, $20\n\t" + "nop\n\t" + "nop\n\t" + "ori %0, 1\n\t" + "mtc0 %0, $20\n\t" + "nop\n\t" + "nop\n\t" + ".set\tat\n\t" + ".set\treorder" + : "=r" (garbage_tmp)); + return 0; +} +#endif + +#ifdef _USE_DRAM_ +//init DRAM +void r3k_enable_DRAM(void) +{ + int tmp, tmp1; + //--- initialize and start COP3 + __asm__ __volatile__( + ".set\tnoreorder\n\t" + ".set\tnoat\n\t" + "mfc0 %0,$12\n\t" + "nop\n\t" + "lui %1,0x8000\n\t" + "or %1,%0\n\t" + "mtc0 %1,$12\n\t" + "nop\n\t" + "nop\n\t" + ".set\tat\n\t" + ".set\treorder" + : "=r" (tmp), "=r" (tmp1)); + + //set base + __asm__ __volatile__( + ".set\tnoreorder\n\t" + ".set\tnoat\n\t" + "mtc3 %0, $4 # $4: d-ram base\n\t" + "nop\n\t" + "nop\n\t" + ".set\tat\n\t" + ".set\treorder" + : + : "r" (DRAM_START_ADDR&0x0fffffff)); + + //set size + __asm__ __volatile__( + ".set\tnoreorder\n\t" + ".set\tnoat\n\t" + "mtc3 %0, $5 # $5: d-ram top\n\t" + "nop\n\t" + "nop\n\t" + ".set\tat\n\t" + ".set\treorder" + : + : "r" (R3K_DRAM_SIZE-1)); + + //clear DRAM + __asm__ __volatile__( +"1:\n\t" + "sw $0,0(%1)\n\t" + "addiu %1,4\n\t" + "bne %0,%1,1b\n\t" + "nop\n\t" + ".set\tat\n\t" + ".set\treorder" + : + : "r" (DRAM_START_ADDR+R3K_DRAM_SIZE), "r" (DRAM_START_ADDR)); +} +#endif // _USE_DRAM_ +#endif // CONFIG_RTL8671 + + +#ifdef __KERNEL__ +#if defined(CONFIG_WIRELESS_LAN_MODULE) +MODULE_LICENSE("GPL"); +#endif +module_init(rtl8192cd_init); +module_exit(rtl8192cd_exit); +#endif + |