rk: temp revert rk change
[firefly-linux-kernel-4.4.55.git] / drivers / net / wireless / bcmdhd / dhd_custom_gpio.c
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 339054 2012-06-15 04:56:55Z $
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 }