2007-05-08 12:58:36 +10:00
/*
* Copyright 2006 - 2007 , Michael Ellerman , IBM Corporation .
*
* 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 ; version 2 of the
* License .
*
*/
# include <linux/irq.h>
# include <linux/bitmap.h>
# include <linux/msi.h>
# include <asm/mpic.h>
# include <asm/prom.h>
# include <asm/hw_irq.h>
# include <asm/ppc-pci.h>
2008-08-06 09:10:03 +10:00
# include <asm/msi_bitmap.h>
2007-05-08 12:58:36 +10:00
2008-05-08 14:27:11 +10:00
# include <sysdev/mpic.h>
2007-05-08 12:58:36 +10:00
void mpic_msi_reserve_hwirq ( struct mpic * mpic , irq_hw_number_t hwirq )
{
/* The mpic calls this even when there is no allocator setup */
2008-08-06 09:10:03 +10:00
if ( ! mpic - > msi_bitmap . bitmap )
2007-05-08 12:58:36 +10:00
return ;
2008-08-06 09:10:03 +10:00
msi_bitmap_reserve_hwirq ( & mpic - > msi_bitmap , hwirq ) ;
2007-05-08 12:58:36 +10:00
}
# ifdef CONFIG_MPIC_U3_HT_IRQS
static int mpic_msi_reserve_u3_hwirqs ( struct mpic * mpic )
{
irq_hw_number_t hwirq ;
struct irq_host_ops * ops = mpic - > irqhost - > ops ;
struct device_node * np ;
int flags , index , i ;
struct of_irq oirq ;
pr_debug ( " mpic: found U3, guessing msi allocator setup \n " ) ;
2009-12-14 15:31:13 +00:00
/* Reserve source numbers we know are reserved in the HW.
*
* This is a bit of a mix of U3 and U4 reserves but that ' s going
* to work fine , we have plenty enugh numbers left so let ' s just
* mark anything we don ' t like reserved .
*/
2007-05-08 12:58:36 +10:00
for ( i = 0 ; i < 8 ; i + + )
2008-08-06 09:10:03 +10:00
msi_bitmap_reserve_hwirq ( & mpic - > msi_bitmap , i ) ;
2007-05-08 12:58:36 +10:00
for ( i = 42 ; i < 46 ; i + + )
2008-08-06 09:10:03 +10:00
msi_bitmap_reserve_hwirq ( & mpic - > msi_bitmap , i ) ;
2007-05-08 12:58:36 +10:00
for ( i = 100 ; i < 105 ; i + + )
2008-08-06 09:10:03 +10:00
msi_bitmap_reserve_hwirq ( & mpic - > msi_bitmap , i ) ;
2007-05-08 12:58:36 +10:00
2009-12-14 15:31:13 +00:00
for ( i = 124 ; i < mpic - > irq_count ; i + + )
msi_bitmap_reserve_hwirq ( & mpic - > msi_bitmap , i ) ;
2007-05-08 12:58:36 +10:00
np = NULL ;
while ( ( np = of_find_all_nodes ( np ) ) ) {
pr_debug ( " mpic: mapping hwirqs for %s \n " , np - > full_name ) ;
index = 0 ;
while ( of_irq_map_one ( np , index + + , & oirq ) = = 0 ) {
ops - > xlate ( mpic - > irqhost , NULL , oirq . specifier ,
oirq . size , & hwirq , & flags ) ;
2008-08-06 09:10:03 +10:00
msi_bitmap_reserve_hwirq ( & mpic - > msi_bitmap , hwirq ) ;
2007-05-08 12:58:36 +10:00
}
}
return 0 ;
}
# else
static int mpic_msi_reserve_u3_hwirqs ( struct mpic * mpic )
{
return - 1 ;
}
# endif
int mpic_msi_init_allocator ( struct mpic * mpic )
{
2008-08-06 09:10:03 +10:00
int rc ;
2007-05-08 12:58:36 +10:00
2008-08-06 09:10:03 +10:00
rc = msi_bitmap_alloc ( & mpic - > msi_bitmap , mpic - > irq_count ,
mpic - > irqhost - > of_node ) ;
if ( rc )
return rc ;
2007-05-08 12:58:36 +10:00
2008-08-06 09:10:03 +10:00
rc = msi_bitmap_reserve_dt_hwirqs ( & mpic - > msi_bitmap ) ;
if ( rc > 0 ) {
2007-05-08 12:58:36 +10:00
if ( mpic - > flags & MPIC_U3_HT_IRQS )
rc = mpic_msi_reserve_u3_hwirqs ( mpic ) ;
2008-08-06 09:10:03 +10:00
if ( rc ) {
msi_bitmap_free ( & mpic - > msi_bitmap ) ;
return rc ;
}
2007-05-08 12:58:36 +10:00
}
return 0 ;
}