173 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			173 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  *  gpio.h: GPIO Support for PNX833X.
 | |
|  *
 | |
|  *  Copyright 2008 NXP Semiconductors
 | |
|  *	  Chris Steel <chris.steel@nxp.com>
 | |
|  *    Daniel Laird <daniel.j.laird@nxp.com>
 | |
|  *
 | |
|  *  This program is free software; you can redistribute it and/or modify
 | |
|  *  it under the terms of the GNU General Public License as published by
 | |
|  *  the Free Software Foundation; either version 2 of the License, or
 | |
|  *  (at your option) any later version.
 | |
|  *
 | |
|  *  This program is distributed in the hope that it will be useful,
 | |
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  *  GNU General Public License for more details.
 | |
|  *
 | |
|  *  You should have received a copy of the GNU General Public License
 | |
|  *  along with this program; if not, write to the Free Software
 | |
|  *  Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 | |
|  */
 | |
| #ifndef __ASM_MIPS_MACH_PNX833X_GPIO_H
 | |
| #define __ASM_MIPS_MACH_PNX833X_GPIO_H
 | |
| 
 | |
| /* BIG FAT WARNING: races danger!
 | |
|    No protections exist here. Current users are only early init code,
 | |
|    when locking is not needed because no concurrency yet exists there,
 | |
|    and GPIO IRQ dispatcher, which does locking.
 | |
|    However, if many uses will ever happen, proper locking will be needed
 | |
|    - including locking between different uses
 | |
| */
 | |
| 
 | |
| #include <asm/mach-pnx833x/pnx833x.h>
 | |
| 
 | |
| #define SET_REG_BIT(reg, bit)		do { (reg |= (1 << (bit))); } while (0)
 | |
| #define CLEAR_REG_BIT(reg, bit)		do { (reg &= ~(1 << (bit))); } while (0)
 | |
| 
 | |
| /* Initialize GPIO to a known state */
 | |
| static inline void pnx833x_gpio_init(void)
 | |
| {
 | |
| 	PNX833X_PIO_DIR = 0;
 | |
| 	PNX833X_PIO_DIR2 = 0;
 | |
| 	PNX833X_PIO_SEL = 0;
 | |
| 	PNX833X_PIO_SEL2 = 0;
 | |
| 	PNX833X_PIO_INT_EDGE = 0;
 | |
| 	PNX833X_PIO_INT_HI = 0;
 | |
| 	PNX833X_PIO_INT_LO = 0;
 | |
| 
 | |
| 	/* clear any GPIO interrupt requests */
 | |
| 	PNX833X_PIO_INT_CLEAR = 0xffff;
 | |
| 	PNX833X_PIO_INT_CLEAR = 0;
 | |
| 	PNX833X_PIO_INT_ENABLE = 0;
 | |
| }
 | |
| 
 | |
| /* Select GPIO direction for a pin */
 | |
| static inline void pnx833x_gpio_select_input(unsigned int pin)
 | |
| {
 | |
| 	if (pin < 32)
 | |
| 		CLEAR_REG_BIT(PNX833X_PIO_DIR, pin);
 | |
| 	else
 | |
| 		CLEAR_REG_BIT(PNX833X_PIO_DIR2, pin & 31);
 | |
| }
 | |
| static inline void pnx833x_gpio_select_output(unsigned int pin)
 | |
| {
 | |
| 	if (pin < 32)
 | |
| 		SET_REG_BIT(PNX833X_PIO_DIR, pin);
 | |
| 	else
 | |
| 		SET_REG_BIT(PNX833X_PIO_DIR2, pin & 31);
 | |
| }
 | |
| 
 | |
| /* Select GPIO or alternate function for a pin */
 | |
| static inline void pnx833x_gpio_select_function_io(unsigned int pin)
 | |
| {
 | |
| 	if (pin < 32)
 | |
| 		CLEAR_REG_BIT(PNX833X_PIO_SEL, pin);
 | |
| 	else
 | |
| 		CLEAR_REG_BIT(PNX833X_PIO_SEL2, pin & 31);
 | |
| }
 | |
| static inline void pnx833x_gpio_select_function_alt(unsigned int pin)
 | |
| {
 | |
| 	if (pin < 32)
 | |
| 		SET_REG_BIT(PNX833X_PIO_SEL, pin);
 | |
| 	else
 | |
| 		SET_REG_BIT(PNX833X_PIO_SEL2, pin & 31);
 | |
| }
 | |
| 
 | |
| /* Read GPIO pin */
 | |
| static inline int pnx833x_gpio_read(unsigned int pin)
 | |
