2005-11-15 15:53:48 -05:00
/*
* Low - level SPU handling
*
* ( C ) Copyright IBM Deutschland Entwicklung GmbH 2005
*
* Author : Arnd Bergmann < arndb @ de . ibm . 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 , 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 0213 9 , USA .
*/
2005-12-05 22:52:24 -05:00
# undef DEBUG
2005-11-15 15:53:48 -05:00
# include <linux/interrupt.h>
# include <linux/list.h>
# include <linux/module.h>
# include <linux/ptrace.h>
# include <linux/slab.h>
# include <linux/wait.h>
2006-11-23 00:46:49 +01:00
# include <linux/mm.h>
# include <linux/io.h>
2006-03-26 01:37:14 -08:00
# include <linux/mutex.h>
2007-07-17 04:05:52 -07:00
# include <linux/linux_logo.h>
2005-11-15 15:53:48 -05:00
# include <asm/spu.h>
2006-06-19 20:33:29 +02:00
# include <asm/spu_priv1.h>
2007-12-05 13:49:31 +11:00
# include <asm/spu_csa.h>
2006-10-24 18:31:27 +02:00
# include <asm/xmon.h>
2007-07-20 21:39:46 +02:00
# include <asm/prom.h>
2005-11-15 15:53:48 -05:00
2006-11-23 00:46:49 +01:00
const struct spu_management_ops * spu_management_ops ;
2007-04-23 21:08:29 +02:00
EXPORT_SYMBOL_GPL ( spu_management_ops ) ;
2006-06-19 20:33:29 +02:00
const struct spu_priv1_ops * spu_priv1_ops ;
2007-07-20 21:39:51 +02:00
EXPORT_SYMBOL_GPL ( spu_priv1_ops ) ;
2006-06-19 20:33:29 +02:00
2007-07-20 21:39:51 +02:00
struct cbe_spu_info cbe_spu_info [ MAX_NUMNODES ] ;
EXPORT_SYMBOL_GPL ( cbe_spu_info ) ;
2007-03-10 00:05:37 +01:00
2007-12-05 13:49:31 +11:00
/*
* The spufs fault - handling code needs to call force_sig_info to raise signals
* on DMA errors . Export it here to avoid general kernel - wide access to this
* function
*/
EXPORT_SYMBOL_GPL ( force_sig_info ) ;
2007-07-20 21:39:51 +02:00
/*
* Protects cbe_spu_info and spu - > number .
*/
static DEFINE_SPINLOCK ( spu_lock ) ;
/*
* List of all spus in the system .
*
* This list is iterated by callers from irq context and callers that
* want to sleep . Thus modifications need to be done with both
* spu_full_list_lock and spu_full_list_mutex held , while iterating
* through it requires either of these locks .
*
* In addition spu_full_list_lock protects all assignmens to
* spu - > mm .
*/
static LIST_HEAD ( spu_full_list ) ;
static DEFINE_SPINLOCK ( spu_full_list_lock ) ;
static DEFINE_MUTEX ( spu_full_list_mutex ) ;
2006-06-19 20:33:29 +02:00
2007-12-05 13:49:31 +11:00
struct spu_slb {
u64 esid , vsid ;
} ;
2007-03-10 00:05:37 +01:00
void spu_invalidate_slbs ( struct spu * spu )
{
struct spu_priv2 __iomem * priv2 = spu - > priv2 ;
if ( spu_mfc_sr1_get ( spu ) & MFC_STATE1_RELOCATE_MASK )
out_be64 ( & priv2 - > slb_invalidate_all_W , 0UL ) ;
}
EXPORT_SYMBOL_GPL ( spu_invalidate_slbs ) ;
/* This is called by the MM core when a segment size is changed, to
* request a flush of all the SPEs using a given mm
*/
void spu_flush_all_slbs ( struct mm_struct * mm )
{
struct spu * spu ;
unsigned long flags ;
2007-07-20 21:39:51 +02:00
spin_lock_irqsave ( & spu_full_list_lock , flags ) ;
2007-03-10 00:05:37 +01:00
list_for_each_entry ( spu , & spu_full_list , full_list ) {
if ( spu - > mm = = mm )
spu_invalidate_slbs ( spu ) ;
}
2007-07-20 21:39:51 +02:00
spin_unlock_irqrestore ( & spu_full_list_lock , flags ) ;
2007-03-10 00:05:37 +01:00
}
/* The hack below stinks... try to do something better one of
* these days . . . Does it even work properly with NR_CPUS = = 1 ?
*/
static inline void mm_needs_global_tlbie ( struct mm_struct * mm )
{
int nr = ( NR_CPUS > 1 ) ? NR_CPUS : NR_CPUS + 1 ;
/* Global TLBIE broadcast required with SPEs. */
__cpus_setall ( & mm - > cpu_vm_mask , nr ) ;
}
void spu_associate_mm ( struct spu * spu , struct mm_struct * mm )
{
unsigned long flags ;
2007-07-20 21:39:51 +02:00
spin_lock_irqsave ( & spu_full_list_lock , flags ) ;
2007-03-10 00:05:37 +01:00
spu - > mm = mm ;
2007-07-20 21:39:51 +02:00
spin_unlock_irqrestore ( & spu_full_list_lock , flags ) ;
2007-03-10 00:05:37 +01:00
if ( mm )
mm_needs_global_tlbie ( mm ) ;
}
EXPORT_SYMBOL_GPL ( spu_associate_mm ) ;
2007-12-05 13:49:31 +11:00
int spu_64k_pages_available ( void )
{
return mmu_psize_defs [ MMU_PAGE_64K ] . shift ! = 0 ;
}
EXPORT_SYMBOL_GPL ( spu_64k_pages_available ) ;
2005-11-15 15:53:48 -05:00
static void spu_restart_dma ( struct spu * spu )
{
struct spu_priv2 __iomem * priv2 = spu - > priv2 ;
2005-11-15 15:53:49 -05:00
2006-01-04 20:31:28 +01:00
if ( ! test_bit ( SPU_CONTEXT_SWITCH_PENDING , & spu - > flags ) )
2005-11-15 15:53:49 -05:00
out_be64 ( & priv2 - > mfc_control_RW , MFC_CNTL_RESTART_DMA_COMMAND ) ;
2005-11-15 15:53:48 -05:00
}
2007-12-05 13:49:31 +11:00
static inline void spu_load_slb ( struct spu * spu , int slbe , struct spu_slb * slb )
{
struct spu_priv2 __iomem * priv2 = spu - > priv2 ;
pr_debug ( " %s: adding SLB[%d] 0x%016lx 0x%016lx \n " ,
__func__ , slbe , slb - > vsid , slb - > esid ) ;
out_be64 ( & priv2 - > slb_index_W , slbe ) ;
out_be64 ( & priv2 - > slb_vsid_RW , slb - > vsid ) ;
out_be64 ( & priv2 - > slb_esid_RW , slb - > esid ) ;
}
2005-11-15 15:53:48 -05:00
static int __spu_trap_data_seg ( struct spu * spu , unsigned long ea )
{
2005-11-15 15:53:52 -05:00
struct mm_struct * mm = spu - > mm ;
2007-12-05 13:49:31 +11:00
struct spu_slb slb ;
2007-03-10 00:05:37 +01:00
int psize ;
2005-11-15 15:53:48 -05:00
pr_debug ( " %s \n " , __FUNCTION__ ) ;
2006-01-04 20:31:28 +01:00
if ( test_bit ( SPU_CONTEXT_SWITCH_ACTIVE , & spu - > flags ) ) {
2005-11-15 15:53:52 -05:00
/* SLBs are pre-loaded for context switch, so
* we should never get here !
*/
2005-11-15 15:53:49 -05:00
printk ( " %s: invalid access during switch! \n " , __func__ ) ;
return 1 ;
}
2007-12-05 13:49:31 +11:00
slb . esid = ( ea & ESID_MASK ) | SLB_ESID_V ;
2006-10-24 18:31:18 +02:00
switch ( REGION_ID ( ea ) ) {
case USER_REGION_ID :
2007-05-08 16:27:27 +10:00
# ifdef CONFIG_PPC_MM_SLICES
psize = get_slice_psize ( mm , ea ) ;
# else
psize = mm - > context . user_psize ;
2006-10-24 18:31:18 +02:00
# endif
2007-12-05 13:49:31 +11:00
slb . vsid = ( get_vsid ( mm - > context . id , ea , MMU_SEGSIZE_256M )
< < SLB_VSID_SHIFT ) | SLB_VSID_USER ;
2006-10-24 18:31:18 +02:00
break ;
case VMALLOC_REGION_ID :
2007-03-10 00:05:37 +01:00
if ( ea < VMALLOC_END )
psize = mmu_vmalloc_psize ;
else
psize = mmu_io_psize ;
2007-12-05 13:49:31 +11:00
slb . vsid = ( get_kernel_vsid ( ea , MMU_SEGSIZE_256M )
< < SLB_VSID_SHIFT ) | SLB_VSID_KERNEL ;
2006-10-24 18:31:18 +02:00
break ;
case KERNEL_REGION_ID :
2007-03-10 00:05:37 +01:00
psize = mmu_linear_psize ;
2007-12-05 13:49:31 +11:00
slb . vsid = ( get_kernel_vsid ( ea , MMU_SEGSIZE_256M )
< < SLB_VSID_SHIFT ) | SLB_VSID_KERNEL ;
2006-10-24 18:31:18 +02:00
break ;
default :
2005-11-15 15:53:52 -05:00
/* Future: support kernel segments so that drivers
* can use SPUs .
*/
2005-11-15 15:53:48 -05:00
pr_debug ( " invalid region access at %016lx \n " , ea ) ;
return 1 ;
}
2007-12-05 13:49:31 +11:00
slb . vsid | = mmu_psize_defs [ psize ] . sllp ;
2005-11-15 15:53:48 -05:00
2007-12-05 13:49:31 +11:00
spu_load_slb ( spu , spu - > slb_replace , & slb ) ;
2005-11-15 15:53:52 -05:00
spu - > slb_replace + + ;
2005-11-15 15:53:48 -05:00
if ( spu - > slb_replace > = 8 )
spu - > slb_replace = 0 ;
spu_restart_dma ( spu ) ;
2007-06-29 10:58:03 +10:00
spu - > stats . slb_flt + + ;
2005-11-15 15:53:48 -05:00
return 0 ;
}
2005-11-15 15:53:49 -05:00
extern int hash_page ( unsigned long ea , unsigned long access , unsigned long trap ) ; //XXX
2005-11-15 15:53:52 -05:00
static int __spu_trap_data_map ( struct spu * spu , unsigned long ea , u64 dsisr )
2005-11-15 15:53:48 -05:00
{
2006-03-23 00:00:11 +01:00
pr_debug ( " %s, %lx, %lx \n " , __FUNCTION__ , dsisr , ea ) ;
2005-11-15 15:53:48 -05:00
2005-11-15 15:53:49 -05:00
/* Handle kernel space hash faults immediately.
User hash faults need to be deferred to process context . */
if ( ( dsisr & MFC_DSISR_PTE_NOT_FOUND )
& & REGION_ID ( ea ) ! = USER_REGION_ID
& & hash_page ( ea , _PAGE_PRESENT , 0x300 ) = = 0 ) {
spu_restart_dma ( spu ) ;
return 0 ;
}
2006-01-04 20:31:28 +01:00
if ( test_bit ( SPU_CONTEXT_SWITCH_ACTIVE , & spu - > flags ) ) {
2005-11-15 15:53:49 -05:00
printk ( " %s: invalid access during switch! \n " , __func__ ) ;
return 1 ;
}
2005-11-15 15:53:48 -05:00
2007-12-20 16:39:59 +09:00
spu - > class_0_pending = 0 ;
2005-11-15 15:53:52 -05:00
spu - > dar = ea ;
spu - > dsisr = dsisr ;
2007-12-20 16:39:59 +09:00
2006-06-19 20:33:33 +02:00
spu - > stop_callback ( spu ) ;
2007-12-20 16:39:59 +09:00
2005-11-15 15:53:48 -05:00
return 0 ;
}
2007-12-05 13:49:31 +11:00
static void __spu_kernel_slb ( void * addr , struct spu_slb * slb )
{
unsigned long ea = ( unsigned long ) addr ;
u64 llp ;
if ( REGION_ID ( ea ) = = KERNEL_REGION_ID )
llp = mmu_psize_defs [ mmu_linear_psize ] . sllp ;
else
llp = mmu_psize_defs [ mmu_virtual_psize ] . sllp ;
slb - > vsid = ( get_kernel_vsid ( ea , MMU_SEGSIZE_256M ) < < SLB_VSID_SHIFT ) |
SLB_VSID_KERNEL | llp ;
slb - > esid = ( ea & ESID_MASK ) | SLB_ESID_V ;
}
2007-12-05 13:49:31 +11:00
/**
* Given an array of @ nr_slbs SLB entries , @ slbs , return non - zero if the
* address @ new_addr is present .
*/
static inline int __slb_present ( struct spu_slb * slbs , int nr_slbs ,
void * new_addr )
{
unsigned long ea = ( unsigned long ) new_addr ;
int i ;
for ( i = 0 ; i < nr_slbs ; i + + )
if ( ! ( ( slbs [ i ] . esid ^ ea ) & ESID_MASK ) )
return 1 ;
return 0 ;
}
2007-12-05 13:49:31 +11:00
/**
* Setup the SPU kernel SLBs , in preparation for a context save / restore . We
* need to map both the context save area , and the save / restore code .
2007-12-05 13:49:31 +11:00
*
* Because the lscsa and code may cross segment boundaires , we check to see
* if mappings are required for the start and end of each range . We currently
* assume that the mappings are smaller that one segment - if not , something
* is seriously wrong .
2007-12-05 13:49:31 +11:00
*/
2007-12-05 13:49:31 +11:00
void spu_setup_kernel_slbs ( struct spu * spu , struct spu_lscsa * lscsa ,
void * code , int code_size )
2007-12-05 13:49:31 +11:00
{
2007-12-05 13:49:31 +11:00
struct spu_slb slbs [ 4 ] ;
int i , nr_slbs = 0 ;
/* start and end addresses of both mappings */
void * addrs [ ] = {
lscsa , ( void * ) lscsa + sizeof ( * lscsa ) - 1 ,
code , code + code_size - 1
} ;
/* check the set of addresses, and create a new entry in the slbs array
* if there isn ' t already a SLB for that address */
for ( i = 0 ; i < ARRAY_SIZE ( addrs ) ; i + + ) {
if ( __slb_present ( slbs , nr_slbs , addrs [ i ] ) )
continue ;
__spu_kernel_slb ( addrs [ i ] , & slbs [ nr_slbs ] ) ;
nr_slbs + + ;
}
2007-12-05 13:49:31 +11:00
2007-12-05 13:49:31 +11:00
/* Add the set of SLBs */
for ( i = 0 ; i < nr_slbs ; i + + )
spu_load_slb ( spu , i , & slbs [ i ] ) ;
2007-12-05 13:49:31 +11:00
}
EXPORT_SYMBOL_GPL ( spu_setup_kernel_slbs ) ;
2005-11-15 15:53:48 -05:00
static irqreturn_t
2006-10-06 22:52:16 +02:00
spu_irq_class_0 ( int irq , void * data )
2005-11-15 15:53:48 -05:00
{
struct spu * spu ;
2007-09-07 18:28:27 +10:00
unsigned long stat , mask ;
2005-11-15 15:53:48 -05:00
spu = data ;
2007-09-07 18:28:27 +10:00
2007-12-20 16:39:59 +09:00
spin_lock ( & spu - > register_lock ) ;
2007-09-07 18:28:27 +10:00
mask = spu_int_mask_get ( spu , 0 ) ;
2007-12-20 16:39:59 +09:00
stat = spu_int_stat_get ( spu , 0 ) & mask ;
2007-09-07 18:28:27 +10:00
spu - > class_0_pending | = stat ;
2007-12-20 16:39:59 +09:00
spu - > dsisr = spu_mfc_dsisr_get ( spu ) ;
spu - > dar = spu_mfc_dar_get ( spu ) ;
2007-09-07 18:28:27 +10:00
spin_unlock ( & spu - > register_lock ) ;
2006-06-19 20:33:33 +02:00
spu - > stop_callback ( spu ) ;
2005-11-15 15:53:48 -05:00
2007-09-07 18:28:27 +10:00
spu_int_stat_clear ( spu , 0 , stat ) ;
2005-11-15 15:53:48 -05:00
return IRQ_HANDLED ;
}
static irqreturn_t
2006-10-06 22:52:16 +02:00
spu_irq_class_1 ( int irq , void * data )
2005-11-15 15:53:48 -05:00
{
struct spu * spu ;
2005-11-15 15:53:52 -05:00
unsigned long stat , mask , dar , dsisr ;
2005-11-15 15:53:48 -05:00
spu = data ;
2005-11-15 15:53:52 -05:00
/* atomically read & clear class1 status. */
spin_lock ( & spu - > register_lock ) ;
2006-01-04 20:31:30 +01:00
mask = spu_int_mask_get ( spu , 1 ) ;
stat = spu_int_stat_get ( spu , 1 ) & mask ;
dar = spu_mfc_dar_get ( spu ) ;
dsisr = spu_mfc_dsisr_get ( spu ) ;
2007-12-20 16:39:59 +09:00
if ( stat & CLASS1_STORAGE_FAULT_INTR )
2006-01-04 20:31:30 +01:00
spu_mfc_dsisr_set ( spu , 0ul ) ;
spu_int_stat_clear ( spu , 1 , stat ) ;
2005-11-15 15:53:52 -05:00
spin_unlock ( & spu - > register_lock ) ;
2006-03-23 00:00:11 +01:00
pr_debug ( " %s: %lx %lx %lx %lx \n " , __FUNCTION__ , mask , stat ,
dar , dsisr ) ;
2005-11-15 15:53:48 -05:00
2007-12-20 16:39:59 +09:00
if ( stat & CLASS1_SEGMENT_FAULT_INTR )
2005-11-15 15:53:48 -05:00
__spu_trap_data_seg ( spu , dar ) ;
2007-12-20 16:39:59 +09:00
if ( stat & CLASS1_STORAGE_FAULT_INTR )
2005-11-15 15:53:52 -05:00
__spu_trap_data_map ( spu , dar , dsisr ) ;
2005-11-15 15:53:48 -05:00
2007-12-20 16:39:59 +09:00
if ( stat & CLASS1_LS_COMPARE_SUSPEND_ON_GET_INTR )
2005-11-15 15:53:48 -05:00
;
2007-12-20 16:39:59 +09:00
if ( stat & CLASS1_LS_COMPARE_SUSPEND_ON_PUT_INTR )
2005-11-15 15:53:48 -05:00
;
return stat ? IRQ_HANDLED : IRQ_NONE ;
}
static irqreturn_t
2006-10-06 22:52:16 +02:00
spu_irq_class_2 ( int irq , void * data )
2005-11-15 15:53:48 -05:00
{
struct spu * spu ;
unsigned long stat ;
2005-12-05 22:52:27 -05:00
unsigned long mask ;
2007-12-20 16:39:59 +09:00
const int mailbox_intrs =
CLASS2_MAILBOX_THRESHOLD_INTR | CLASS2_MAILBOX_INTR ;
2005-11-15 15:53:48 -05:00
spu = data ;
2006-06-19 20:33:33 +02:00
spin_lock ( & spu - > register_lock ) ;
2006-01-04 20:31:30 +01:00
stat = spu_int_stat_get ( spu , 2 ) ;
mask = spu_int_mask_get ( spu , 2 ) ;
2006-06-19 20:33:33 +02:00
/* ignore interrupts we're not waiting for */
stat & = mask ;
2007-12-20 16:39:59 +09:00
/* mailbox interrupts are level triggered. mask them now before
* acknowledging */
if ( stat & mailbox_intrs )
spu_int_mask_and ( spu , 2 , ~ ( stat & mailbox_intrs ) ) ;
2006-06-19 20:33:33 +02:00
/* acknowledge all interrupts before the callbacks */
spu_int_stat_clear ( spu , 2 , stat ) ;
spin_unlock ( & spu - > register_lock ) ;
2005-11-15 15:53:48 -05:00
2005-12-05 22:52:27 -05:00
pr_debug ( " class 2 interrupt %d, %lx, %lx \n " , irq , stat , mask ) ;
2005-11-15 15:53:48 -05:00
2007-12-20 16:39:59 +09:00
if ( stat & CLASS2_MAILBOX_INTR )
2006-06-19 20:33:33 +02:00
spu - > ibox_callback ( spu ) ;
2005-11-15 15:53:48 -05:00
2007-12-20 16:39:59 +09:00
if ( stat & CLASS2_SPU_STOP_INTR )
2006-06-19 20:33:33 +02:00
spu - > stop_callback ( spu ) ;
2005-11-15 15:53:48 -05:00
2007-12-20 16:39:59 +09:00
if ( stat & CLASS2_SPU_HALT_INTR )
2006-06-19 20:33:33 +02:00
spu - > stop_callback ( spu ) ;
2005-11-15 15:53:48 -05:00
2007-12-20 16:39:59 +09:00
if ( stat & CLASS2_SPU_DMA_TAG_GROUP_COMPLETE_INTR )
2006-06-19 20:33:33 +02:00
spu - > mfc_callback ( spu ) ;
2005-11-15 15:53:48 -05:00
2007-12-20 16:39:59 +09:00
if ( stat & CLASS2_MAILBOX_THRESHOLD_INTR )
2006-06-19 20:33:33 +02:00
spu - > wbox_callback ( spu ) ;
2005-11-15 15:53:48 -05:00
2007-06-29 10:58:03 +10:00
spu - > stats . class2_intr + + ;
2005-11-15 15:53:48 -05:00
return stat ? IRQ_HANDLED : IRQ_NONE ;
}
2006-07-03 21:36:01 +10:00
static int spu_request_irqs ( struct spu * spu )
2005-11-15 15:53:48 -05:00
{
2006-07-03 21:36:01 +10:00
int ret = 0 ;
2005-11-15 15:53:48 -05:00
2006-07-03 21:36:01 +10:00
if ( spu - > irqs [ 0 ] ! = NO_IRQ ) {
snprintf ( spu - > irq_c0 , sizeof ( spu - > irq_c0 ) , " spe%02d.0 " ,
spu - > number ) ;
ret = request_irq ( spu - > irqs [ 0 ] , spu_irq_class_0 ,
IRQF_DISABLED ,
spu - > irq_c0 , spu ) ;
if ( ret )
goto bail0 ;
}
if ( spu - > irqs [ 1 ] ! = NO_IRQ ) {
snprintf ( spu - > irq_c1 , sizeof ( spu - > irq_c1 ) , " spe%02d.1 " ,
spu - > number ) ;
ret = request_irq ( spu - > irqs [ 1 ] , spu_irq_class_1 ,
IRQF_DISABLED ,
spu - > irq_c1 , spu ) ;
if ( ret )
goto bail1 ;
}
if ( spu - > irqs [ 2 ] ! = NO_IRQ ) {
snprintf ( spu - > irq_c2 , sizeof ( spu - > irq_c2 ) , " spe%02d.2 " ,
spu - > number ) ;
ret = request_irq ( spu - > irqs [ 2 ] , spu_irq_class_2 ,
IRQF_DISABLED ,
spu - > irq_c2 , spu ) ;
if ( ret )
goto bail2 ;
}
return 0 ;
2005-11-15 15:53:48 -05:00
2006-07-03 21:36:01 +10:00
bail2 :
if ( spu - > irqs [ 1 ] ! = NO_IRQ )
free_irq ( spu - > irqs [ 1 ] , spu ) ;
bail1 :
if ( spu - > irqs [ 0 ] ! = NO_IRQ )
free_irq ( spu - > irqs [ 0 ] , spu ) ;
bail0 :
2005-11-15 15:53:48 -05:00
return ret ;
}
2006-07-03 21:36:01 +10:00
static void spu_free_irqs ( struct spu * spu )
2005-11-15 15:53:48 -05:00
{
2006-07-03 21:36:01 +10:00
if ( spu - > irqs [ 0 ] ! = NO_IRQ )
free_irq ( spu - > irqs [ 0 ] , spu ) ;
if ( spu - > irqs [ 1 ] ! = NO_IRQ )
free_irq ( spu - > irqs [ 1 ] , spu ) ;
if ( spu - > irqs [ 2 ] ! = NO_IRQ )
free_irq ( spu - > irqs [ 2 ] , spu ) ;
2005-11-15 15:53:48 -05:00
}
2007-07-20 21:39:54 +02:00
void spu_init_channels ( struct spu * spu )
2005-11-15 15:53:48 -05:00
{
static const struct {
unsigned channel ;
unsigned count ;
} zero_list [ ] = {
{ 0x00 , 1 , } , { 0x01 , 1 , } , { 0x03 , 1 , } , { 0x04 , 1 , } ,
{ 0x18 , 1 , } , { 0x19 , 1 , } , { 0x1b , 1 , } , { 0x1d , 1 , } ,
} , count_list [ ] = {
{ 0x00 , 0 , } , { 0x03 , 0 , } , { 0x04 , 0 , } , { 0x15 , 16 , } ,
{ 0x17 , 1 , } , { 0x18 , 0 , } , { 0x19 , 0 , } , { 0x1b , 0 , } ,
{ 0x1c , 1 , } , { 0x1d , 0 , } , { 0x1e , 1 , } ,
} ;
2006-01-04 20:31:31 +01:00
struct spu_priv2 __iomem * priv2 ;
2005-11-15 15:53:48 -05:00
int i ;
priv2 = spu - > priv2 ;
/* initialize all channel data to zero */
for ( i = 0 ; i < ARRAY_SIZE ( zero_list ) ; i + + ) {
int count ;
out_be64 ( & priv2 - > spu_chnlcntptr_RW , zero_list [ i ] . channel ) ;
for ( count = 0 ; count < zero_list [ i ] . count ; count + + )
out_be64 ( & priv2 - > spu_chnldata_RW , 0 ) ;
}
/* initialize channel counts to meaningful values */
for ( i = 0 ; i < ARRAY_SIZE ( count_list ) ; i + + ) {
out_be64 ( & priv2 - > spu_chnlcntptr_RW , count_list [ i ] . channel ) ;
out_be64 ( & priv2 - > spu_chnlcnt_RW , count_list [ i ] . count ) ;
}
}
2007-07-20 21:39:54 +02:00
EXPORT_SYMBOL_GPL ( spu_init_channels ) ;
2005-11-15 15:53:48 -05:00
2007-06-16 07:17:32 +10:00
static int spu_shutdown ( struct sys_device * sysdev )
{
struct spu * spu = container_of ( sysdev , struct spu , sysdev ) ;
spu_free_irqs ( spu ) ;
spu_destroy_spu ( spu ) ;
return 0 ;
}
2007-09-19 14:38:12 +10:00
static struct sysdev_class spu_sysdev_class = {
2007-06-16 07:17:32 +10:00
set_kset_name ( " spu " ) ,
. shutdown = spu_shutdown ,
2006-06-19 20:33:19 +02:00
} ;
2006-10-24 18:31:23 +02:00
int spu_add_sysdev_attr ( struct sysdev_attribute * attr )
{
struct spu * spu ;
2007-07-20 21:39:51 +02:00
mutex_lock ( & spu_full_list_mutex ) ;
2006-10-24 18:31:23 +02:00
list_for_each_entry ( spu , & spu_full_list , full_list )
sysdev_create_file ( & spu - > sysdev , attr ) ;
2007-07-20 21:39:51 +02:00
mutex_unlock ( & spu_full_list_mutex ) ;
2006-10-24 18:31:23 +02:00
return 0 ;
}
EXPORT_SYMBOL_GPL ( spu_add_sysdev_attr ) ;
int spu_add_sysdev_attr_group ( struct attribute_group * attrs )
{
struct spu * spu ;
2007-12-05 13:49:31 +11:00
int rc = 0 ;
2006-10-24 18:31:23 +02:00
2007-07-20 21:39:51 +02:00
mutex_lock ( & spu_full_list_mutex ) ;
2007-12-05 13:49:31 +11:00
list_for_each_entry ( spu , & spu_full_list , full_list ) {
rc = sysfs_create_group ( & spu - > sysdev . kobj , attrs ) ;
/* we're in trouble here, but try unwinding anyway */
if ( rc ) {
printk ( KERN_ERR " %s: can't create sysfs group '%s' \n " ,
__func__ , attrs - > name ) ;
list_for_each_entry_continue_reverse ( spu ,
& spu_full_list , full_list )
sysfs_remove_group ( & spu - > sysdev . kobj , attrs ) ;
break ;
}
}
2007-07-20 21:39:51 +02:00
mutex_unlock ( & spu_full_list_mutex ) ;
2006-10-24 18:31:23 +02:00
2007-12-05 13:49:31 +11:00
return rc ;
2006-10-24 18:31:23 +02:00
}
EXPORT_SYMBOL_GPL ( spu_add_sysdev_attr_group ) ;
void spu_remove_sysdev_attr ( struct sysdev_attribute * attr )
{
struct spu * spu ;
2007-07-20 21:39:51 +02:00
mutex_lock ( & spu_full_list_mutex ) ;
2006-10-24 18:31:23 +02:00
list_for_each_entry ( spu , & spu_full_list , full_list )
sysdev_remove_file ( & spu - > sysdev , attr ) ;
2007-07-20 21:39:51 +02:00
mutex_unlock ( & spu_full_list_mutex ) ;
2006-10-24 18:31:23 +02:00
}
EXPORT_SYMBOL_GPL ( spu_remove_sysdev_attr ) ;
void spu_remove_sysdev_attr_group ( struct attribute_group * attrs )
{
struct spu * spu ;
2007-07-20 21:39:51 +02:00
mutex_lock ( & spu_full_list_mutex ) ;
2006-10-24 18:31:23 +02:00
list_for_each_entry ( spu , & spu_full_list , full_list )
sysfs_remove_group ( & spu - > sysdev . kobj , attrs ) ;
2007-07-20 21:39:51 +02:00
mutex_unlock ( & spu_full_list_mutex ) ;
2006-10-24 18:31:23 +02:00
}
EXPORT_SYMBOL_GPL ( spu_remove_sysdev_attr_group ) ;
2006-06-19 20:33:19 +02:00
static int spu_create_sysdev ( struct spu * spu )
{
int ret ;
spu - > sysdev . id = spu - > number ;
spu - > sysdev . cls = & spu_sysdev_class ;
ret = sysdev_register ( & spu - > sysdev ) ;
if ( ret ) {
printk ( KERN_ERR " Can't register SPU %d with sysfs \n " ,
spu - > number ) ;
return ret ;
}
2006-11-20 18:45:02 +01:00
sysfs_add_device_to_node ( & spu - > sysdev , spu - > node ) ;
2006-06-19 20:33:19 +02:00
return 0 ;
}
2006-11-23 00:46:49 +01:00
static int __init create_spu ( void * data )
2005-11-15 15:53:48 -05:00
{
struct spu * spu ;
int ret ;
static int number ;
2007-03-10 00:05:37 +01:00
unsigned long flags ;
2007-07-20 21:39:33 +02:00
struct timespec ts ;
2005-11-15 15:53:48 -05:00
ret = - ENOMEM ;
2006-06-19 20:33:26 +02:00
spu = kzalloc ( sizeof ( * spu ) , GFP_KERNEL ) ;
2005-11-15 15:53:48 -05:00
if ( ! spu )
goto out ;
2007-07-20 21:39:54 +02:00
spu - > alloc_state = SPU_FREE ;
2006-11-23 00:46:49 +01:00
spin_lock_init ( & spu - > register_lock ) ;
2007-07-20 21:39:51 +02:00
spin_lock ( & spu_lock ) ;
2006-11-23 00:46:49 +01:00
spu - > number = number + + ;
2007-07-20 21:39:51 +02:00
spin_unlock ( & spu_lock ) ;
2006-11-23 00:46:49 +01:00
ret = spu_create_spu ( spu , data ) ;
2006-10-10 15:14:12 +10:00
2005-11-15 15:53:48 -05:00
if ( ret )
goto out_free ;
2006-10-24 18:31:14 +02:00
spu_mfc_sdr_setup ( spu ) ;
2006-01-04 20:31:30 +01:00
spu_mfc_sr1_set ( spu , 0x33 ) ;
2005-11-15 15:53:48 -05:00
ret = spu_request_irqs ( spu ) ;
if ( ret )
2006-11-23 00:46:49 +01:00
goto out_destroy ;
2005-11-15 15:53:48 -05:00
2006-06-19 20:33:19 +02:00
ret = spu_create_sysdev ( spu ) ;
if ( ret )
goto out_free_irqs ;
2007-07-20 21:39:54 +02:00
mutex_lock ( & cbe_spu_info [ spu - > node ] . list_mutex ) ;
2007-07-20 21:39:44 +02:00
list_add ( & spu - > cbe_list , & cbe_spu_info [ spu - > node ] . spus ) ;
cbe_spu_info [ spu - > node ] . n_spus + + ;
2007-07-20 21:39:54 +02:00
mutex_unlock ( & cbe_spu_info [ spu - > node ] . list_mutex ) ;
2007-07-20 21:39:51 +02:00
mutex_lock ( & spu_full_list_mutex ) ;
spin_lock_irqsave ( & spu_full_list_lock , flags ) ;
2006-10-24 18:31:23 +02:00
list_add ( & spu - > full_list , & spu_full_list ) ;
2007-07-20 21:39:51 +02:00
spin_unlock_irqrestore ( & spu_full_list_lock , flags ) ;
mutex_unlock ( & spu_full_list_mutex ) ;
2005-11-15 15:53:48 -05:00
2007-07-20 21:39:33 +02:00
spu - > stats . util_state = SPU_UTIL_IDLE_LOADED ;
ktime_get_ts ( & ts ) ;
spu - > stats . tstamp = timespec_to_ns ( & ts ) ;
2007-06-29 10:58:07 +10:00
2007-07-20 21:39:45 +02:00
INIT_LIST_HEAD ( & spu - > aff_list ) ;
2005-11-15 15:53:48 -05:00
goto out ;
2006-06-19 20:33:19 +02:00
out_free_irqs :
spu_free_irqs ( spu ) ;
2006-11-23 00:46:49 +01:00
out_destroy :
spu_destroy_spu ( spu ) ;
2005-11-15 15:53:48 -05:00
out_free :
kfree ( spu ) ;
out :
return ret ;
}
2007-06-29 10:58:07 +10:00
static const char * spu_state_names [ ] = {
" user " , " system " , " iowait " , " idle "
} ;
static unsigned long long spu_acct_time ( struct spu * spu ,
enum spu_utilization_state state )
{
2007-07-20 21:39:33 +02:00
struct timespec ts ;
2007-06-29 10:58:07 +10:00
unsigned long long time = spu - > stats . times [ state ] ;
2007-07-20 21:39:33 +02:00
/*
* If the spu is idle or the context is stopped , utilization
* statistics are not updated . Apply the time delta from the
* last recorded state of the spu .
*/
if ( spu - > stats . util_state = = state ) {
ktime_get_ts ( & ts ) ;
time + = timespec_to_ns ( & ts ) - spu - > stats . tstamp ;
}
2007-06-29 10:58:07 +10:00
2007-07-20 21:39:33 +02:00
return time / NSEC_PER_MSEC ;
2007-06-29 10:58:07 +10:00
}
static ssize_t spu_stat_show ( struct sys_device * sysdev , char * buf )
{
struct spu * spu = container_of ( sysdev , struct spu , sysdev ) ;
return sprintf ( buf , " %s %llu %llu %llu %llu "
" %llu %llu %llu %llu %llu %llu %llu %llu \n " ,
2007-07-20 21:39:33 +02:00
spu_state_names [ spu - > stats . util_state ] ,
2007-06-29 10:58:07 +10:00
spu_acct_time ( spu , SPU_UTIL_USER ) ,
spu_acct_time ( spu , SPU_UTIL_SYSTEM ) ,
spu_acct_time ( spu , SPU_UTIL_IOWAIT ) ,
2007-07-20 21:39:33 +02:00
spu_acct_time ( spu , SPU_UTIL_IDLE_LOADED ) ,
2007-06-29 10:58:07 +10:00
spu - > stats . vol_ctx_switch ,
spu - > stats . invol_ctx_switch ,
spu - > stats . slb_flt ,
spu - > stats . hash_flt ,
spu - > stats . min_flt ,
spu - > stats . maj_flt ,
spu - > stats . class2_intr ,
spu - > stats . libassist ) ;
}
static SYSDEV_ATTR ( stat , 0644 , spu_stat_show , NULL ) ;
2005-11-15 15:53:48 -05:00
static int __init init_spu_base ( void )
{
2007-04-23 21:08:28 +02:00
int i , ret = 0 ;
2005-11-15 15:53:48 -05:00
2007-07-20 21:39:44 +02:00
for ( i = 0 ; i < MAX_NUMNODES ; i + + ) {
2007-07-20 21:39:54 +02:00
mutex_init ( & cbe_spu_info [ i ] . list_mutex ) ;
2007-07-20 21:39:44 +02:00
INIT_LIST_HEAD ( & cbe_spu_info [ i ] . spus ) ;
}
2007-04-23 21:08:29 +02:00
2006-11-27 19:18:54 +01:00
if ( ! spu_management_ops )
2007-04-23 21:08:28 +02:00
goto out ;
2006-11-27 19:18:54 +01:00
2006-06-19 20:33:19 +02:00
/* create sysdev class for spus */
ret = sysdev_class_register ( & spu_sysdev_class ) ;
if ( ret )
2007-04-23 21:08:28 +02:00
goto out ;
2006-06-19 20:33:19 +02:00
2006-11-23 00:46:49 +01:00
ret = spu_enumerate_spus ( create_spu ) ;
2007-07-17 04:05:52 -07:00
if ( ret < 0 ) {
2006-11-23 00:46:49 +01:00
printk ( KERN_WARNING " %s: Error initializing spus \n " ,
__FUNCTION__ ) ;
2007-04-23 21:08:28 +02:00
goto out_unregister_sysdev_class ;
2005-11-15 15:53:48 -05:00
}
2006-10-24 18:31:27 +02:00
2007-07-17 04:05:52 -07:00
if ( ret > 0 ) {
/*
* We cannot put the forward declaration in
* < linux / linux_logo . h > because of conflicting session type
* conflicts for const and __initdata with different compiler
* versions
*/
extern const struct linux_logo logo_spe_clut224 ;
fb_append_extra_logo ( & logo_spe_clut224 , ret ) ;
}
2007-07-20 21:39:51 +02:00
mutex_lock ( & spu_full_list_mutex ) ;
2006-10-24 18:31:27 +02:00
xmon_register_spus ( & spu_full_list ) ;
2007-07-20 21:39:27 +02:00
crash_register_spus ( & spu_full_list ) ;
2007-07-20 21:39:51 +02:00
mutex_unlock ( & spu_full_list_mutex ) ;
2007-06-29 10:58:07 +10:00
spu_add_sysdev_attr ( & attr_stat ) ;
2007-08-03 18:53:46 -07:00
spu_init_affinity ( ) ;
2007-07-20 21:39:46 +02:00
2007-04-23 21:08:28 +02:00
return 0 ;
out_unregister_sysdev_class :
sysdev_class_unregister ( & spu_sysdev_class ) ;
out :
2005-11-15 15:53:48 -05:00
return ret ;
}
module_init ( init_spu_base ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( " Arnd Bergmann <arndb@de.ibm.com> " ) ;