xref: /freebsd/sys/contrib/dev/ath/ath_hal/ar9300/ar9300_gpio.c (revision 3823d5e198425b4f5e5a80267d195769d1063773)
1 /*
2  * Copyright (c) 2013 Qualcomm Atheros, Inc.
3  *
4  * Permission to use, copy, modify, and/or distribute this software for any
5  * purpose with or without fee is hereby granted, provided that the above
6  * copyright notice and this permission notice appear in all copies.
7  *
8  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
9  * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
10  * AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
11  * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
12  * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
13  * OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
14  * PERFORMANCE OF THIS SOFTWARE.
15  */
16 
17 #include "opt_ah.h"
18 
19 #include "ah.h"
20 #include "ah_internal.h"
21 #include "ah_devid.h"
22 #ifdef AH_DEBUG
23 #include "ah_desc.h"                    /* NB: for HAL_PHYERR* */
24 #endif
25 
26 #include "ar9300/ar9300.h"
27 #include "ar9300/ar9300reg.h"
28 #include "ar9300/ar9300phy.h"
29 
30 #define AR_GPIO_BIT(_gpio)                      (1 << (_gpio))
31 
32 /*
33  * Configure GPIO Output Mux control
34  */
35 #if UMAC_SUPPORT_SMARTANTENNA
36 static void  ar9340_soc_gpio_cfg_output_mux(
37     struct ath_hal *ah,
38     u_int32_t gpio,
39     u_int32_t ah_signal_type)
40 {
41 #define ADDR_READ(addr)      (*((volatile u_int32_t *)(addr)))
42 #define ADDR_WRITE(addr, b)   (void)((*(volatile u_int32_t *) (addr)) = (b))
43 #define AR9340_SOC_GPIO_FUN0    0xB804002c
44 #define AR9340_SOC_GPIO_OE      0xB8040000
45 #if ATH_SMARTANTENNA_DISABLE_JTAG
46 #define AR9340_SOC_GPIO_FUNCTION   (volatile u_int32_t*) 0xB804006c
47 #define WASP_DISABLE_JTAG  0x2
48 #define MAX_JTAG_GPIO_PIN 1
49 #endif
50     u_int8_t out_func, shift;
51     u_int32_t  flags;
52     volatile u_int32_t* address;
53 
54     if (!ah_signal_type){
55         return;
56     }
57 #if ATH_SMARTANTENNA_DISABLE_JTAG
58 /*
59  * To use GPIO pins 0 and 1 for controling antennas, JTAG needs to disabled.
60  */
61     if (gpio <= MAX_JTAG_GPIO_PIN) {
62         flags = ADDR_READ(AR9340_SOC_GPIO_FUNCTION);
63         flags |= WASP_DISABLE_JTAG;
64         ADDR_WRITE(AR9340_SOC_GPIO_FUNCTION, flags);
65     }
66 #endif
67     out_func = gpio / 4;
68     shift = (gpio % 4);
69     address = (volatile u_int32_t *)(AR9340_SOC_GPIO_FUN0 + (out_func*4));
70 
71     flags = ADDR_READ(address);
72     flags |= ah_signal_type << (8*shift);
73     ADDR_WRITE(address, flags);
74     flags = ADDR_READ(AR9340_SOC_GPIO_OE);
75     flags &= ~(1 << gpio);
76     ADDR_WRITE(AR9340_SOC_GPIO_OE, flags);
77 
78 }
79 #endif
80 
81 static void
82 ar9300_gpio_cfg_output_mux(struct ath_hal *ah, u_int32_t gpio, u_int32_t type)
83 {
84     int          addr;
85     u_int32_t    gpio_shift;
86 
87     /* each MUX controls 6 GPIO pins */
88     if (gpio > 11) {
89         addr = AR_HOSTIF_REG(ah, AR_GPIO_OUTPUT_MUX3);
90     } else if (gpio > 5) {
91         addr = AR_HOSTIF_REG(ah, AR_GPIO_OUTPUT_MUX2);
92     } else {
93         addr = AR_HOSTIF_REG(ah, AR_GPIO_OUTPUT_MUX1);
94     }
95 
96     /*
97      * 5 bits per GPIO pin.
98      * Bits 0..4 for 1st pin in that mux,
99      * bits 5..9 for 2nd pin, etc.
100      */
101     gpio_shift = (gpio % 6) * 5;
102 
103     OS_REG_RMW(ah, addr, (type << gpio_shift), (0x1f << gpio_shift));
104 }
105 
106 /*
107  * Configure GPIO Output lines
108  */
109 HAL_BOOL
110 ar9300_gpio_cfg_output(
111     struct ath_hal *ah,
112     u_int32_t gpio,
113     HAL_GPIO_MUX_TYPE hal_signal_type)
114 {
115     u_int32_t    ah_signal_type;
116     u_int32_t    gpio_shift;
117     u_int8_t    smart_ant = 0;
118     static const u_int32_t    mux_signal_conversion_table[] = {
119                     /* HAL_GPIO_OUTPUT_MUX_AS_OUTPUT             */
120         AR_GPIO_OUTPUT_MUX_AS_OUTPUT,
121                     /* HAL_GPIO_OUTPUT_MUX_AS_PCIE_ATTENTION_LED */
122         AR_GPIO_OUTPUT_MUX_AS_PCIE_ATTENTION_LED,
123                     /* HAL_GPIO_OUTPUT_MUX_AS_PCIE_POWER_LED     */
124         AR_GPIO_OUTPUT_MUX_AS_PCIE_POWER_LED,
125                     /* HAL_GPIO_OUTPUT_MUX_AS_MAC_NETWORK_LED    */
126         AR_GPIO_OUTPUT_MUX_AS_MAC_NETWORK_LED,
127                     /* HAL_GPIO_OUTPUT_MUX_AS_MAC_POWER_LED      */
128         AR_GPIO_OUTPUT_MUX_AS_MAC_POWER_LED,
129                     /* HAL_GPIO_OUTPUT_MUX_AS_WLAN_ACTIVE        */
130         AR_GPIO_OUTPUT_MUX_AS_RX_CLEAR_EXTERNAL,
131                     /* HAL_GPIO_OUTPUT_MUX_AS_TX_FRAME           */
132         AR_GPIO_OUTPUT_MUX_AS_TX_FRAME,
133                     /* HAL_GPIO_OUTPUT_MUX_AS_MCI_WLAN_DATA      */
134         AR_GPIO_OUTPUT_MUX_AS_MCI_WLAN_DATA,
135                     /* HAL_GPIO_OUTPUT_MUX_AS_MCI_WLAN_CLK       */
136         AR_GPIO_OUTPUT_MUX_AS_MCI_WLAN_CLK,
137                     /* HAL_GPIO_OUTPUT_MUX_AS_MCI_BT_DATA        */
138         AR_GPIO_OUTPUT_MUX_AS_MCI_BT_DATA,
139                     /* HAL_GPIO_OUTPUT_MUX_AS_MCI_BT_CLK         */
140         AR_GPIO_OUTPUT_MUX_AS_MCI_BT_CLK,
141 	            /* HAL_GPIO_OUTPUT_MUX_AS_WL_IN_TX           */
142         AR_GPIO_OUTPUT_MUX_AS_WL_IN_TX,
143                     /* HAL_GPIO_OUTPUT_MUX_AS_WL_IN_RX           */
144         AR_GPIO_OUTPUT_MUX_AS_WL_IN_RX,
145                     /* HAL_GPIO_OUTPUT_MUX_AS_BT_IN_TX           */
146         AR_GPIO_OUTPUT_MUX_AS_BT_IN_TX,
147                     /* HAL_GPIO_OUTPUT_MUX_AS_BT_IN_RX           */
148         AR_GPIO_OUTPUT_MUX_AS_BT_IN_RX,
149                     /* HAL_GPIO_OUTPUT_MUX_AS_RUCKUS_STROBE      */
150         AR_GPIO_OUTPUT_MUX_AS_RUCKUS_STROBE,
151                     /* HAL_GPIO_OUTPUT_MUX_AS_RUCKUS_DATA        */
152         AR_GPIO_OUTPUT_MUX_AS_RUCKUS_DATA,
153                     /* HAL_GPIO_OUTPUT_MUX_AS_SMARTANT_CTRL0     */
154         AR_GPIO_OUTPUT_MUX_AS_SMARTANT_CTRL0,
155                     /* HAL_GPIO_OUTPUT_MUX_AS_SMARTANT_CTRL1     */
156         AR_GPIO_OUTPUT_MUX_AS_SMARTANT_CTRL1,
157                     /* HAL_GPIO_OUTPUT_MUX_AS_SMARTANT_CTRL2     */
158         AR_GPIO_OUTPUT_MUX_AS_SMARTANT_CTRL2,
159                     /* HAL_GPIO_OUTPUT_MUX_AS_SMARTANT_SWCOM3    */
160         AR_GPIO_OUTPUT_MUX_AS_SWCOM3,
161     };
162 
163     HALASSERT(gpio < AH_PRIVATE(ah)->ah_caps.halNumGpioPins);
164     if ((gpio == AR9382_GPIO_PIN_8_RESERVED)  ||
165         (gpio == AR9382_GPIO_PIN_11_RESERVED) ||
166         (gpio == AR9382_GPIO_9_INPUT_ONLY))
167     {
168         return AH_FALSE;
169     }
170 
171     /* Convert HAL signal type definitions to hardware-specific values. */
172     if ((int) hal_signal_type < ARRAY_LENGTH(mux_signal_conversion_table))
173     {
174         ah_signal_type = mux_signal_conversion_table[hal_signal_type];
175     } else {
176         return AH_FALSE;
177     }
178 
179     if (gpio <= AR9382_MAX_JTAG_GPIO_PIN_NUM) {
180         OS_REG_SET_BIT(ah,
181             AR_HOSTIF_REG(ah, AR_GPIO_INPUT_EN_VAL), AR_GPIO_JTAG_DISABLE);
182     }
183 
184 #if UMAC_SUPPORT_SMARTANTENNA
185     /* Get the pin and func values for smart antenna */
186     switch (ah_signal_type)
187     {
188         case AR_GPIO_OUTPUT_MUX_AS_SMARTANT_CTRL0:
189             gpio = ATH_GPIOPIN_ANTCHAIN0;
190             ah_signal_type = ATH_GPIOFUNC_ANTCHAIN0;
191             smart_ant = 1;
192             break;
193         case AR_GPIO_OUTPUT_MUX_AS_SMARTANT_CTRL1:
194             gpio = ATH_GPIOPIN_ANTCHAIN1;
195             ah_signal_type = ATH_GPIOFUNC_ANTCHAIN1;
196             smart_ant = 1;
197             break;
198         case AR_GPIO_OUTPUT_MUX_AS_SMARTANT_CTRL2:
199             gpio = ATH_GPIOPIN_ANTCHAIN2;
200             ah_signal_type = ATH_GPIOFUNC_ANTCHAIN2;
201             smart_ant = 1;
202             break;
203 #if ATH_SMARTANTENNA_ROUTE_SWCOM_TO_GPIO
204         case AR_GPIO_OUTPUT_MUX_AS_SWCOM3:
205             gpio = ATH_GPIOPIN_ROUTE_SWCOM3;
206             ah_signal_type = ATH_GPIOFUNC_ROUTE_SWCOM3;
207             smart_ant = 1;
208             break;
209 #endif
210         default:
211             break;
212     }
213 #endif
214 
215     if (smart_ant && (AR_SREV_WASP(ah) || AR_SREV_SCORPION(ah)))
216     {
217 #if UMAC_SUPPORT_SMARTANTENNA
218         ar9340_soc_gpio_cfg_output_mux(ah, gpio, ah_signal_type);
219 #endif
220         return AH_TRUE;
221     } else
222     {
223         /* Configure the MUX */
224         ar9300_gpio_cfg_output_mux(ah, gpio, ah_signal_type);
225     }
226 
227     /* 2 bits per output mode */
228     gpio_shift = 2 * gpio;
229 
230     OS_REG_RMW(ah,
231                AR_HOSTIF_REG(ah, AR_GPIO_OE_OUT),
232                (AR_GPIO_OE_OUT_DRV_ALL << gpio_shift),
233                (AR_GPIO_OE_OUT_DRV << gpio_shift));
234     return AH_TRUE;
235 }
236 
237 /*
238  * Configure GPIO Output lines -LED off
239  */
240 HAL_BOOL
241 ar9300_gpio_cfg_output_led_off(
242     struct ath_hal *ah,
243     u_int32_t gpio,
244     HAL_GPIO_MUX_TYPE halSignalType)
245 {
246 #define N(a)    (sizeof(a) / sizeof(a[0]))
247     u_int32_t    ah_signal_type;
248     u_int32_t    gpio_shift;
249     u_int8_t    smart_ant = 0;
250 
251     static const u_int32_t    mux_signal_conversion_table[] = {
252         /* HAL_GPIO_OUTPUT_MUX_AS_OUTPUT             */
253         AR_GPIO_OUTPUT_MUX_AS_OUTPUT,
254         /* HAL_GPIO_OUTPUT_MUX_AS_PCIE_ATTENTION_LED */
255         AR_GPIO_OUTPUT_MUX_AS_PCIE_ATTENTION_LED,
256         /* HAL_GPIO_OUTPUT_MUX_AS_PCIE_POWER_LED     */
257         AR_GPIO_OUTPUT_MUX_AS_PCIE_POWER_LED,
258         /* HAL_GPIO_OUTPUT_MUX_AS_MAC_NETWORK_LED    */
259         AR_GPIO_OUTPUT_MUX_AS_MAC_NETWORK_LED,
260         /* HAL_GPIO_OUTPUT_MUX_AS_MAC_POWER_LED      */
261         AR_GPIO_OUTPUT_MUX_AS_MAC_POWER_LED,
262         /* HAL_GPIO_OUTPUT_MUX_AS_WLAN_ACTIVE        */
263         AR_GPIO_OUTPUT_MUX_AS_RX_CLEAR_EXTERNAL,
264         /* HAL_GPIO_OUTPUT_MUX_AS_TX_FRAME           */
265         AR_GPIO_OUTPUT_MUX_AS_TX_FRAME,
266         /* HAL_GPIO_OUTPUT_MUX_AS_MCI_WLAN_DATA      */
267         AR_GPIO_OUTPUT_MUX_AS_MCI_WLAN_DATA,
268         /* HAL_GPIO_OUTPUT_MUX_AS_MCI_WLAN_CLK       */
269         AR_GPIO_OUTPUT_MUX_AS_MCI_WLAN_CLK,
270         /* HAL_GPIO_OUTPUT_MUX_AS_MCI_BT_DATA        */
271         AR_GPIO_OUTPUT_MUX_AS_MCI_BT_DATA,
272         /* HAL_GPIO_OUTPUT_MUX_AS_MCI_BT_CLK         */
273         AR_GPIO_OUTPUT_MUX_AS_MCI_BT_CLK,
274         /* HAL_GPIO_OUTPUT_MUX_AS_WL_IN_TX           */
275         AR_GPIO_OUTPUT_MUX_AS_WL_IN_TX,
276         /* HAL_GPIO_OUTPUT_MUX_AS_WL_IN_RX           */
277         AR_GPIO_OUTPUT_MUX_AS_WL_IN_RX,
278         /* HAL_GPIO_OUTPUT_MUX_AS_BT_IN_TX           */
279         AR_GPIO_OUTPUT_MUX_AS_BT_IN_TX,
280         /* HAL_GPIO_OUTPUT_MUX_AS_BT_IN_RX           */
281         AR_GPIO_OUTPUT_MUX_AS_BT_IN_RX,
282         AR_GPIO_OUTPUT_MUX_AS_RUCKUS_STROBE,
283         AR_GPIO_OUTPUT_MUX_AS_RUCKUS_DATA,
284         AR_GPIO_OUTPUT_MUX_AS_SMARTANT_CTRL0,
285         AR_GPIO_OUTPUT_MUX_AS_SMARTANT_CTRL1,
286         AR_GPIO_OUTPUT_MUX_AS_SMARTANT_CTRL2
287     };
288     HALASSERT(gpio < AH_PRIVATE(ah)->ah_caps.hal_num_gpio_pins);
289 
290     /* Convert HAL signal type definitions to hardware-specific values. */
291     if ((int) halSignalType < ARRAY_LENGTH(mux_signal_conversion_table))
292     {
293         ah_signal_type = mux_signal_conversion_table[halSignalType];
294     } else {
295         return AH_FALSE;
296     }
297 #if UMAC_SUPPORT_SMARTANTENNA
298     /* Get the pin and func values for smart antenna */
299     switch (halSignalType)
300     {
301         case AR_GPIO_OUTPUT_MUX_AS_SMARTANT_CTRL0:
302             gpio = ATH_GPIOPIN_ANTCHAIN0;
303             ah_signal_type = ATH_GPIOFUNC_ANTCHAIN0;
304             smart_ant = 1;
305             break;
306         case AR_GPIO_OUTPUT_MUX_AS_SMARTANT_CTRL1:
307             gpio = ATH_GPIOPIN_ANTCHAIN1;
308             ah_signal_type = ATH_GPIOFUNC_ANTCHAIN1;
309             smart_ant = 1;
310             break;
311         case AR_GPIO_OUTPUT_MUX_AS_SMARTANT_CTRL2:
312             gpio = ATH_GPIOPIN_ANTCHAIN2;
313             ah_signal_type = ATH_GPIOFUNC_ANTCHAIN2;
314             smart_ant = 1;
315             break;
316         default:
317             break;
318     }
319 #endif
320 
321     if (smart_ant && AR_SREV_WASP(ah))
322     {
323         return AH_FALSE;
324     }
325 
326     // Configure the MUX
327     ar9300_gpio_cfg_output_mux(ah, gpio, ah_signal_type);
328 
329     // 2 bits per output mode
330     gpio_shift = 2*gpio;
331 
332     OS_REG_RMW(ah,
333                AR_HOSTIF_REG(ah, AR_GPIO_OE_OUT),
334                (AR_GPIO_OE_OUT_DRV_NO << gpio_shift),
335                (AR_GPIO_OE_OUT_DRV << gpio_shift));
336 
337     return AH_TRUE;
338 #undef N
339 }
340 
341 /*
342  * Configure GPIO Input lines
343  */
344 HAL_BOOL
345 ar9300_gpio_cfg_input(struct ath_hal *ah, u_int32_t gpio)
346 {
347     u_int32_t    gpio_shift;
348 
349     HALASSERT(gpio < AH_PRIVATE(ah)->ah_caps.halNumGpioPins);
350     if ((gpio == AR9382_GPIO_PIN_8_RESERVED)  ||
351         (gpio == AR9382_GPIO_PIN_11_RESERVED) ||
352         (gpio > AR9382_MAX_GPIO_INPUT_PIN_NUM))
353     {
354         return AH_FALSE;
355     }
356 
357     if (gpio <= AR9382_MAX_JTAG_GPIO_PIN_NUM) {
358         OS_REG_SET_BIT(ah,
359             AR_HOSTIF_REG(ah, AR_GPIO_INPUT_EN_VAL), AR_GPIO_JTAG_DISABLE);
360     }
361     /* TODO: configure input mux for AR9300 */
362     /* If configured as input, set output to tristate */
363     gpio_shift = 2 * gpio;
364 
365     OS_REG_RMW(ah,
366                AR_HOSTIF_REG(ah, AR_GPIO_OE_OUT),
367                (AR_GPIO_OE_OUT_DRV_NO << gpio_shift),
368                (AR_GPIO_OE_OUT_DRV << gpio_shift));
369     return AH_TRUE;
370 }
371 
372 /*
373  * Once configured for I/O - set output lines
374  * output the level of GPio PIN without care work mode
375  */
376 HAL_BOOL
377 ar9300_gpio_set(struct ath_hal *ah, u_int32_t gpio, u_int32_t val)
378 {
379     HALASSERT(gpio < AH_PRIVATE(ah)->ah_caps.halNumGpioPins);
380     if ((gpio == AR9382_GPIO_PIN_8_RESERVED)  ||
381         (gpio == AR9382_GPIO_PIN_11_RESERVED) ||
382         (gpio == AR9382_GPIO_9_INPUT_ONLY))
383     {
384         return AH_FALSE;
385     }
386     OS_REG_RMW(ah, AR_HOSTIF_REG(ah, AR_GPIO_OUT),
387         ((val & 1) << gpio), AR_GPIO_BIT(gpio));
388 
389     return AH_TRUE;
390 }
391 
392 /*
393  * Once configured for I/O - get input lines
394  */
395 u_int32_t
396 ar9300_gpio_get(struct ath_hal *ah, u_int32_t gpio)
397 {
398     u_int32_t gpio_in;
399     HALASSERT(gpio < AH_PRIVATE(ah)->ah_caps.halNumGpioPins);
400     if ((gpio == AR9382_GPIO_PIN_8_RESERVED) ||
401         (gpio == AR9382_GPIO_PIN_11_RESERVED))
402     {
403         return 0xffffffff;
404     }
405 
406     gpio_in = OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_GPIO_IN));
407     OS_REG_RMW(ah, AR_HOSTIF_REG(ah, AR_GPIO_IN),
408         (1 << gpio), AR_GPIO_BIT(gpio));
409     return (MS(gpio_in, AR_GPIO_IN_VAL) & AR_GPIO_BIT(gpio)) != 0;
410 }
411 
412 u_int32_t
413 ar9300_gpio_get_intr(struct ath_hal *ah)
414 {
415     unsigned int mask = 0;
416     struct ath_hal_9300 *ahp = AH9300(ah);
417 
418     mask = ahp->ah_gpio_cause;
419     return mask;
420 }
421 
422 /*
423  * Set the GPIO Interrupt
424  * Sync and Async interrupts are both set/cleared.
425  * Async GPIO interrupts may not be raised when the chip is put to sleep.
426  */
427 void
428 ar9300_gpio_set_intr(struct ath_hal *ah, u_int gpio, u_int32_t ilevel)
429 {
430 
431 
432     int i, reg_bit;
433     u_int32_t reg_val;
434     u_int32_t regs[2], shifts[2];
435 
436 #ifdef AH_ASSERT
437     u_int32_t gpio_mask;
438     u_int32_t old_field_val = 0, field_val = 0;
439 #endif
440 
441 #ifdef ATH_GPIO_USE_ASYNC_CAUSE
442     regs[0] = AR_HOSTIF_REG(ah, AR_INTR_ASYNC_ENABLE);
443     regs[1] = AR_HOSTIF_REG(ah, AR_INTR_ASYNC_MASK);
444     shifts[0] = AR_INTR_ASYNC_ENABLE_GPIO_S;
445     shifts[1] = AR_INTR_ASYNC_MASK_GPIO_S;
446 #else
447     regs[0] = AR_HOSTIF_REG(ah, AR_INTR_SYNC_ENABLE);
448     regs[1] = AR_HOSTIF_REG(ah, AR_INTR_SYNC_MASK);
449     shifts[0] = AR_INTR_SYNC_ENABLE_GPIO_S;
450     shifts[1] = AR_INTR_SYNC_MASK_GPIO_S;
451 #endif
452 
453     HALASSERT(gpio < AH_PRIVATE(ah)->ah_caps.halNumGpioPins);
454 
455     if ((gpio == AR9382_GPIO_PIN_8_RESERVED) ||
456         (gpio == AR9382_GPIO_PIN_11_RESERVED) ||
457         (gpio > AR9382_MAX_GPIO_INPUT_PIN_NUM))
458     {
459         return;
460     }
461 
462 #ifdef AH_ASSERT
463     gpio_mask = (1 << AH_PRIVATE(ah)->ah_caps.halNumGpioPins) - 1;
464 #endif
465     if (ilevel == HAL_GPIO_INTR_DISABLE) {
466         /* clear this GPIO's bit in the interrupt registers */
467         for (i = 0; i < ARRAY_LENGTH(regs); i++) {
468             reg_val = OS_REG_READ(ah, regs[i]);
469             reg_bit = shifts[i] + gpio;
470             reg_val &= ~(1 << reg_bit);
471             OS_REG_WRITE(ah, regs[i], reg_val);
472 
473             /* check that each register has same GPIOs enabled */
474 #ifdef AH_ASSERT
475             field_val = (reg_val >> shifts[i]) & gpio_mask;
476             HALASSERT(i == 0 || old_field_val == field_val);
477             old_field_val = field_val;
478 #endif
479         }
480 
481     } else {
482         reg_val = OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_GPIO_INTR_POL));
483         reg_bit = gpio;
484         if (ilevel == HAL_GPIO_INTR_HIGH) {
485             /* 0 == interrupt on pin high */
486             reg_val &= ~(1 << reg_bit);
487         } else if (ilevel == HAL_GPIO_INTR_LOW) {
488             /* 1 == interrupt on pin low */
489             reg_val |= (1 << reg_bit);
490         }
491         OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_GPIO_INTR_POL), reg_val);
492 
493         /* set this GPIO's bit in the interrupt registers */
494         for (i = 0; i < ARRAY_LENGTH(regs); i++) {
495             reg_val = OS_REG_READ(ah, regs[i]);
496             reg_bit = shifts[i] + gpio;
497             reg_val |= (1 << reg_bit);
498             OS_REG_WRITE(ah, regs[i], reg_val);
499 
500             /* check that each register has same GPIOs enabled */
501 #ifdef AH_ASSERT
502             field_val = (reg_val >> shifts[i]) & gpio_mask;
503             HALASSERT(i == 0 || old_field_val == field_val);
504             old_field_val = field_val;
505 #endif
506         }
507     }
508 }
509 
510 u_int32_t
511 ar9300_gpio_get_polarity(struct ath_hal *ah)
512 {
513     return OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_GPIO_INTR_POL));
514 
515 }
516 
517 void
518 ar9300_gpio_set_polarity(struct ath_hal *ah, u_int32_t pol_map,
519                          u_int32_t changed_mask)
520 {
521     u_int32_t gpio_mask;
522 
523     gpio_mask = (1 << AH_PRIVATE(ah)->ah_caps.halNumGpioPins) - 1;
524     OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_GPIO_INTR_POL), gpio_mask & pol_map);
525 
526 #ifndef ATH_GPIO_USE_ASYNC_CAUSE
527     /*
528      * For SYNC_CAUSE type interrupts, we need to clear the cause register
529      * explicitly. Otherwise an interrupt with the original polarity setting
530      * will come up immediately (if there is already an interrupt source),
531      * which is not what we want usually.
532      */
533     OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_INTR_SYNC_CAUSE_CLR),
534                  changed_mask << AR_INTR_SYNC_ENABLE_GPIO_S);
535     OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_INTR_SYNC_CAUSE_CLR));
536 #endif
537 }
538 
539 /*
540  * get the GPIO input pin mask
541  * gpio0 - gpio13
542  * gpio8, gpio11, regard as reserved by the chip ar9382
543  */
544 
545 u_int32_t
546 ar9300_gpio_get_mask(struct ath_hal *ah)
547 {
548     u_int32_t mask = (1 << (AR9382_MAX_GPIO_INPUT_PIN_NUM + 1) ) - 1;
549 
550     if (AH_PRIVATE(ah)->ah_devid == AR9300_DEVID_AR9380_PCIE) {
551         mask = (1 << AR9382_MAX_GPIO_PIN_NUM) - 1;
552         mask &= ~(1 << AR9382_GPIO_PIN_8_RESERVED |
553                   1 << AR9382_GPIO_PIN_11_RESERVED);
554     }
555     return mask;
556 }
557 
558 int
559 ar9300_gpio_set_mask(struct ath_hal *ah, u_int32_t mask, u_int32_t pol_map)
560 {
561     u_int32_t invalid = ~((1 << (AR9382_MAX_GPIO_INPUT_PIN_NUM + 1)) - 1);
562 
563     if (AH_PRIVATE(ah)->ah_devid == AR9300_DEVID_AR9380_PCIE) {
564         invalid = ~((1 << AR9382_MAX_GPIO_PIN_NUM) - 1);
565         invalid |= 1 << AR9382_GPIO_PIN_8_RESERVED |
566                    1 << AR9382_GPIO_PIN_11_RESERVED;
567     }
568     if (mask & invalid) {
569         ath_hal_printf(ah, "%s: invalid GPIO mask 0x%x\n", __func__, mask);
570         return -1;
571     }
572     AH9300(ah)->ah_gpio_mask = mask;
573     OS_REG_WRITE(ah, AR_HOSTIF_REG(ah, AR_GPIO_INTR_POL), mask & pol_map);
574 
575     return 0;
576 }
577 
578 #ifdef AH_DEBUG
579 void ar9300_gpio_show(struct ath_hal *ah);
580 void ar9300_gpio_show(struct ath_hal *ah)
581 {
582     ath_hal_printf(ah, "--- 9382 GPIOs ---(ah=%p)\n", ah );
583     ath_hal_printf(ah,
584         "AH9300(_ah)->ah_hostifregs:%p\r\n", &(AH9300(ah)->ah_hostifregs));
585     ath_hal_printf(ah,
586         "GPIO_OUT:         0x%08X\n",
587         OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_GPIO_OUT)));
588     ath_hal_printf(ah,
589         "GPIO_IN:          0x%08X\n",
590          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_GPIO_IN)));
591     ath_hal_printf(ah,
592         "GPIO_OE:          0x%08X\n",
593          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_GPIO_OE_OUT)));
594     ath_hal_printf(ah,
595         "GPIO_OE1_OUT:     0x%08X\n",
596          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_GPIO_OE1_OUT)));
597     ath_hal_printf(ah,
598         "GPIO_INTR_POLAR:  0x%08X\n",
599          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_GPIO_INTR_POL)));
600     ath_hal_printf(ah,
601         "GPIO_INPUT_VALUE: 0x%08X\n",
602          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_GPIO_INPUT_EN_VAL)));
603     ath_hal_printf(ah,
604         "GPIO_INPUT_MUX1:  0x%08X\n",
605          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_GPIO_INPUT_MUX1)));
606     ath_hal_printf(ah,
607         "GPIO_INPUT_MUX2:  0x%08X\n",
608          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_GPIO_INPUT_MUX2)));
609     ath_hal_printf(ah,
610         "GPIO_OUTPUT_MUX1: 0x%08X\n",
611          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_GPIO_OUTPUT_MUX1)));
612     ath_hal_printf(ah,
613         "GPIO_OUTPUT_MUX2: 0x%08X\n",
614          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_GPIO_OUTPUT_MUX2)));
615     ath_hal_printf(ah,
616         "GPIO_OUTPUT_MUX3: 0x%08X\n",
617          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_GPIO_OUTPUT_MUX3)));
618     ath_hal_printf(ah,
619         "GPIO_INPUT_STATE: 0x%08X\n",
620          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_INPUT_STATE)));
621     ath_hal_printf(ah,
622         "GPIO_PDPU:        0x%08X\n",
623          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_GPIO_PDPU)));
624     ath_hal_printf(ah,
625         "GPIO_DS:          0x%08X\n",
626          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_GPIO_DS)));
627     ath_hal_printf(ah,
628         "AR_INTR_ASYNC_ENABLE: 0x%08X\n",
629          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_INTR_ASYNC_ENABLE)));
630     ath_hal_printf(ah,
631         "AR_INTR_ASYNC_MASK:   0x%08X\n",
632          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_INTR_ASYNC_MASK)));
633     ath_hal_printf(ah,
634         "AR_INTR_SYNC_ENABLE:  0x%08X\n",
635          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_INTR_SYNC_ENABLE)));
636     ath_hal_printf(ah,
637         "AR_INTR_SYNC_MASK:    0x%08X\n",
638          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_INTR_SYNC_MASK)));
639     ath_hal_printf(ah,
640         "AR_INTR_ASYNC_CAUSE:  0x%08X\n",
641          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_INTR_ASYNC_CAUSE)));
642     ath_hal_printf(ah,
643         "AR_INTR_SYNC_CAUSE:   0x%08X\n",
644          OS_REG_READ(ah, AR_HOSTIF_REG(ah, AR_INTR_SYNC_CAUSE)));
645 
646 }
647 #endif /*AH_DEBUG*/
648