| {
 | |
| 	if (pin < 32)
 | |
| 		return (PNX833X_PIO_IN >> pin) & 1;
 | |
| 	else
 | |
| 		return (PNX833X_PIO_IN2 >> (pin & 31)) & 1;
 | |
| }
 | |
| 
 | |
| /* Write GPIO pin */
 | |
| static inline void pnx833x_gpio_write(unsigned int val, unsigned int pin)
 | |
| {
 | |
| 	if (pin < 32) {
 | |
| 		if (val)
 | |
| 			SET_REG_BIT(PNX833X_PIO_OUT, pin);
 | |
| 		else
 | |
| 			CLEAR_REG_BIT(PNX833X_PIO_OUT, pin);
 | |
| 	} else {
 | |
| 		if (val)
 | |
| 			SET_REG_BIT(PNX833X_PIO_OUT2, pin & 31);
 | |
| 		else
 | |
| 			CLEAR_REG_BIT(PNX833X_PIO_OUT2, pin & 31);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /* Configure GPIO interrupt */
 | |
| #define GPIO_INT_NONE		0
 | |
| #define GPIO_INT_LEVEL_LOW	1
 | |
| #define GPIO_INT_LEVEL_HIGH	2
 | |
| #define GPIO_INT_EDGE_RISING	3
 | |
| #define GPIO_INT_EDGE_FALLING	4
 | |
| #define GPIO_INT_EDGE_BOTH	5
 | |
| static inline void pnx833x_gpio_setup_irq(int when, unsigned int pin)
 | |
| {
 | |
| 	switch (when) {
 | |
| 	case GPIO_INT_LEVEL_LOW:
 | |
| 		CLEAR_REG_BIT(PNX833X_PIO_INT_EDGE, pin);
 | |
| 		CLEAR_REG_BIT(PNX833X_PIO_INT_HI, pin);
 | |
| 		SET_REG_BIT(PNX833X_PIO_INT_LO, pin);
 | |
| 		break;
 | |
| 	case GPIO_INT_LEVEL_HIGH:
 | |
| 		CLEAR_REG_BIT(PNX833X_PIO_INT_EDGE, pin);
 | |
| 		SET_REG_BIT(PNX833X_PIO_INT_HI, pin);
 | |
| 		CLEAR_REG_BIT(PNX833X_PIO_INT_LO, pin);
 | |
| 		break;
 | |
| 	case GPIO_INT_EDGE_RISING:
 | |
| 		SET_REG_BIT(PNX833X_PIO_INT_EDGE, pin);
 | |
| 		SET_REG_BIT(PNX833X_PIO_INT_HI, pin);
 | |
| 		CLEAR_REG_BIT(PNX833X_PIO_INT_LO, pin);
 | |
| 		break;
 | |
| 	case GPIO_INT_EDGE_FALLING:
 | |
| 		SET_REG_BIT(PNX833X_PIO_INT_EDGE, pin);
 | |
| 		CLEAR_REG_BIT(PNX833X_PIO_INT_HI, pin);
 | |
| 		SET_REG_BIT(PNX833X_PIO_INT_LO, pin);
 | |
| 		break;
 | |
| 	case GPIO_INT_EDGE_BOTH:
 | |
| 		SET_REG_BIT(PNX833X_PIO_INT_EDGE, pin);
 | |
| 		SET_REG_BIT(PNX833X_PIO_INT_HI, pin);
 | |
| 		SET_REG_BIT(PNX833X_PIO_INT_LO, pin);
 | |
| 		break;
 | |
| 	default:
 | |
| 		CLEAR_REG_BIT(PNX833X_PIO_INT_EDGE, pin);
 | |
| 		CLEAR_REG_BIT(PNX833X_PIO_INT_HI, pin);
 | |
| 		CLEAR_REG_BIT(PNX833X_PIO_INT_LO, pin);
 | |
| 		break;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /* Enable/disable GPIO interrupt */
 | |
| static inline void pnx833x_gpio_enable_irq(unsigned int pin)
 | |
| {
 | |
| 	SET_REG_BIT(PNX833X_PIO_INT_ENABLE, pin);
 | |
| }
 | |
| static inline void pnx833x_gpio_disable_irq(unsigned int pin)
 | |
| {
 | |
| 	CLEAR_REG_BIT(PNX833X_PIO_INT_ENABLE, pin);
 | |
| }
 | |
| 
 | |
| /* Clear GPIO interrupt request */
 | |
| static inline void pnx833x_gpio_clear_irq(unsigned int pin)
 | |
| {
 | |
| 	SET_REG_BIT(PNX833X_PIO_INT_CLEAR, pin);
 | |
| 	CLEAR_REG_BIT(PNX833X_PIO_INT_CLEAR, pin);
 | |
| }
 | |
| 
 | |
| #endif
 | 
