2006-01-12 00:46:18 +03:00
/*
* vSMPowered ( tm ) systems specific initialization
* Copyright ( C ) 2005 ScaleMP Inc .
*
* Use of this code is subject to the terms and conditions of the
* GNU general public license version 2. See " COPYING " or
* http : //www.gnu.org/licenses/gpl.html
*
* Ravikiran Thirumalai < kiran @ scalemp . com > ,
* Shai Fultheim < shai @ scalemp . com >
2008-02-11 22:16:04 +03:00
* Paravirt ops integration : Glauber de Oliveira Costa < gcosta @ redhat . com > ,
* Ravikiran Thirumalai < kiran @ scalemp . com >
2006-01-12 00:46:18 +03:00
*/
# include <linux/init.h>
# include <linux/pci_ids.h>
# include <linux/pci_regs.h>
# include <asm/pci-direct.h>
2006-10-12 23:17:52 +04:00
# include <asm/io.h>
2008-02-11 22:16:04 +03:00
# include <asm/paravirt.h>
/*
* Interrupt control on vSMPowered systems :
* ~ AC is a shadow of IF . If IF is ' on ' AC should be ' off '
* and vice versa .
*/
static unsigned long vsmp_save_fl ( void )
{
unsigned long flags = native_save_fl ( ) ;
if ( ! ( flags & X86_EFLAGS_IF ) | | ( flags & X86_EFLAGS_AC ) )
flags & = ~ X86_EFLAGS_IF ;
return flags ;
}
static void vsmp_restore_fl ( unsigned long flags )
{
if ( flags & X86_EFLAGS_IF )
flags & = ~ X86_EFLAGS_AC ;
else
flags | = X86_EFLAGS_AC ;
native_restore_fl ( flags ) ;
}
static void vsmp_irq_disable ( void )
{
unsigned long flags = native_save_fl ( ) ;
native_restore_fl ( ( flags & ~ X86_EFLAGS_IF ) | X86_EFLAGS_AC ) ;
}
static void vsmp_irq_enable ( void )
{
unsigned long flags = native_save_fl ( ) ;
native_restore_fl ( ( flags | X86_EFLAGS_IF ) & ( ~ X86_EFLAGS_AC ) ) ;
}
static unsigned __init vsmp_patch ( u8 type , u16 clobbers , void * ibuf ,
unsigned long addr , unsigned len )
{
switch ( type ) {
case PARAVIRT_PATCH ( pv_irq_ops . irq_enable ) :
case PARAVIRT_PATCH ( pv_irq_ops . irq_disable ) :
case PARAVIRT_PATCH ( pv_irq_ops . save_fl ) :
case PARAVIRT_PATCH ( pv_irq_ops . restore_fl ) :
return paravirt_patch_default ( type , clobbers , ibuf , addr , len ) ;
default :
return native_patch ( type , clobbers , ibuf , addr , len ) ;
}
}
2006-01-12 00:46:18 +03:00
2008-02-25 08:36:28 +03:00
static int vsmp = - 1 ;
int is_vsmp_box ( void )
{
if ( vsmp ! = - 1 )
return vsmp ;
vsmp = 0 ;
if ( ! early_pci_allowed ( ) )
return vsmp ;
/* Check if we are running on a ScaleMP vSMP box */
if ( read_pci_config ( 0 , 0x1f , 0 , PCI_VENDOR_ID ) = =
( PCI_VENDOR_ID_SCALEMP | | ( PCI_DEVICE_ID_SCALEMP_VSMP_CTL < < 16 ) ) )
vsmp = 1 ;
return vsmp ;
}
2008-02-11 22:16:03 +03:00
void __init vsmp_init ( void )
2006-01-12 00:46:18 +03:00
{
void * address ;
2008-02-11 22:16:03 +03:00
unsigned int cap , ctl , cfg ;
2006-01-12 00:46:18 +03:00
2008-02-25 08:36:28 +03:00
if ( ! is_vsmp_box ( ) )
2008-02-11 22:16:02 +03:00
return ;
2006-09-26 12:52:41 +04:00
2008-02-25 08:36:28 +03:00
if ( ! early_pci_allowed ( ) )
2008-02-11 22:16:02 +03:00
return ;
2006-01-12 00:46:18 +03:00
2008-02-11 22:16:05 +03:00
/* If we are, use the distinguished irq functions */
pv_irq_ops . irq_disable = vsmp_irq_disable ;
pv_irq_ops . irq_enable = vsmp_irq_enable ;
pv_irq_ops . save_fl = vsmp_save_fl ;
pv_irq_ops . restore_fl = vsmp_restore_fl ;
pv_init_ops . patch = vsmp_patch ;
2006-01-12 00:46:18 +03:00
/* set vSMP magic bits to indicate vSMP capable kernel */
2008-02-11 22:16:03 +03:00
cfg = read_pci_config ( 0 , 0x1f , 0 , PCI_BASE_ADDRESS_0 ) ;
address = early_ioremap ( cfg , 8 ) ;
2006-01-12 00:46:18 +03:00
cap = readl ( address ) ;
ctl = readl ( address + 4 ) ;
2008-01-30 15:30:24 +03:00
printk ( KERN_INFO " vSMP CTL: capabilities:0x%08x control:0x%08x \n " ,
cap , ctl ) ;
2006-01-12 00:46:18 +03:00
if ( cap & ctl & ( 1 < < 4 ) ) {
/* Turn on vSMP IRQ fastpath handling (see system.h) */
ctl & = ~ ( 1 < < 4 ) ;
writel ( ctl , address + 4 ) ;
ctl = readl ( address + 4 ) ;
2008-01-30 15:30:24 +03:00
printk ( KERN_INFO " vSMP CTL: control set to:0x%08x \n " , ctl ) ;
2006-01-12 00:46:18 +03:00
}
2008-02-11 22:16:03 +03:00
early_iounmap ( address , 8 ) ;
2008-02-11 22:16:02 +03:00
return ;
2006-01-12 00:46:18 +03:00
}