2005-04-16 15:20:36 -07:00
/*
* arch / arm / mach - ixp4xx / common . c
*
* Generic code shared across all IXP4XX platforms
*
* Maintainer : Deepak Saxena < dsaxena @ plexity . net >
*
* Copyright 2002 ( c ) Intel Corporation
* Copyright 2003 - 2004 ( c ) MontaVista , Software , Inc .
*
* This file is licensed under the terms of the GNU General Public
* License version 2. This program is licensed " as is " without any
* warranty of any kind , whether express or implied .
*/
# include <linux/kernel.h>
# include <linux/mm.h>
# include <linux/init.h>
# include <linux/serial.h>
# include <linux/sched.h>
# include <linux/tty.h>
2005-10-29 19:07:23 +01:00
# include <linux/platform_device.h>
2005-04-16 15:20:36 -07:00
# include <linux/serial_core.h>
# include <linux/bootmem.h>
# include <linux/interrupt.h>
# include <linux/bitops.h>
# include <linux/time.h>
# include <linux/timex.h>
2006-09-22 00:58:57 +01:00
# include <linux/clocksource.h>
2007-03-08 20:23:59 +01:00
# include <linux/clockchips.h>
2005-04-16 15:20:36 -07:00
2008-08-05 16:14:15 +01:00
# include <mach/udc.h>
# include <mach/hardware.h>
2005-04-16 15:20:36 -07:00
# include <asm/uaccess.h>
# include <asm/io.h>
# include <asm/pgtable.h>
# include <asm/page.h>
# include <asm/irq.h>
# include <asm/mach/map.h>
# include <asm/mach/irq.h>
# include <asm/mach/time.h>
2006-12-06 00:45:07 +01:00
static int __init ixp4xx_clocksource_init ( void ) ;
2007-03-08 20:23:59 +01:00
static int __init ixp4xx_clockevent_init ( void ) ;
static struct clock_event_device clockevent_ixp4xx ;
2006-12-06 00:45:07 +01:00
2005-04-16 15:20:36 -07:00
/*************************************************************************
* IXP4xx chipset I / O mapping
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static struct map_desc ixp4xx_io_desc [ ] __initdata = {
{ /* UART, Interrupt ctrl, GPIO, timers, NPEs, MACs, USB .... */
. virtual = IXP4XX_PERIPHERAL_BASE_VIRT ,
2005-10-28 15:18:59 +01:00
. pfn = __phys_to_pfn ( IXP4XX_PERIPHERAL_BASE_PHYS ) ,
2005-04-16 15:20:36 -07:00
. length = IXP4XX_PERIPHERAL_REGION_SIZE ,
. type = MT_DEVICE
} , { /* Expansion Bus Config Registers */
. virtual = IXP4XX_EXP_CFG_BASE_VIRT ,
2005-10-28 15:18:59 +01:00
. pfn = __phys_to_pfn ( IXP4XX_EXP_CFG_BASE_PHYS ) ,
2005-04-16 15:20:36 -07:00
. length = IXP4XX_EXP_CFG_REGION_SIZE ,
. type = MT_DEVICE
} , { /* PCI Registers */
. virtual = IXP4XX_PCI_CFG_BASE_VIRT ,
2005-10-28 15:18:59 +01:00
. pfn = __phys_to_pfn ( IXP4XX_PCI_CFG_BASE_PHYS ) ,
2005-04-16 15:20:36 -07:00
. length = IXP4XX_PCI_CFG_REGION_SIZE ,
. type = MT_DEVICE
2005-06-24 20:54:35 +01:00
} ,
# ifdef CONFIG_DEBUG_LL
{ /* Debug UART mapping */
. virtual = IXP4XX_DEBUG_UART_BASE_VIRT ,
2005-10-28 15:18:59 +01:00
. pfn = __phys_to_pfn ( IXP4XX_DEBUG_UART_BASE_PHYS ) ,
2005-06-24 20:54:35 +01:00
. length = IXP4XX_DEBUG_UART_REGION_SIZE ,
. type = MT_DEVICE
2005-04-16 15:20:36 -07:00
}
2005-06-24 20:54:35 +01:00
# endif
2005-04-16 15:20:36 -07:00
} ;
void __init ixp4xx_map_io ( void )
{
iotable_init ( ixp4xx_io_desc , ARRAY_SIZE ( ixp4xx_io_desc ) ) ;
}
/*************************************************************************
* IXP4xx chipset IRQ handling
*
* TODO : GPIO IRQs should be marked invalid until the user of the IRQ
* ( be it PCI or something else ) configures that GPIO line
* as an IRQ .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-08-29 22:46:30 +01:00
enum ixp4xx_irq_type {
IXP4XX_IRQ_LEVEL , IXP4XX_IRQ_EDGE
} ;
2006-11-03 01:47:20 +01:00
/* Each bit represents an IRQ: 1: edge-triggered, 0: level triggered */
static unsigned long long ixp4xx_irq_edge = 0 ;
2005-08-29 22:46:30 +01:00
/*
* IRQ - > GPIO mapping table
*/
2006-04-20 21:24:38 +01:00
static signed char irq2gpio [ 32 ] = {
2005-08-29 22:46:30 +01:00
- 1 , - 1 , - 1 , - 1 , - 1 , - 1 , 0 , 1 ,
- 1 , - 1 , - 1 , - 1 , - 1 , - 1 , - 1 , - 1 ,
- 1 , - 1 , - 1 , 2 , 3 , 4 , 5 , 6 ,
7 , 8 , 9 , 10 , 11 , 12 , - 1 , - 1 ,
} ;
2007-03-21 14:04:08 +01:00
int gpio_to_irq ( int gpio )
{
int irq ;
for ( irq = 0 ; irq < 32 ; irq + + ) {
if ( irq2gpio [ irq ] = = gpio )
return irq ;
}
return - EINVAL ;
}
EXPORT_SYMBOL ( gpio_to_irq ) ;
int irq_to_gpio ( int irq )
{
int gpio = ( irq < 32 ) ? irq2gpio [ irq ] : - EINVAL ;
if ( gpio = = - 1 )
return - EINVAL ;
return gpio ;
}
EXPORT_SYMBOL ( irq_to_gpio ) ;
2005-08-29 22:46:30 +01:00
static int ixp4xx_set_irq_type ( unsigned int irq , unsigned int type )
{
int line = irq2gpio [ irq ] ;
u32 int_style ;
enum ixp4xx_irq_type irq_type ;
volatile u32 * int_reg ;
/*
* Only for GPIO IRQs
*/
if ( line < 0 )
return - EINVAL ;
2006-02-22 22:27:23 +00:00
switch ( type ) {
2008-07-27 04:23:31 +01:00
case IRQ_TYPE_EDGE_BOTH :
2005-08-29 22:46:30 +01:00
int_style = IXP4XX_GPIO_STYLE_TRANSITIONAL ;
irq_type = IXP4XX_IRQ_EDGE ;
2006-02-22 22:27:23 +00:00
break ;
2008-07-27 04:23:31 +01:00
case IRQ_TYPE_EDGE_RISING :
2005-08-29 22:46:30 +01:00
int_style = IXP4XX_GPIO_STYLE_RISING_EDGE ;
irq_type = IXP4XX_IRQ_EDGE ;
2006-02-22 22:27:23 +00:00
break ;
2008-07-27 04:23:31 +01:00
case IRQ_TYPE_EDGE_FALLING :
2005-08-29 22:46:30 +01:00
int_style = IXP4XX_GPIO_STYLE_FALLING_EDGE ;
irq_type = IXP4XX_IRQ_EDGE ;
2006-02-22 22:27:23 +00:00
break ;
2008-07-27 04:23:31 +01:00
case IRQ_TYPE_LEVEL_HIGH :
2005-08-29 22:46:30 +01:00
int_style = IXP4XX_GPIO_STYLE_ACTIVE_HIGH ;
irq_type = IXP4XX_IRQ_LEVEL ;
2006-02-22 22:27:23 +00:00
break ;
2008-07-27 04:23:31 +01:00
case IRQ_TYPE_LEVEL_LOW :
2005-08-29 22:46:30 +01:00
int_style = IXP4XX_GPIO_STYLE_ACTIVE_LOW ;
irq_type = IXP4XX_IRQ_LEVEL ;
2006-02-22 22:27:23 +00:00
break ;
default :
2005-09-26 19:52:56 +01:00
return - EINVAL ;
2006-02-22 22:27:23 +00:00
}
2006-11-03 01:47:20 +01:00
if ( irq_type = = IXP4XX_IRQ_EDGE )
ixp4xx_irq_edge | = ( 1 < < irq ) ;
else
ixp4xx_irq_edge & = ~ ( 1 < < irq ) ;
2005-08-29 22:46:30 +01:00
if ( line > = 8 ) { /* pins 8-15 */
line - = 8 ;
int_reg = IXP4XX_GPIO_GPIT2R ;
} else { /* pins 0-7 */
int_reg = IXP4XX_GPIO_GPIT1R ;
}
/* Clear the style for the appropriate pin */
* int_reg & = ~ ( IXP4XX_GPIO_STYLE_CLEAR < <
( line * IXP4XX_GPIO_STYLE_SIZE ) ) ;
2006-01-04 17:17:10 +00:00
* IXP4XX_GPIO_GPISR = ( 1 < < line ) ;
2005-08-29 22:46:30 +01:00
/* Set the new style */
* int_reg | = ( int_style < < ( line * IXP4XX_GPIO_STYLE_SIZE ) ) ;
2005-09-26 19:52:56 +01:00
2006-03-20 17:10:12 +00:00
/* Configure the line as an input */
2007-07-14 11:15:05 +02:00
gpio_line_config ( irq2gpio [ irq ] , IXP4XX_GPIO_IN ) ;
2006-03-20 17:10:12 +00:00
2005-09-26 19:52:56 +01:00
return 0 ;
2005-08-29 22:46:30 +01:00
}
2005-04-16 15:20:36 -07:00
static void ixp4xx_irq_mask ( unsigned int irq )
{
2007-04-06 15:00:31 +01:00
if ( ( cpu_is_ixp46x ( ) | | cpu_is_ixp43x ( ) ) & & irq > = 32 )
2005-04-16 15:20:36 -07:00
* IXP4XX_ICMR2 & = ~ ( 1 < < ( irq - 32 ) ) ;
else
* IXP4XX_ICMR & = ~ ( 1 < < irq ) ;
}
static void ixp4xx_irq_ack ( unsigned int irq )
{
int line = ( irq < 32 ) ? irq2gpio [ irq ] : - 1 ;
if ( line > = 0 )
2006-01-04 17:17:10 +00:00
* IXP4XX_GPIO_GPISR = ( 1 < < line ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Level triggered interrupts on GPIO lines can only be cleared when the
* interrupt condition disappears .
*/
2006-11-03 01:47:20 +01:00
static void ixp4xx_irq_unmask ( unsigned int irq )
2005-04-16 15:20:36 -07:00
{
2006-11-03 01:47:20 +01:00
if ( ! ( ixp4xx_irq_edge & ( 1 < < irq ) ) )
ixp4xx_irq_ack ( irq ) ;
2005-04-16 15:20:36 -07:00
2007-04-06 15:00:31 +01:00
if ( ( cpu_is_ixp46x ( ) | | cpu_is_ixp43x ( ) ) & & irq > = 32 )
2006-11-03 01:47:20 +01:00
* IXP4XX_ICMR2 | = ( 1 < < ( irq - 32 ) ) ;
else
* IXP4XX_ICMR | = ( 1 < < irq ) ;
}
2005-04-16 15:20:36 -07:00
2006-11-23 11:41:32 +00:00
static struct irq_chip ixp4xx_irq_chip = {
2006-11-03 01:47:20 +01:00
. name = " IXP4xx " ,
2005-09-06 23:13:17 +01:00
. ack = ixp4xx_irq_ack ,
. mask = ixp4xx_irq_mask ,
. unmask = ixp4xx_irq_unmask ,
. set_type = ixp4xx_set_irq_type ,
2005-04-16 15:20:36 -07:00
} ;
void __init ixp4xx_init_irq ( void )
{
int i = 0 ;
/* Route all sources to IRQ instead of FIQ */
* IXP4XX_ICLR = 0x0 ;
/* Disable all interrupt */
* IXP4XX_ICMR = 0x0 ;
2007-04-06 15:00:31 +01:00
if ( cpu_is_ixp46x ( ) | | cpu_is_ixp43x ( ) ) {
2005-04-16 15:20:36 -07:00
/* Route upper 32 sources to IRQ instead of FIQ */
* IXP4XX_ICLR2 = 0x00 ;
/* Disable upper 32 interrupts */
* IXP4XX_ICMR2 = 0x00 ;
}
/* Default to all level triggered */
2006-11-03 01:47:20 +01:00
for ( i = 0 ; i < NR_IRQS ; i + + ) {
set_irq_chip ( i , & ixp4xx_irq_chip ) ;
2006-11-23 11:41:32 +00:00
set_irq_handler ( i , handle_level_irq ) ;
2006-11-03 01:47:20 +01:00
set_irq_flags ( i , IRQF_VALID ) ;
}
2005-04-16 15:20:36 -07:00
}
/*************************************************************************
* IXP4xx timer tick
* We use OS timer1 on the CPU for the timer tick and the timestamp
* counter as a source of real clock ticks to account for missed jiffies .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-10-06 10:53:39 -07:00
static irqreturn_t ixp4xx_timer_interrupt ( int irq , void * dev_id )
2005-04-16 15:20:36 -07:00
{
2007-03-08 20:23:59 +01:00
struct clock_event_device * evt = & clockevent_ixp4xx ;
2005-04-16 15:20:36 -07:00
/* Clear Pending Interrupt by writing '1' to it */
* IXP4XX_OSST = IXP4XX_OSST_TIMER_1_PEND ;
2007-03-08 20:23:59 +01:00
evt - > event_handler ( evt ) ;
2005-04-16 15:20:36 -07:00
return IRQ_HANDLED ;
}
static struct irqaction ixp4xx_timer_irq = {
2007-03-08 20:23:59 +01:00
. name = " timer1 " ,
2007-05-08 00:35:39 -07:00
. flags = IRQF_DISABLED | IRQF_TIMER | IRQF_IRQPOLL ,
2005-06-26 17:06:36 +01:00
. handler = ixp4xx_timer_interrupt ,
2005-04-16 15:20:36 -07:00
} ;
2007-05-23 22:38:45 +01:00
void __init ixp4xx_timer_init ( void )
2005-04-16 15:20:36 -07:00
{
2007-03-08 20:23:59 +01:00
/* Reset/disable counter */
* IXP4XX_OSRT1 = 0 ;
2005-04-16 15:20:36 -07:00
/* Clear Pending Interrupt by writing '1' to it */
* IXP4XX_OSST = IXP4XX_OSST_TIMER_1_PEND ;
/* Reset time-stamp counter */
* IXP4XX_OSTS = 0 ;
/* Connect the interrupt handler and enable the interrupt */
setup_irq ( IRQ_IXP4XX_TIMER1 , & ixp4xx_timer_irq ) ;
2006-12-06 00:45:07 +01:00
ixp4xx_clocksource_init ( ) ;
2007-03-08 20:23:59 +01:00
ixp4xx_clockevent_init ( ) ;
2005-04-16 15:20:36 -07:00
}
struct sys_timer ixp4xx_timer = {
. init = ixp4xx_timer_init ,
} ;
2006-12-01 11:36:41 +01:00
static struct pxa2xx_udc_mach_info ixp4xx_udc_info ;
void __init ixp4xx_set_udc_info ( struct pxa2xx_udc_mach_info * info )
{
memcpy ( & ixp4xx_udc_info , info , sizeof * info ) ;
}
static struct resource ixp4xx_udc_resources [ ] = {
[ 0 ] = {
. start = 0xc800b000 ,
. end = 0xc800bfff ,
. flags = IORESOURCE_MEM ,
} ,
[ 1 ] = {
. start = IRQ_IXP4XX_USB ,
. end = IRQ_IXP4XX_USB ,
. flags = IORESOURCE_IRQ ,
} ,
} ;
/*
2008-06-22 23:36:39 +01:00
* USB device controller . The IXP4xx uses the same controller as PXA25X ,
2006-12-01 11:36:41 +01:00
* so we just use the same device .
*/
static struct platform_device ixp4xx_udc_device = {
2008-06-22 23:36:39 +01:00
. name = " pxa25x-udc " ,
2006-12-01 11:36:41 +01:00
. id = - 1 ,
. num_resources = 2 ,
. resource = ixp4xx_udc_resources ,
. dev = {
. platform_data = & ixp4xx_udc_info ,
} ,
} ;
static struct platform_device * ixp4xx_devices [ ] __initdata = {
& ixp4xx_udc_device ,
} ;
2005-04-16 15:20:36 -07:00
static struct resource ixp46x_i2c_resources [ ] = {
[ 0 ] = {
. start = 0xc8011000 ,
. end = 0xc801101c ,
. flags = IORESOURCE_MEM ,
} ,
[ 1 ] = {
. start = IRQ_IXP4XX_I2C ,
. end = IRQ_IXP4XX_I2C ,
. flags = IORESOURCE_IRQ
}
} ;
/*
* I2C controller . The IXP46x uses the same block as the IOP3xx , so
* we just use the same device name .
*/
static struct platform_device ixp46x_i2c_controller = {
. name = " IOP3xx-I2C " ,
. id = 0 ,
. num_resources = 2 ,
. resource = ixp46x_i2c_resources
} ;
static struct platform_device * ixp46x_devices [ ] __initdata = {
& ixp46x_i2c_controller
} ;
2006-01-05 20:59:29 +00:00
unsigned long ixp4xx_exp_bus_size ;
2006-01-18 22:46:43 +00:00
EXPORT_SYMBOL ( ixp4xx_exp_bus_size ) ;
2006-01-05 20:59:29 +00:00
2005-04-16 15:20:36 -07:00
void __init ixp4xx_sys_init ( void )
{
2006-01-05 20:59:29 +00:00
ixp4xx_exp_bus_size = SZ_16M ;
2006-12-01 11:36:41 +01:00
platform_add_devices ( ixp4xx_devices , ARRAY_SIZE ( ixp4xx_devices ) ) ;
2005-04-16 15:20:36 -07:00
if ( cpu_is_ixp46x ( ) ) {
2006-01-05 20:59:29 +00:00
int region ;
2005-04-16 15:20:36 -07:00
platform_add_devices ( ixp46x_devices ,
ARRAY_SIZE ( ixp46x_devices ) ) ;
2006-01-05 20:59:29 +00:00
for ( region = 0 ; region < 7 ; region + + ) {
if ( ( * ( IXP4XX_EXP_REG ( 0x4 * region ) ) & 0x200 ) ) {
ixp4xx_exp_bus_size = SZ_32M ;
break ;
}
}
2005-04-16 15:20:36 -07:00
}
2006-01-05 20:59:29 +00:00
2006-01-18 22:46:43 +00:00
printk ( " IXP4xx: Using %luMiB expansion bus window size \n " ,
2006-01-05 20:59:29 +00:00
ixp4xx_exp_bus_size > > 20 ) ;
2005-04-16 15:20:36 -07:00
}
2007-03-08 20:23:59 +01:00
/*
* clocksource
*/
2006-09-22 00:58:57 +01:00
cycle_t ixp4xx_get_cycles ( void )
{
return * IXP4XX_OSTS ;
}
static struct clocksource clocksource_ixp4xx = {
. name = " OSTS " ,
. rating = 200 ,
. read = ixp4xx_get_cycles ,
. mask = CLOCKSOURCE_MASK ( 32 ) ,
. shift = 20 ,
2007-02-16 01:27:37 -08:00
. flags = CLOCK_SOURCE_IS_CONTINUOUS ,
2006-09-22 00:58:57 +01:00
} ;
unsigned long ixp4xx_timer_freq = FREQ ;
static int __init ixp4xx_clocksource_init ( void )
{
clocksource_ixp4xx . mult =
clocksource_hz2mult ( ixp4xx_timer_freq ,
clocksource_ixp4xx . shift ) ;
clocksource_register ( & clocksource_ixp4xx ) ;
return 0 ;
}
2007-03-08 20:23:59 +01:00
/*
* clockevents
*/
static int ixp4xx_set_next_event ( unsigned long evt ,
struct clock_event_device * unused )
{
unsigned long opts = * IXP4XX_OSRT1 & IXP4XX_OST_RELOAD_MASK ;
* IXP4XX_OSRT1 = ( evt & ~ IXP4XX_OST_RELOAD_MASK ) | opts ;
return 0 ;
}
static void ixp4xx_set_mode ( enum clock_event_mode mode ,
struct clock_event_device * evt )
{
2007-12-12 00:32:58 +01:00
unsigned long opts = * IXP4XX_OSRT1 & IXP4XX_OST_RELOAD_MASK ;
unsigned long osrt = * IXP4XX_OSRT1 & ~ IXP4XX_OST_RELOAD_MASK ;
2007-03-08 20:23:59 +01:00
switch ( mode ) {
case CLOCK_EVT_MODE_PERIODIC :
osrt = LATCH & ~ IXP4XX_OST_RELOAD_MASK ;
opts = IXP4XX_OST_ENABLE ;
break ;
case CLOCK_EVT_MODE_ONESHOT :
/* period set by 'set next_event' */
osrt = 0 ;
opts = IXP4XX_OST_ENABLE | IXP4XX_OST_ONE_SHOT ;
break ;
case CLOCK_EVT_MODE_SHUTDOWN :
2007-12-12 00:32:58 +01:00
opts & = ~ IXP4XX_OST_ENABLE ;
break ;
case CLOCK_EVT_MODE_RESUME :
opts | = IXP4XX_OST_ENABLE ;
break ;
2007-03-08 20:23:59 +01:00
case CLOCK_EVT_MODE_UNUSED :
default :
osrt = opts = 0 ;
break ;
}
* IXP4XX_OSRT1 = osrt | opts ;
}
static struct clock_event_device clockevent_ixp4xx = {
. name = " ixp4xx timer1 " ,
. features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT ,
. rating = 200 ,
. shift = 24 ,
. set_mode = ixp4xx_set_mode ,
. set_next_event = ixp4xx_set_next_event ,
} ;
static int __init ixp4xx_clockevent_init ( void )
{
clockevent_ixp4xx . mult = div_sc ( FREQ , NSEC_PER_SEC ,
clockevent_ixp4xx . shift ) ;
clockevent_ixp4xx . max_delta_ns =
clockevent_delta2ns ( 0xfffffffe , & clockevent_ixp4xx ) ;
clockevent_ixp4xx . min_delta_ns =
clockevent_delta2ns ( 0xf , & clockevent_ixp4xx ) ;
clockevent_ixp4xx . cpumask = cpumask_of_cpu ( 0 ) ;
clockevents_register_device ( & clockevent_ixp4xx ) ;
return 0 ;
}