blob: 79f213ef1c2c79f85f71b411d7503080d8ac2868 [file] [log] [blame]
Santosh Sajjan374d6592012-01-19 23:16:46 +05301/* Copyright (c) 2012, Code Aurora Forum. All rights reserved.
2 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 */
12#include <linux/delay.h>
13#include <linux/device.h>
14#include <linux/err.h>
15#include <linux/gpio.h>
16#include <linux/platform_device.h>
17#include <linux/regulator/consumer.h>
18#include <asm/mach-types.h>
19#include <mach/rpc_pmapp.h>
20#include "board-msm7627a.h"
21#include "devices-msm7x2xa.h"
22#include "timer.h"
23
24#define GPIO_WLAN_3V3_EN 119
25static const char *id = "WLAN";
26
27enum {
28 WLAN_VREG_S3 = 0,
29 WLAN_VREG_L17,
30 WLAN_VREG_L19
31};
32
33struct wlan_vreg_info {
34 const char *vreg_id;
35 unsigned int level_min;
36 unsigned int level_max;
37 unsigned int pmapp_id;
38 unsigned int is_vreg_pin_controlled;
39 struct regulator *reg;
40};
41
42static struct wlan_vreg_info vreg_info[] = {
43 {"msme1", 1800000, 1800000, 2, 0, NULL},
44 {"bt", 3300000, 3300000, 21, 1, NULL},
45 {"wlan4", 1800000, 1800000, 23, 1, NULL}
46};
47
48int gpio_wlan_sys_rest_en = 134;
49static void gpio_wlan_config(void)
50{
Chintan Pandyaf4ad4002012-02-28 19:49:03 +053051 if (machine_is_msm7627a_qrd1() || machine_is_msm7627a_evb()
Aparna Mallavarapu093e7fd2012-04-05 13:08:50 +053052 || machine_is_msm8625_evb()
Aparna Mallavarapu5a326242012-05-09 19:49:02 +053053 || machine_is_msm8625_evt()
Aparna Mallavarapu093e7fd2012-04-05 13:08:50 +053054 || machine_is_msm7627a_qrd3()
55 || machine_is_msm8625_qrd7())
Santosh Sajjan374d6592012-01-19 23:16:46 +053056 gpio_wlan_sys_rest_en = 124;
57}
58
59static unsigned int qrf6285_init_regs(void)
60{
61 struct regulator_bulk_data regs[ARRAY_SIZE(vreg_info)];
62 int i = 0, rc = 0;
63
64 for (i = 0; i < ARRAY_SIZE(regs); i++) {
65 regs[i].supply = vreg_info[i].vreg_id;
66 regs[i].min_uV = vreg_info[i].level_min;
67 regs[i].max_uV = vreg_info[i].level_max;
68 }
69
70 rc = regulator_bulk_get(NULL, ARRAY_SIZE(regs), regs);
71 if (rc) {
72 pr_err("%s: could not get regulators: %d\n", __func__, rc);
73 goto out;
74 }
75
76 for (i = 0; i < ARRAY_SIZE(regs); i++)
77 vreg_info[i].reg = regs[i].consumer;
78
79out:
80 return rc;
81}
82
83static unsigned int setup_wlan_gpio(bool on)
84{
85 int rc = 0;
86
87 if (on) {
88 rc = gpio_direction_output(gpio_wlan_sys_rest_en, 1);
89 msleep(100);
90 } else {
91 gpio_set_value_cansleep(gpio_wlan_sys_rest_en, 0);
92 rc = gpio_direction_input(gpio_wlan_sys_rest_en);
93 msleep(100);
94 }
95
96 if (rc)
97 pr_err("%s: WLAN sys_reset_en GPIO: Error", __func__);
98
99 return rc;
100}
101
102static unsigned int setup_wlan_clock(bool on)
103{
104 int rc = 0;
105
106 if (on) {
107 /* Vote for A0 clock */
108 rc = pmapp_clock_vote(id, PMAPP_CLOCK_ID_A0,
109 PMAPP_CLOCK_VOTE_ON);
110 } else {
111 /* Vote against A0 clock */
112 rc = pmapp_clock_vote(id, PMAPP_CLOCK_ID_A0,
113 PMAPP_CLOCK_VOTE_OFF);
114 }
115
116 if (rc)
117 pr_err("%s: Configuring A0 clock for WLAN: Error", __func__);
118
119 return rc;
120}
121
122static unsigned int wlan_switch_regulators(int on)
123{
124 int rc = 0, index = 0;
125
126 if (machine_is_msm7627a_qrd1())
127 index = 2;
128
129 for ( ; index < ARRAY_SIZE(vreg_info); index++) {
130 if (on) {
131 rc = regulator_set_voltage(vreg_info[index].reg,
132 vreg_info[index].level_min,
133 vreg_info[index].level_max);
134 if (rc) {
135 pr_err("%s:%s set voltage failed %d\n",
136 __func__, vreg_info[index].vreg_id, rc);
137 goto reg_disable;
138 }
139
140 rc = regulator_enable(vreg_info[index].reg);
141 if (rc) {
142 pr_err("%s:%s vreg enable failed %d\n",
143 __func__, vreg_info[index].vreg_id, rc);
144 goto reg_disable;
145 }
146
147 if (vreg_info[index].is_vreg_pin_controlled) {
148 rc = pmapp_vreg_lpm_pincntrl_vote(id,
149 vreg_info[index].pmapp_id,
150 PMAPP_CLOCK_ID_A0, 1);
151 if (rc) {
152 pr_err("%s:%s pincntrl failed %d\n",
153 __func__,
154 vreg_info[index].vreg_id, rc);
155 goto pin_cnt_fail;
156 }
157 }
158 } else {
159 if (vreg_info[index].is_vreg_pin_controlled) {
160 rc = pmapp_vreg_lpm_pincntrl_vote(id,
161 vreg_info[index].pmapp_id,
162 PMAPP_CLOCK_ID_A0, 0);
163 if (rc) {
164 pr_err("%s:%s pincntrl failed %d\n",
165 __func__,
166 vreg_info[index].vreg_id, rc);
167 goto pin_cnt_fail;
168 }
169 }
170
171 rc = regulator_disable(vreg_info[index].reg);
172 if (rc) {
173 pr_err("%s:%s vreg disable failed %d\n",
174 __func__,
175 vreg_info[index].vreg_id, rc);
176 goto reg_disable;
177 }
178 }
179 }
180 return 0;
181pin_cnt_fail:
182 if (on)
183 regulator_disable(vreg_info[index].reg);
184reg_disable:
185 if (!machine_is_msm7627a_qrd1()) {
186 while (index) {
187 if (on) {
188 index--;
189 regulator_disable(vreg_info[index].reg);
190 regulator_put(vreg_info[index].reg);
191 }
192 }
193 }
194 return rc;
195}
196
197static unsigned int msm_AR600X_setup_power(bool on)
198{
199 int rc = 0;
200 static bool init_done;
201
202 if (unlikely(!init_done)) {
203 gpio_wlan_config();
204 rc = qrf6285_init_regs();
205 if (rc) {
206 pr_err("%s: qrf6285 init failed = %d\n", __func__, rc);
207 return rc;
208 } else {
209 init_done = true;
210 }
211 }
212
213 rc = wlan_switch_regulators(on);
214 if (rc) {
215 pr_err("%s: wlan_switch_regulators error = %d\n", __func__, rc);
216 goto out;
217 }
218
219 /* GPIO_WLAN_3V3_EN is only required for the QRD7627a */
220 if (machine_is_msm7627a_qrd1()) {
221 rc = gpio_tlmm_config(GPIO_CFG(GPIO_WLAN_3V3_EN, 0,
222 GPIO_CFG_OUTPUT, GPIO_CFG_NO_PULL,
223 GPIO_CFG_2MA), GPIO_CFG_ENABLE);
224 if (rc) {
225 pr_err("%s gpio_tlmm_config 119 failed,error = %d\n",
226 __func__, rc);
227 goto reg_disable;
228 }
229 gpio_set_value(GPIO_WLAN_3V3_EN, 1);
230 }
231
232 /*
233 * gpio_wlan_sys_rest_en is not from the GPIO expander for QRD7627a,
234 * EVB1.0 and QRD8625,so the below step is required for those devices.
235 */
Chintan Pandyaf4ad4002012-02-28 19:49:03 +0530236 if (machine_is_msm7627a_qrd1() || machine_is_msm7627a_evb()
Aparna Mallavarapu093e7fd2012-04-05 13:08:50 +0530237 || machine_is_msm8625_evb()
Aparna Mallavarapu5a326242012-05-09 19:49:02 +0530238 || machine_is_msm8625_evt()
Aparna Mallavarapu093e7fd2012-04-05 13:08:50 +0530239 || machine_is_msm7627a_qrd3()
240 || machine_is_msm8625_qrd7()) {
Santosh Sajjan374d6592012-01-19 23:16:46 +0530241 rc = gpio_tlmm_config(GPIO_CFG(gpio_wlan_sys_rest_en, 0,
242 GPIO_CFG_OUTPUT, GPIO_CFG_NO_PULL,
243 GPIO_CFG_2MA), GPIO_CFG_ENABLE);
244 if (rc) {
245 pr_err("%s gpio_tlmm_config 119 failed,error = %d\n",
246 __func__, rc);
247 goto qrd_gpio_fail;
248 }
249 gpio_set_value(gpio_wlan_sys_rest_en, 1);
250 } else {
251 rc = gpio_request(gpio_wlan_sys_rest_en, "WLAN_DEEP_SLEEP_N");
252 if (rc) {
253 pr_err("%s: WLAN sys_rest_en GPIO %d request failed %d\n",
254 __func__,
255 gpio_wlan_sys_rest_en, rc);
256 goto qrd_gpio_fail;
257 }
258 rc = setup_wlan_gpio(on);
259 if (rc) {
260 pr_err("%s: wlan_set_gpio = %d\n", __func__, rc);
261 goto gpio_fail;
262 }
263 }
264
265 /* Enable the A0 clock */
266 rc = setup_wlan_clock(on);
267 if (rc) {
268 pr_err("%s: setup_wlan_clock = %d\n", __func__, rc);
269 goto set_gpio_fail;
270 }
271
272 /* Configure A0 clock to be slave to WLAN_CLK_PWR_REQ */
273 rc = pmapp_clock_vote(id, PMAPP_CLOCK_ID_A0,
274 PMAPP_CLOCK_VOTE_PIN_CTRL);
275 if (rc) {
276 pr_err("%s: Configuring A0 to Pin controllable failed %d\n",
277 __func__, rc);
278 goto set_clock_fail;
279 }
280
281 pr_info("WLAN power-up success\n");
282 return 0;
283set_clock_fail:
284 setup_wlan_clock(0);
285set_gpio_fail:
286 setup_wlan_gpio(0);
287gpio_fail:
288 gpio_free(gpio_wlan_sys_rest_en);
289qrd_gpio_fail:
Santosh Sajjan7a435d32012-04-23 18:34:38 +0530290 /* GPIO_WLAN_3V3_EN is only required for the QRD7627a */
291 if (machine_is_msm7627a_qrd1())
292 gpio_free(GPIO_WLAN_3V3_EN);
Santosh Sajjan374d6592012-01-19 23:16:46 +0530293reg_disable:
294 wlan_switch_regulators(0);
295out:
296 pr_info("WLAN power-up failed\n");
297 return rc;
298}
299
300static unsigned int msm_AR600X_shutdown_power(bool on)
301{
302 int rc = 0;
303
304 /* Disable the A0 clock */
305 rc = setup_wlan_clock(on);
306 if (rc) {
307 pr_err("%s: setup_wlan_clock = %d\n", __func__, rc);
308 goto set_clock_fail;
309 }
310
311 /*
312 * gpio_wlan_sys_rest_en is not from the GPIO expander for QRD7627a,
313 * EVB1.0 and QRD8625,so the below step is required for those devices.
314 */
Chintan Pandyaf4ad4002012-02-28 19:49:03 +0530315 if (machine_is_msm7627a_qrd1() || machine_is_msm7627a_evb()
Aparna Mallavarapu093e7fd2012-04-05 13:08:50 +0530316 || machine_is_msm8625_evb()
Aparna Mallavarapu5a326242012-05-09 19:49:02 +0530317 || machine_is_msm8625_evt()
Aparna Mallavarapu093e7fd2012-04-05 13:08:50 +0530318 || machine_is_msm7627a_qrd3()
319 || machine_is_msm8625_qrd7()) {
Santosh Sajjan374d6592012-01-19 23:16:46 +0530320 rc = gpio_tlmm_config(GPIO_CFG(gpio_wlan_sys_rest_en, 0,
321 GPIO_CFG_OUTPUT, GPIO_CFG_NO_PULL,
322 GPIO_CFG_2MA), GPIO_CFG_ENABLE);
323 if (rc) {
324 pr_err("%s gpio_tlmm_config 119 failed,error = %d\n",
325 __func__, rc);
326 goto gpio_fail;
327 }
328 gpio_set_value(gpio_wlan_sys_rest_en, 0);
329 } else {
Santosh Sajjaneb602fc2012-05-29 14:44:12 +0530330 rc = gpio_request(gpio_wlan_sys_rest_en, "WLAN_DEEP_SLEEP_N");
331 if (!rc) {
332 rc = setup_wlan_gpio(on);
333 if (rc) {
334 pr_err("%s: setup_wlan_gpio = %d\n",
335 __func__, rc);
336 goto set_gpio_fail;
337 }
338 gpio_free(gpio_wlan_sys_rest_en);
Ming-yi Lin920f2202012-09-04 11:30:16 +0530339 } else {
340 pr_err("%s: WLAN sys_rest_en GPIO %d request failed %d\n",
341 __func__, gpio_wlan_sys_rest_en, rc);
342 goto out;
Santosh Sajjan374d6592012-01-19 23:16:46 +0530343 }
Santosh Sajjan374d6592012-01-19 23:16:46 +0530344 }
345
346 /* GPIO_WLAN_3V3_EN is only required for the QRD7627a */
347 if (machine_is_msm7627a_qrd1()) {
348 rc = gpio_tlmm_config(GPIO_CFG(GPIO_WLAN_3V3_EN, 0,
349 GPIO_CFG_OUTPUT, GPIO_CFG_NO_PULL,
350 GPIO_CFG_2MA), GPIO_CFG_ENABLE);
351 if (rc) {
352 pr_err("%s gpio_tlmm_config 119 failed,error = %d\n",
353 __func__, rc);
354 goto qrd_gpio_fail;
355 }
356 gpio_set_value(GPIO_WLAN_3V3_EN, 0);
357 }
358
359 rc = wlan_switch_regulators(on);
360 if (rc) {
361 pr_err("%s: wlan_switch_regulators error = %d\n",
362 __func__, rc);
363 goto reg_disable;
364 }
365
366 pr_info("WLAN power-down success\n");
367 return 0;
368set_clock_fail:
369 setup_wlan_clock(0);
370set_gpio_fail:
371 setup_wlan_gpio(0);
372gpio_fail:
373 gpio_free(gpio_wlan_sys_rest_en);
374qrd_gpio_fail:
Santosh Sajjan7a435d32012-04-23 18:34:38 +0530375 /* GPIO_WLAN_3V3_EN is only required for the QRD7627a */
376 if (machine_is_msm7627a_qrd1())
377 gpio_free(GPIO_WLAN_3V3_EN);
Santosh Sajjan374d6592012-01-19 23:16:46 +0530378reg_disable:
379 wlan_switch_regulators(0);
Ming-yi Lin920f2202012-09-04 11:30:16 +0530380out:
Santosh Sajjan374d6592012-01-19 23:16:46 +0530381 pr_info("WLAN power-down failed\n");
382 return rc;
383}
384
385int ar600x_wlan_power(bool on)
386{
387 if (on)
388 msm_AR600X_setup_power(on);
389 else
390 msm_AR600X_shutdown_power(on);
391
392 return 0;
393}