1/* 2 * Copyright (c) 2002-2008 Sam Leffler, Errno Consulting 3 * Copyright (c) 2002-2008 Atheros Communications, Inc. 4 * 5 * Permission to use, copy, modify, and/or distribute this software for any 6 * purpose with or without fee is hereby granted, provided that the above 7 * copyright notice and this permission notice appear in all copies. 8 * 9 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 10 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 11 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR 12 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 13 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 14 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 15 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 16 * 17 * $FreeBSD$ 18 */ 19#include "opt_ah.h" 20 21#include "ah.h" 22#include "ah_internal.h" 23#include "ah_devid.h" 24#ifdef AH_DEBUG 25#include "ah_desc.h" /* NB: for HAL_PHYERR* */ 26#endif 27 28#include "ar5212/ar5212.h" 29#include "ar5212/ar5212reg.h" 30#include "ar5212/ar5212phy.h" 31 32#define AR_NUM_GPIO 6 /* 6 GPIO pins */ 33#define AR_GPIOD_MASK 0x0000002F /* GPIO data reg r/w mask */ 34 35/* 36 * Configure GPIO Output lines 37 */ 38HAL_BOOL 39ar5212GpioCfgOutput(struct ath_hal *ah, uint32_t gpio, HAL_GPIO_MUX_TYPE type) 40{ 41 HALASSERT(gpio < AR_NUM_GPIO); 42 43 /* 44 * NB: AR_GPIOCR_CR_A(pin) is all 1's so there's no need 45 * to clear the field before or'ing in the new value. 46 */ 47 OS_REG_WRITE(ah, AR_GPIOCR, 48 OS_REG_READ(ah, AR_GPIOCR) | AR_GPIOCR_CR_A(gpio)); 49 50 return AH_TRUE; 51} 52 53/* 54 * Configure GPIO Input lines 55 */ 56HAL_BOOL 57ar5212GpioCfgInput(struct ath_hal *ah, uint32_t gpio) 58{ 59 HALASSERT(gpio < AR_NUM_GPIO); 60 61 OS_REG_WRITE(ah, AR_GPIOCR, 62 (OS_REG_READ(ah, AR_GPIOCR) &~ AR_GPIOCR_CR_A(gpio)) 63 | AR_GPIOCR_CR_N(gpio)); 64 65 return AH_TRUE; 66} 67 68/* 69 * Once configured for I/O - set output lines 70 */ 71HAL_BOOL 72ar5212GpioSet(struct ath_hal *ah, uint32_t gpio, uint32_t val) 73{ 74 uint32_t reg; 75 76 HALASSERT(gpio < AR_NUM_GPIO); 77 78 reg = OS_REG_READ(ah, AR_GPIODO); 79 reg &= ~(1 << gpio); 80 reg |= (val&1) << gpio; 81 82 OS_REG_WRITE(ah, AR_GPIODO, reg); 83 return AH_TRUE; 84} 85 86/* 87 * Once configured for I/O - get input lines 88 */ 89uint32_t 90ar5212GpioGet(struct ath_hal *ah, uint32_t gpio) 91{ 92 if (gpio < AR_NUM_GPIO) { 93 uint32_t val = OS_REG_READ(ah, AR_GPIODI); 94 val = ((val & AR_GPIOD_MASK) >> gpio) & 0x1; 95 return val; 96 } else { 97 return 0xffffffff; 98 } 99} 100 101/* 102 * Set the GPIO Interrupt 103 */ 104void 105ar5212GpioSetIntr(struct ath_hal *ah, u_int gpio, uint32_t ilevel) 106{ 107 uint32_t val; 108 109 /* XXX bounds check gpio */ 110 val = OS_REG_READ(ah, AR_GPIOCR); 111 val &= ~(AR_GPIOCR_CR_A(gpio) | 112 AR_GPIOCR_INT_MASK | AR_GPIOCR_INT_ENA | AR_GPIOCR_INT_SEL); 113 val |= AR_GPIOCR_CR_N(gpio) | AR_GPIOCR_INT(gpio) | AR_GPIOCR_INT_ENA; 114 if (ilevel) 115 val |= AR_GPIOCR_INT_SELH; /* interrupt on pin high */ 116 else 117 val |= AR_GPIOCR_INT_SELL; /* interrupt on pin low */ 118 119 /* Don't need to change anything for low level interrupt. */ 120 OS_REG_WRITE(ah, AR_GPIOCR, val); 121 122 /* Change the interrupt mask. */ 123 (void) ar5212SetInterrupts(ah, AH5212(ah)->ah_maskReg | HAL_INT_GPIO); 124} 125