2006-01-11 22:46:18 +01: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 17:16:04 -02:00
* Paravirt ops integration : Glauber de Oliveira Costa < gcosta @ redhat . com > ,
* Ravikiran Thirumalai < kiran @ scalemp . com >
2006-01-11 22:46:18 +01:00
*/
# include <linux/init.h>
# include <linux/pci_ids.h>
# include <linux/pci_regs.h>
2008-05-12 15:43:34 +02:00
# include <asm/apic.h>
2006-01-11 22:46:18 +01:00
# include <asm/pci-direct.h>
2006-10-12 12:17:52 -07:00
# include <asm/io.h>
2008-02-11 17:16:04 -02:00
# include <asm/paravirt.h>
2008-05-12 15:43:34 +02:00
# include <asm/setup.h>
2008-02-11 17:16:04 -02:00
2008-03-20 00:41:16 -07:00
# if defined CONFIG_PCI && defined CONFIG_PARAVIRT
2008-02-11 17:16:04 -02:00
/*
* 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 ) ) ;
}
2008-09-22 22:58:47 -07:00
static unsigned __init_or_module vsmp_patch ( u8 type , u16 clobbers , void * ibuf ,
2008-02-11 17:16:04 -02:00
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-11 22:46:18 +01:00
2008-03-20 00:41:16 -07:00
static void __init set_vsmp_pv_ops ( void )
2006-01-11 22:46:18 +01:00
{
void * address ;
2008-02-11 17:16:03 -02:00
unsigned int cap , ctl , cfg ;
2006-01-11 22:46:18 +01:00
/* set vSMP magic bits to indicate vSMP capable kernel */
2008-02-11 17:16:03 -02:00
cfg = read_pci_config ( 0 , 0x1f , 0 , PCI_BASE_ADDRESS_0 ) ;
address = early_ioremap ( cfg , 8 ) ;
2006-01-11 22:46:18 +01:00
cap = readl ( address ) ;
ctl = readl ( address + 4 ) ;
2008-01-30 13:30:24 +01:00
printk ( KERN_INFO " vSMP CTL: capabilities:0x%08x control:0x%08x \n " ,
cap , ctl ) ;
2006-01-11 22:46:18 +01:00
if ( cap & ctl & ( 1 < < 4 ) ) {
2008-03-20 00:43:16 -07:00
/* Setup irq ops and turn on vSMP IRQ fastpath handling */
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-11 22:46:18 +01:00
ctl & = ~ ( 1 < < 4 ) ;
writel ( ctl , address + 4 ) ;
ctl = readl ( address + 4 ) ;
2008-01-30 13:30:24 +01:00
printk ( KERN_INFO " vSMP CTL: control set to:0x%08x \n " , ctl ) ;
2006-01-11 22:46:18 +01:00
}
2008-02-11 17:16:03 -02:00
early_iounmap ( address , 8 ) ;
2008-03-20 00:41:16 -07:00
}
# else
static void __init set_vsmp_pv_ops ( void )
{
}
# endif
# ifdef CONFIG_PCI
2008-03-24 14:48:36 -07:00
static int is_vsmp = - 1 ;
2008-03-20 00:41:16 -07:00
2008-03-24 14:48:36 -07:00
static void __init detect_vsmp_box ( void )
2008-03-20 00:41:16 -07:00
{
2008-03-24 14:48:36 -07:00
is_vsmp = 0 ;
2008-03-20 00:41:16 -07:00
if ( ! early_pci_allowed ( ) )
2008-03-24 14:48:36 -07:00
return ;
2008-03-20 00:41:16 -07:00
2008-03-24 14:48:36 -07:00
/* Check if we are running on a ScaleMP vSMPowered box */
2008-03-21 09:55:06 +01:00
if ( read_pci_config ( 0 , 0x1f , 0 , PCI_VENDOR_ID ) = =
( PCI_VENDOR_ID_SCALEMP | ( PCI_DEVICE_ID_SCALEMP_VSMP_CTL < < 16 ) ) )
2008-03-24 14:48:36 -07:00
is_vsmp = 1 ;
}
2008-03-20 00:41:16 -07:00
2008-03-24 14:48:36 -07:00
int is_vsmp_box ( void )
{
if ( is_vsmp ! = - 1 )
return is_vsmp ;
else {
WARN_ON_ONCE ( 1 ) ;
return 0 ;
}
2008-03-20 00:41:16 -07:00
}
# else
2008-04-16 18:45:35 +02:00
static void __init detect_vsmp_box ( void )
2008-03-24 14:48:36 -07:00
{
}
2008-03-20 00:41:16 -07:00
int is_vsmp_box ( void )
{
return 0 ;
}
# endif
void __init vsmp_init ( void )
{
2008-03-24 14:48:36 -07:00
detect_vsmp_box ( ) ;
2008-03-20 00:41:16 -07:00
if ( ! is_vsmp_box ( ) )
return ;
set_vsmp_pv_ops ( ) ;
2008-02-11 17:16:02 -02:00
return ;
2006-01-11 22:46:18 +01:00
}