diff options
Diffstat (limited to 'drivers/net/wireless/bcmdhd/dhd_custom_gpio.c')
| -rw-r--r-- | drivers/net/wireless/bcmdhd/dhd_custom_gpio.c | 293 |
1 files changed, 293 insertions, 0 deletions
diff --git a/drivers/net/wireless/bcmdhd/dhd_custom_gpio.c b/drivers/net/wireless/bcmdhd/dhd_custom_gpio.c new file mode 100644 index 00000000000..9750eeb23bc --- /dev/null +++ b/drivers/net/wireless/bcmdhd/dhd_custom_gpio.c | |||
| @@ -0,0 +1,293 @@ | |||
| 1 | /* | ||
| 2 | * Customer code to add GPIO control during WLAN start/stop | ||
| 3 | * Copyright (C) 1999-2011, Broadcom Corporation | ||
| 4 | * | ||
| 5 | * Unless you and Broadcom execute a separate written software license | ||
| 6 | * agreement governing use of this software, this software is licensed to you | ||
| 7 | * under the terms of the GNU General Public License version 2 (the "GPL"), | ||
| 8 | * available at http://www.broadcom.com/licenses/GPLv2.php, with the | ||
| 9 | * following added to such license: | ||
| 10 | * | ||
| 11 | * As a special exception, the copyright holders of this software give you | ||
| 12 | * permission to link this software with independent modules, and to copy and | ||
| 13 | * distribute the resulting executable under terms of your choice, provided that | ||
| 14 | * you also meet, for each linked independent module, the terms and conditions of | ||
| 15 | * the license of that module. An independent module is a module which is not | ||
| 16 | * derived from this software. The special exception does not apply to any | ||
| 17 | * modifications of the software. | ||
| 18 | * | ||
| 19 | * Notwithstanding the above, under no circumstances may you combine this | ||
| 20 | * software in any way with any other Broadcom software provided under a license | ||
| 21 | * other than the GPL, without Broadcom's express prior written consent. | ||
| 22 | * | ||
| 23 | * $Id: dhd_custom_gpio.c,v 1.2.42.1 2010-10-19 00:41:09 Exp $ | ||
| 24 | */ | ||
| 25 | |||
| 26 | #include <typedefs.h> | ||
| 27 | #include <linuxver.h> | ||
| 28 | #include <osl.h> | ||
| 29 | #include <bcmutils.h> | ||
| 30 | |||
| 31 | #include <dngl_stats.h> | ||
| 32 | #include <dhd.h> | ||
| 33 | |||
| 34 | #include <wlioctl.h> | ||
| 35 | #include <wl_iw.h> | ||
| 36 | |||
| 37 | #define WL_ERROR(x) printf x | ||
| 38 | #define WL_TRACE(x) | ||
| 39 | |||
| 40 | #ifdef CUSTOMER_HW | ||
| 41 | extern void bcm_wlan_power_off(int); | ||
| 42 | extern void bcm_wlan_power_on(int); | ||
| 43 | #endif /* CUSTOMER_HW */ | ||
| 44 | #if defined(CUSTOMER_HW2) | ||
| 45 | #ifdef CONFIG_WIFI_CONTROL_FUNC | ||
| 46 | int wifi_set_power(int on, unsigned long msec); | ||
| 47 | int wifi_get_irq_number(unsigned long *irq_flags_ptr); | ||
| 48 | int wifi_get_mac_addr(unsigned char *buf); | ||
| 49 | void *wifi_get_country_code(char *ccode); | ||
| 50 | #else | ||
| 51 | int wifi_set_power(int on, unsigned long msec) { return -1; } | ||
| 52 | int wifi_get_irq_number(unsigned long *irq_flags_ptr) { return -1; } | ||
| 53 | int wifi_get_mac_addr(unsigned char *buf) { return -1; } | ||
| 54 | void *wifi_get_country_code(char *ccode) { return NULL; } | ||
| 55 | #endif /* CONFIG_WIFI_CONTROL_FUNC */ | ||
| 56 | #endif /* CUSTOMER_HW2 */ | ||
| 57 | |||
| 58 | #if defined(OOB_INTR_ONLY) | ||
| 59 | |||
| 60 | #if defined(BCMLXSDMMC) | ||
| 61 | extern int sdioh_mmc_irq(int irq); | ||
| 62 | #endif /* (BCMLXSDMMC) */ | ||
| 63 | |||
| 64 | #ifdef CUSTOMER_HW3 | ||
| 65 | #include <mach/gpio.h> | ||
| 66 | #endif | ||
| 67 | |||
| 68 | /* Customer specific Host GPIO defintion */ | ||
| 69 | static int dhd_oob_gpio_num = -1; | ||
| 70 | |||
| 71 | module_param(dhd_oob_gpio_num, int, 0644); | ||
| 72 | MODULE_PARM_DESC(dhd_oob_gpio_num, "DHD oob gpio number"); | ||
| 73 | |||
| 74 | /* This function will return: | ||
| 75 | * 1) return : Host gpio interrupt number per customer platform | ||
| 76 | * 2) irq_flags_ptr : Type of Host interrupt as Level or Edge | ||
| 77 | * | ||
| 78 | * NOTE : | ||
| 79 | * Customer should check his platform definitions | ||
| 80 | * and his Host Interrupt spec | ||
| 81 | * to figure out the proper setting for his platform. | ||
| 82 | * Broadcom provides just reference settings as example. | ||
| 83 | * | ||
| 84 | */ | ||
| 85 | int dhd_customer_oob_irq_map(unsigned long *irq_flags_ptr) | ||
| 86 | { | ||
| 87 | int host_oob_irq = 0; | ||
| 88 | |||
| 89 | #ifdef CUSTOMER_HW2 | ||
| 90 | host_oob_irq = wifi_get_irq_number(irq_flags_ptr); | ||
| 91 | |||
| 92 | #else | ||
| 93 | #if defined(CUSTOM_OOB_GPIO_NUM) | ||
| 94 | if (dhd_oob_gpio_num < 0) { | ||
| 95 | dhd_oob_gpio_num = CUSTOM_OOB_GPIO_NUM; | ||
| 96 | } | ||
| 97 | #endif /* CUSTOMER_HW2 */ | ||
| 98 | |||
| 99 | if (dhd_oob_gpio_num < 0) { | ||
| 100 | WL_ERROR(("%s: ERROR customer specific Host GPIO is NOT defined \n", | ||
| 101 | __FUNCTION__)); | ||
| 102 | return (dhd_oob_gpio_num); | ||
| 103 | } | ||
| 104 | |||
| 105 | WL_ERROR(("%s: customer specific Host GPIO number is (%d)\n", | ||
| 106 | __FUNCTION__, dhd_oob_gpio_num)); | ||
| 107 | |||
| 108 | #if defined CUSTOMER_HW | ||
| 109 | host_oob_irq = MSM_GPIO_TO_INT(dhd_oob_gpio_num); | ||
| 110 | #elif defined CUSTOMER_HW3 | ||
| 111 | gpio_request(dhd_oob_gpio_num, "oob irq"); | ||
| 112 | host_oob_irq = gpio_to_irq(dhd_oob_gpio_num); | ||
| 113 | gpio_direction_input(dhd_oob_gpio_num); | ||
| 114 | #endif /* CUSTOMER_HW */ | ||
| 115 | #endif /* CUSTOMER_HW2 */ | ||
| 116 | |||
| 117 | return (host_oob_irq); | ||
| 118 | } | ||
| 119 | #endif /* defined(OOB_INTR_ONLY) */ | ||
| 120 | |||
| 121 | /* Customer function to control hw specific wlan gpios */ | ||
| 122 | void | ||
| 123 | dhd_customer_gpio_wlan_ctrl(int onoff) | ||
| 124 | { | ||
| 125 | switch (onoff) { | ||
| 126 | case WLAN_RESET_OFF: | ||
| 127 | WL_TRACE(("%s: call customer specific GPIO to insert WLAN RESET\n", | ||
| 128 | __FUNCTION__)); | ||
| 129 | #ifdef CUSTOMER_HW | ||
| 130 | bcm_wlan_power_off(2); | ||
| 131 | #endif /* CUSTOMER_HW */ | ||
| 132 | #ifdef CUSTOMER_HW2 | ||
| 133 | wifi_set_power(0, 0); | ||
| 134 | #endif | ||
| 135 | WL_ERROR(("=========== WLAN placed in RESET ========\n")); | ||
| 136 | break; | ||
| 137 | |||
| 138 | case WLAN_RESET_ON: | ||
| 139 | WL_TRACE(("%s: callc customer specific GPIO to remove WLAN RESET\n", | ||
| 140 | __FUNCTION__)); | ||
| 141 | #ifdef CUSTOMER_HW | ||
| 142 | bcm_wlan_power_on(2); | ||
| 143 | #endif /* CUSTOMER_HW */ | ||
| 144 | #ifdef CUSTOMER_HW2 | ||
| 145 | wifi_set_power(1, 0); | ||
| 146 | #endif | ||
| 147 | WL_ERROR(("=========== WLAN going back to live ========\n")); | ||
| 148 | break; | ||
| 149 | |||
| 150 | case WLAN_POWER_OFF: | ||
| 151 | WL_TRACE(("%s: call customer specific GPIO to turn off WL_REG_ON\n", | ||
| 152 | __FUNCTION__)); | ||
| 153 | #ifdef CUSTOMER_HW | ||
| 154 | bcm_wlan_power_off(1); | ||
| 155 | #endif /* CUSTOMER_HW */ | ||
| 156 | break; | ||
| 157 | |||
| 158 | case WLAN_POWER_ON: | ||
| 159 | WL_TRACE(("%s: call customer specific GPIO to turn on WL_REG_ON\n", | ||
| 160 | __FUNCTION__)); | ||
| 161 | #ifdef CUSTOMER_HW | ||
| 162 | bcm_wlan_power_on(1); | ||
| 163 | /* Lets customer power to get stable */ | ||
| 164 | OSL_DELAY(200); | ||
| 165 | #endif /* CUSTOMER_HW */ | ||
| 166 | break; | ||
| 167 | } | ||
| 168 | } | ||
| 169 | |||
| 170 | #ifdef GET_CUSTOM_MAC_ENABLE | ||
| 171 | /* Function to get custom MAC address */ | ||
| 172 | int | ||
| 173 | dhd_custom_get_mac_address(unsigned char *buf) | ||
| 174 | { | ||
| 175 | int ret = 0; | ||
| 176 | |||
| 177 | WL_TRACE(("%s Enter\n", __FUNCTION__)); | ||
| 178 | if (!buf) | ||
| 179 | return -EINVAL; | ||
| 180 | |||
| 181 | /* Customer access to MAC address stored outside of DHD driver */ | ||
| 182 | #if defined(CUSTOMER_HW2) && (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 35)) | ||
| 183 | ret = wifi_get_mac_addr(buf); | ||
| 184 | #endif | ||
| 185 | |||
| 186 | #ifdef EXAMPLE_GET_MAC | ||
| 187 | /* EXAMPLE code */ | ||
| 188 | { | ||
| 189 | struct ether_addr ea_example = {{0x00, 0x11, 0x22, 0x33, 0x44, 0xFF}}; | ||
| 190 | bcopy((char *)&ea_example, buf, sizeof(struct ether_addr)); | ||
| 191 | } | ||
| 192 | #endif /* EXAMPLE_GET_MAC */ | ||
| 193 | |||
| 194 | return ret; | ||
| 195 | } | ||
| 196 | #endif /* GET_CUSTOM_MAC_ENABLE */ | ||
| 197 | |||
| 198 | /* Customized Locale table : OPTIONAL feature */ | ||
| 199 | const struct cntry_locales_custom translate_custom_table[] = { | ||
| 200 | /* Table should be filled out based on custom platform regulatory requirement */ | ||
| 201 | #ifdef EXAMPLE_TABLE | ||
| 202 | {"", "XY", 4}, /* Universal if Country code is unknown or empty */ | ||
| 203 | {"US", "US", 69}, /* input ISO "US" to : US regrev 69 */ | ||
| 204 | {"CA", "US", 69}, /* input ISO "CA" to : US regrev 69 */ | ||
| 205 | {"EU", "EU", 5}, /* European union countries to : EU regrev 05 */ | ||
| 206 | {"AT", "EU", 5}, | ||
| 207 | {"BE", "EU", 5}, | ||
| 208 | {"BG", "EU", 5}, | ||
| 209 | {"CY", "EU", 5}, | ||
| 210 | {"CZ", "EU", 5}, | ||
| 211 | {"DK", "EU", 5}, | ||
| 212 | {"EE", "EU", 5}, | ||
| 213 | {"FI", "EU", 5}, | ||
| 214 | {"FR", "EU", 5}, | ||
| 215 | {"DE", "EU", 5}, | ||
| 216 | {"GR", "EU", 5}, | ||
| 217 | {"HU", "EU", 5}, | ||
| 218 | {"IE", "EU", 5}, | ||
| 219 | {"IT", "EU", 5}, | ||
| 220 | {"LV", "EU", 5}, | ||
| 221 | {"LI", "EU", 5}, | ||
| 222 | {"LT", "EU", 5}, | ||
| 223 | {"LU", "EU", 5}, | ||
| 224 | {"MT", "EU", 5}, | ||
| 225 | {"NL", "EU", 5}, | ||
| 226 | {"PL", "EU", 5}, | ||
| 227 | {"PT", "EU", 5}, | ||
| 228 | {"RO", "EU", 5}, | ||
| 229 | {"SK", "EU", 5}, | ||
| 230 | {"SI", "EU", 5}, | ||
| 231 | {"ES", "EU", 5}, | ||
| 232 | {"SE", "EU", 5}, | ||
| 233 | {"GB", "EU", 5}, | ||
| 234 | {"KR", "XY", 3}, | ||
| 235 | {"AU", "XY", 3}, | ||
| 236 | {"CN", "XY", 3}, /* input ISO "CN" to : XY regrev 03 */ | ||
| 237 | {"TW", "XY", 3}, | ||
| 238 | {"AR", "XY", 3}, | ||
| 239 | {"MX", "XY", 3}, | ||
| 240 | {"IL", "IL", 0}, | ||
| 241 | {"CH", "CH", 0}, | ||
| 242 | {"TR", "TR", 0}, | ||
| 243 | {"NO", "NO", 0}, | ||
| 244 | #endif /* EXMAPLE_TABLE */ | ||
| 245 | }; | ||
| 246 | |||
| 247 | |||
| 248 | /* Customized Locale convertor | ||
| 249 | * input : ISO 3166-1 country abbreviation | ||
| 250 | * output: customized cspec | ||
| 251 | */ | ||
| 252 | void get_customized_country_code(char *country_iso_code, wl_country_t *cspec) | ||
| 253 | { | ||
| 254 | #if defined(CUSTOMER_HW2) && (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 39)) | ||
| 255 | |||
| 256 | struct cntry_locales_custom *cloc_ptr; | ||
| 257 | |||
| 258 | if (!cspec) | ||
| 259 | return; | ||
| 260 | |||
| 261 | cloc_ptr = wifi_get_country_code(country_iso_code); | ||
| 262 | if (cloc_ptr) { | ||
| 263 | strlcpy(cspec->ccode, cloc_ptr->custom_locale, WLC_CNTRY_BUF_SZ); | ||
| 264 | cspec->rev = cloc_ptr->custom_locale_rev; | ||
| 265 | } | ||
| 266 | return; | ||
| 267 | #else | ||
| 268 | int size, i; | ||
| 269 | |||
| 270 | size = ARRAYSIZE(translate_custom_table); | ||
| 271 | |||
| 272 | if (cspec == 0) | ||
| 273 | return; | ||
| 274 | |||
| 275 | if (size == 0) | ||
| 276 | return; | ||
| 277 | |||
| 278 | for (i = 0; i < size; i++) { | ||
| 279 | if (strcmp(country_iso_code, translate_custom_table[i].iso_abbrev) == 0) { | ||
| 280 | memcpy(cspec->ccode, | ||
| 281 | translate_custom_table[i].custom_locale, WLC_CNTRY_BUF_SZ); | ||
| 282 | cspec->rev = translate_custom_table[i].custom_locale_rev; | ||
| 283 | return; | ||
| 284 | } | ||
| 285 | } | ||
| 286 | #ifdef EXAMPLE_TABLE | ||
| 287 | /* if no country code matched return first universal code from translate_custom_table */ | ||
| 288 | memcpy(cspec->ccode, translate_custom_table[0].custom_locale, WLC_CNTRY_BUF_SZ); | ||
| 289 | cspec->rev = translate_custom_table[0].custom_locale_rev; | ||
| 290 | #endif /* EXMAPLE_TABLE */ | ||
| 291 | return; | ||
| 292 | #endif /* defined(CUSTOMER_HW2) && (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 39)) */ | ||
| 293 | } | ||
