2005-04-17 02:20:36 +04:00
/*
* PCI Express Hot Plug Controller Driver
*
* Copyright ( C ) 1995 , 2001 Compaq Computer Corporation
* Copyright ( C ) 2001 Greg Kroah - Hartman ( greg @ kroah . com )
* Copyright ( C ) 2001 IBM Corp .
* Copyright ( C ) 2003 - 2004 Intel Corporation
*
* All rights reserved .
*
* 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 of the License , 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 , GOOD TITLE or
* NON INFRINGEMENT . 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-08-17 02:16:10 +04:00
* Send feedback to < greg @ kroah . com > , < kristen . c . accardi @ intel . com >
2005-04-17 02:20:36 +04:00
*
*/
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/types.h>
# include <linux/pci.h>
2007-03-07 02:02:26 +03:00
# include <linux/workqueue.h>
2005-04-17 02:20:36 +04:00
# include "../pci.h"
# include "pciehp.h"
2007-03-07 02:02:26 +03:00
static void interrupt_event_handler ( struct work_struct * work ) ;
2005-04-17 02:20:36 +04:00
2007-03-07 02:02:26 +03:00
static int queue_interrupt_event ( struct slot * p_slot , u32 event_type )
2006-09-22 21:17:10 +04:00
{
2007-03-07 02:02:26 +03:00
struct event_info * info ;
info = kmalloc ( sizeof ( * info ) , GFP_ATOMIC ) ;
if ( ! info )
return - ENOMEM ;
info - > event_type = event_type ;
info - > p_slot = p_slot ;
INIT_WORK ( & info - > work , interrupt_event_handler ) ;
schedule_work ( & info - > work ) ;
return 0 ;
2006-09-22 21:17:10 +04:00
}
2008-05-27 14:03:16 +04:00
u8 pciehp_handle_attention_button ( struct slot * p_slot )
2005-04-17 02:20:36 +04:00
{
2007-03-07 02:02:26 +03:00
u32 event_type ;
2008-09-05 07:11:26 +04:00
struct controller * ctrl = p_slot - > ctrl ;
2005-04-17 02:20:36 +04:00
/* Attention Button Change */
2008-10-23 06:47:32 +04:00
ctrl_dbg ( ctrl , " Attention button interrupt received \n " ) ;
2005-04-17 02:20:36 +04:00
/*
* Button pressed - See if need to TAKE ACTION ! ! !
*/
2008-10-21 03:41:38 +04:00
ctrl_info ( ctrl , " Button pressed on Slot(%s) \n " , slot_name ( p_slot ) ) ;
2007-03-07 02:02:26 +03:00
event_type = INT_BUTTON_PRESS ;
2005-04-17 02:20:36 +04:00
2007-03-07 02:02:26 +03:00
queue_interrupt_event ( p_slot , event_type ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2008-05-27 14:03:16 +04:00
u8 pciehp_handle_switch_change ( struct slot * p_slot )
2005-04-17 02:20:36 +04:00
{
u8 getstatus ;
2007-03-07 02:02:26 +03:00
u32 event_type ;
2008-09-05 07:11:26 +04:00
struct controller * ctrl = p_slot - > ctrl ;
2005-04-17 02:20:36 +04:00
/* Switch Change */
2008-10-23 06:47:32 +04:00
ctrl_dbg ( ctrl , " Switch interrupt received \n " ) ;
2005-04-17 02:20:36 +04:00
2009-09-15 12:30:48 +04:00
pciehp_get_latch_status ( p_slot , & getstatus ) ;
2005-04-17 02:20:36 +04:00
if ( getstatus ) {
/*
* Switch opened
*/
2008-10-21 03:41:38 +04:00
ctrl_info ( ctrl , " Latch open on Slot(%s) \n " , slot_name ( p_slot ) ) ;
2007-03-07 02:02:26 +03:00
event_type = INT_SWITCH_OPEN ;
2005-04-17 02:20:36 +04:00
} else {
/*
* Switch closed
*/
2008-10-21 03:41:38 +04:00
ctrl_info ( ctrl , " Latch close on Slot(%s) \n " , slot_name ( p_slot ) ) ;
2007-03-07 02:02:26 +03:00
event_type = INT_SWITCH_CLOSE ;
2005-04-17 02:20:36 +04:00
}
2007-03-07 02:02:26 +03:00
queue_interrupt_event ( p_slot , event_type ) ;
2005-04-17 02:20:36 +04:00
2007-03-07 02:02:26 +03:00
return 1 ;
2005-04-17 02:20:36 +04:00
}
2008-05-27 14:03:16 +04:00
u8 pciehp_handle_presence_change ( struct slot * p_slot )
2005-04-17 02:20:36 +04:00
{
2007-03-07 02:02:26 +03:00
u32 event_type ;
u8 presence_save ;
2008-09-05 07:11:26 +04:00
struct controller * ctrl = p_slot - > ctrl ;
2005-04-17 02:20:36 +04:00
/* Presence Change */
2008-10-23 06:47:32 +04:00
ctrl_dbg ( ctrl , " Presence/Notify input change \n " ) ;
2005-04-17 02:20:36 +04:00
/* Switch is open, assume a presence change
* Save the presence state
*/
2009-09-15 12:30:48 +04:00
pciehp_get_adapter_status ( p_slot , & presence_save ) ;
2005-11-01 03:20:09 +03:00
if ( presence_save ) {
2005-04-17 02:20:36 +04:00
/*
* Card Present
*/
2008-10-21 03:41:38 +04:00
ctrl_info ( ctrl , " Card present on Slot(%s) \n " , slot_name ( p_slot ) ) ;
2007-03-07 02:02:26 +03:00
event_type = INT_PRESENCE_ON ;
2005-04-17 02:20:36 +04:00
} else {
/*
* Not Present
*/
2008-10-21 03:41:38 +04:00
ctrl_info ( ctrl , " Card not present on Slot(%s) \n " ,
slot_name ( p_slot ) ) ;
2007-03-07 02:02:26 +03:00
event_type = INT_PRESENCE_OFF ;
2005-04-17 02:20:36 +04:00
}
2007-03-07 02:02:26 +03:00
queue_interrupt_event ( p_slot , event_type ) ;
2005-04-17 02:20:36 +04:00
2007-03-07 02:02:26 +03:00
return 1 ;
2005-04-17 02:20:36 +04:00
}
2008-05-27 14:03:16 +04:00
u8 pciehp_handle_power_fault ( struct slot * p_slot )
2005-04-17 02:20:36 +04:00
{
2007-03-07 02:02:26 +03:00
u32 event_type ;
2008-09-05 07:11:26 +04:00
struct controller * ctrl = p_slot - > ctrl ;
2005-04-17 02:20:36 +04:00
/* power fault */
2008-10-23 06:47:32 +04:00
ctrl_dbg ( ctrl , " Power fault interrupt received \n " ) ;
2009-11-13 09:14:10 +03:00
ctrl_err ( ctrl , " Power fault on slot %s \n " , slot_name ( p_slot ) ) ;
event_type = INT_POWER_FAULT ;
ctrl_info ( ctrl , " Power fault bit %x set \n " , 0 ) ;
2007-03-07 02:02:26 +03:00
queue_interrupt_event ( p_slot , event_type ) ;
return 1 ;
2005-04-17 02:20:36 +04:00
}
2007-08-10 03:09:36 +04:00
/* The following routines constitute the bulk of the
2005-04-17 02:20:36 +04:00
hotplug controller logic
*/
static void set_slot_off ( struct controller * ctrl , struct slot * pslot )
{
/* turn off slot, turn on Amber LED, turn off Green LED if supported*/
2008-04-26 01:39:06 +04:00
if ( POWER_CTRL ( ctrl ) ) {
2009-09-15 12:30:48 +04:00
if ( pciehp_power_off_slot ( pslot ) ) {
2008-09-05 07:11:26 +04:00
ctrl_err ( ctrl ,
2008-10-23 06:47:32 +04:00
" Issue of Slot Power Off command failed \n " ) ;
2005-04-17 02:20:36 +04:00
return ;
}
2008-10-24 09:26:35 +04:00
/*
* After turning power off , we must wait for at least 1 second
* before taking any action that relies on power having been
* removed from the slot / adapter .
*/
msleep ( 1000 ) ;
2005-04-17 02:20:36 +04:00
}
2008-04-26 01:39:06 +04:00
if ( PWR_LED ( ctrl ) )
2009-09-15 12:30:48 +04:00
pciehp_green_led_off ( pslot ) ;
2005-04-17 02:20:36 +04:00
2008-04-26 01:39:06 +04:00
if ( ATTN_LED ( ctrl ) ) {
2009-09-15 12:30:48 +04:00
if ( pciehp_set_attention_status ( pslot , 1 ) ) {
2008-10-23 06:47:32 +04:00
ctrl_err ( ctrl ,
" Issue of Set Attention Led command failed \n " ) ;
2005-04-17 02:20:36 +04:00
return ;
}
}
}
/**
* board_added - Called after a board has been added to the system .
2007-11-28 20:04:30 +03:00
* @ p_slot : & slot where board is added
2005-04-17 02:20:36 +04:00
*
2007-11-28 20:04:30 +03:00
* Turns power on for the board .
* Configures board .
2005-04-17 02:20:36 +04:00
*/
2005-11-01 03:20:09 +03:00
static int board_added ( struct slot * p_slot )
2005-04-17 02:20:36 +04:00
{
2006-12-22 04:01:09 +03:00
int retval = 0 ;
2005-11-01 03:20:08 +03:00
struct controller * ctrl = p_slot - > ctrl ;
2009-09-15 12:30:14 +04:00
struct pci_bus * parent = ctrl - > pcie - > port - > subordinate ;
2005-04-17 02:20:36 +04:00
2008-04-26 01:39:06 +04:00
if ( POWER_CTRL ( ctrl ) ) {
2005-04-17 02:20:36 +04:00
/* Power on slot */
2009-09-15 12:30:48 +04:00
retval = pciehp_power_on_slot ( p_slot ) ;
2006-12-22 04:01:09 +03:00
if ( retval )
return retval ;
2005-04-17 02:20:36 +04:00
}
2007-08-10 03:09:36 +04:00
2008-04-26 01:39:06 +04:00
if ( PWR_LED ( ctrl ) )
2009-09-15 12:30:48 +04:00
pciehp_green_led_blink ( p_slot ) ;
2005-04-17 02:20:36 +04:00
2006-12-22 04:01:09 +03:00
/* Check link training status */
2009-09-15 12:30:48 +04:00
retval = pciehp_check_link_status ( ctrl ) ;
2006-12-22 04:01:09 +03:00
if ( retval ) {
2008-10-23 06:47:32 +04:00
ctrl_err ( ctrl , " Failed to check link status \n " ) ;
2009-11-13 09:14:10 +03:00
goto err_exit ;
2005-04-17 02:20:36 +04:00
}
/* Check for a power fault */
2009-11-13 09:14:10 +03:00
if ( ctrl - > power_fault_detected | | pciehp_query_power_fault ( p_slot ) ) {
ctrl_err ( ctrl , " Power fault on slot %s \n " , slot_name ( p_slot ) ) ;
2009-09-15 12:33:30 +04:00
retval = - EIO ;
2005-11-01 03:20:06 +03:00
goto err_exit ;
2005-04-17 02:20:36 +04:00
}
2006-12-22 04:01:09 +03:00
retval = pciehp_configure_device ( p_slot ) ;
if ( retval ) {
2009-09-15 12:26:56 +04:00
ctrl_err ( ctrl , " Cannot add device at %04x:%02x:00 \n " ,
pci_domain_nr ( parent ) , parent - > number ) ;
2005-11-01 03:20:06 +03:00
goto err_exit ;
}
2005-04-17 02:20:36 +04:00
2008-04-26 01:39:06 +04:00
if ( PWR_LED ( ctrl ) )
2009-09-15 12:30:48 +04:00
pciehp_green_led_on ( p_slot ) ;
2006-12-22 04:01:09 +03:00
2005-04-17 02:20:36 +04:00
return 0 ;
2005-11-01 03:20:06 +03:00
err_exit :
set_slot_off ( ctrl , p_slot ) ;
2006-12-22 04:01:09 +03:00
return retval ;
2005-04-17 02:20:36 +04:00
}
/**
2007-11-28 20:04:30 +03:00
* remove_board - Turns off slot and LEDs
* @ p_slot : slot where board is being removed
2005-04-17 02:20:36 +04:00
*/
2005-11-01 03:20:09 +03:00
static int remove_board ( struct slot * p_slot )
2005-04-17 02:20:36 +04:00
{
2006-12-22 04:01:09 +03:00
int retval = 0 ;
2005-11-01 03:20:08 +03:00
struct controller * ctrl = p_slot - > ctrl ;
2005-04-17 02:20:36 +04:00
2006-12-22 04:01:09 +03:00
retval = pciehp_unconfigure_device ( p_slot ) ;
if ( retval )
return retval ;
2005-04-17 02:20:36 +04:00
2008-04-26 01:39:06 +04:00
if ( POWER_CTRL ( ctrl ) ) {
2005-04-17 02:20:36 +04:00
/* power off slot */
2009-09-15 12:30:48 +04:00
retval = pciehp_power_off_slot ( p_slot ) ;
2006-12-22 04:01:09 +03:00
if ( retval ) {
2008-10-23 06:47:32 +04:00
ctrl_err ( ctrl ,
" Issue of Slot Disable command failed \n " ) ;
2006-12-22 04:01:09 +03:00
return retval ;
2005-04-17 02:20:36 +04:00
}
2008-10-24 09:26:35 +04:00
/*
* After turning power off , we must wait for at least 1 second
* before taking any action that relies on power having been
* removed from the slot / adapter .
*/
msleep ( 1000 ) ;
2005-04-17 02:20:36 +04:00
}
2009-09-15 12:30:48 +04:00
/* turn off Green LED */
2008-04-26 01:39:06 +04:00
if ( PWR_LED ( ctrl ) )
2009-09-15 12:30:48 +04:00
pciehp_green_led_off ( p_slot ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2007-03-07 02:02:26 +03:00
struct power_work_info {
struct slot * p_slot ;
struct work_struct work ;
} ;
2005-04-17 02:20:36 +04:00
/**
2007-11-28 20:04:30 +03:00
* pciehp_power_thread - handle pushbutton events
* @ work : & struct work_struct describing work to be done
2005-04-17 02:20:36 +04:00
*
2007-11-28 20:04:30 +03:00
* Scheduled procedure to handle blocking stuff for the pushbuttons .
2005-04-17 02:20:36 +04:00
* Handles all pending events and exits .
*/
2007-03-07 02:02:26 +03:00
static void pciehp_power_thread ( struct work_struct * work )
2005-04-17 02:20:36 +04:00
{
2007-03-07 02:02:26 +03:00
struct power_work_info * info =
container_of ( work , struct power_work_info , work ) ;
struct slot * p_slot = info - > p_slot ;
mutex_lock ( & p_slot - > lock ) ;
switch ( p_slot - > state ) {
case POWEROFF_STATE :
mutex_unlock ( & p_slot - > lock ) ;
2008-10-23 06:47:32 +04:00
ctrl_dbg ( p_slot - > ctrl ,
2009-09-15 12:26:56 +04:00
" Disabling domain:bus:device=%04x:%02x:00 \n " ,
2009-09-15 12:30:14 +04:00
pci_domain_nr ( p_slot - > ctrl - > pcie - > port - > subordinate ) ,
p_slot - > ctrl - > pcie - > port - > subordinate - > number ) ;
2005-04-17 02:20:36 +04:00
pciehp_disable_slot ( p_slot ) ;
2007-03-07 02:02:26 +03:00
mutex_lock ( & p_slot - > lock ) ;
2005-04-17 02:20:36 +04:00
p_slot - > state = STATIC_STATE ;
2007-03-07 02:02:26 +03:00
break ;
case POWERON_STATE :
mutex_unlock ( & p_slot - > lock ) ;
2009-09-15 12:30:48 +04:00
if ( pciehp_enable_slot ( p_slot ) & & PWR_LED ( p_slot - > ctrl ) )
pciehp_green_led_off ( p_slot ) ;
2007-03-07 02:02:26 +03:00
mutex_lock ( & p_slot - > lock ) ;
2005-04-17 02:20:36 +04:00
p_slot - > state = STATIC_STATE ;
2007-03-07 02:02:26 +03:00
break ;
default :
break ;
2005-04-17 02:20:36 +04:00
}
2007-03-07 02:02:26 +03:00
mutex_unlock ( & p_slot - > lock ) ;
2005-04-17 02:20:36 +04:00
2007-03-07 02:02:26 +03:00
kfree ( info ) ;
2005-04-17 02:20:36 +04:00
}
2007-03-21 21:45:31 +03:00
void pciehp_queue_pushbutton_work ( struct work_struct * work )
2005-04-17 02:20:36 +04:00
{
2007-03-07 02:02:26 +03:00
struct slot * p_slot = container_of ( work , struct slot , work . work ) ;
struct power_work_info * info ;
2005-04-17 02:20:36 +04:00
2007-03-07 02:02:26 +03:00
info = kmalloc ( sizeof ( * info ) , GFP_KERNEL ) ;
if ( ! info ) {
2008-09-05 07:11:26 +04:00
ctrl_err ( p_slot - > ctrl , " %s: Cannot allocate memory \n " ,
__func__ ) ;
2005-04-17 02:20:36 +04:00
return ;
}
2007-03-07 02:02:26 +03:00
info - > p_slot = p_slot ;
INIT_WORK ( & info - > work , pciehp_power_thread ) ;
2005-04-17 02:20:36 +04:00
2007-03-07 02:02:26 +03:00
mutex_lock ( & p_slot - > lock ) ;
switch ( p_slot - > state ) {
case BLINKINGOFF_STATE :
2005-04-17 02:20:36 +04:00
p_slot - > state = POWEROFF_STATE ;
2007-03-07 02:02:26 +03:00
break ;
case BLINKINGON_STATE :
2005-04-17 02:20:36 +04:00
p_slot - > state = POWERON_STATE ;
2007-03-07 02:02:26 +03:00
break ;
default :
goto out ;
2005-04-17 02:20:36 +04:00
}
2007-03-07 02:02:26 +03:00
queue_work ( pciehp_wq , & info - > work ) ;
out :
mutex_unlock ( & p_slot - > lock ) ;
2005-04-17 02:20:36 +04:00
}
2007-03-07 02:02:26 +03:00
/*
* Note : This function must be called with slot - > lock held
*/
static void handle_button_press_event ( struct slot * p_slot )
2005-04-17 02:20:36 +04:00
{
2007-03-07 02:02:26 +03:00
struct controller * ctrl = p_slot - > ctrl ;
2005-04-17 02:20:36 +04:00
u8 getstatus ;
2007-03-07 02:02:26 +03:00
switch ( p_slot - > state ) {
case STATIC_STATE :
2009-09-15 12:30:48 +04:00
pciehp_get_power_status ( p_slot , & getstatus ) ;
2007-03-07 02:02:26 +03:00
if ( getstatus ) {
p_slot - > state = BLINKINGOFF_STATE ;
2008-09-05 07:11:26 +04:00
ctrl_info ( ctrl ,
" PCI slot #%s - powering off due to button "
2008-10-21 03:41:38 +04:00
" press. \n " , slot_name ( p_slot ) ) ;
2007-03-07 02:02:26 +03:00
} else {
p_slot - > state = BLINKINGON_STATE ;
2008-09-05 07:11:26 +04:00
ctrl_info ( ctrl ,
" PCI slot #%s - powering on due to button "
2008-10-21 03:41:38 +04:00
" press. \n " , slot_name ( p_slot ) ) ;
2007-03-07 02:02:26 +03:00
}
/* blink green LED and turn off amber */
2008-04-26 01:39:06 +04:00
if ( PWR_LED ( ctrl ) )
2009-09-15 12:30:48 +04:00
pciehp_green_led_blink ( p_slot ) ;
2008-04-26 01:39:06 +04:00
if ( ATTN_LED ( ctrl ) )
2009-09-15 12:30:48 +04:00
pciehp_set_attention_status ( p_slot , 0 ) ;
2007-03-07 02:02:26 +03:00
schedule_delayed_work ( & p_slot - > work , 5 * HZ ) ;
break ;
case BLINKINGOFF_STATE :
case BLINKINGON_STATE :
/*
* Cancel if we are still blinking ; this means that we
* press the attention again before the 5 sec . limit
* expires to cancel hot - add or hot - remove
*/
2008-10-21 03:41:38 +04:00
ctrl_info ( ctrl , " Button cancel on Slot(%s) \n " , slot_name ( p_slot ) ) ;
2007-03-07 02:02:26 +03:00
cancel_delayed_work ( & p_slot - > work ) ;
if ( p_slot - > state = = BLINKINGOFF_STATE ) {
2008-04-26 01:39:06 +04:00
if ( PWR_LED ( ctrl ) )
2009-09-15 12:30:48 +04:00
pciehp_green_led_on ( p_slot ) ;
2007-03-07 02:02:26 +03:00
} else {
2008-04-26 01:39:06 +04:00
if ( PWR_LED ( ctrl ) )
2009-09-15 12:30:48 +04:00
pciehp_green_led_off ( p_slot ) ;
2007-03-07 02:02:26 +03:00
}
2008-04-26 01:39:06 +04:00
if ( ATTN_LED ( ctrl ) )
2009-09-15 12:30:48 +04:00
pciehp_set_attention_status ( p_slot , 0 ) ;
2008-09-05 07:11:26 +04:00
ctrl_info ( ctrl , " PCI slot #%s - action canceled "
2008-10-21 03:41:38 +04:00
" due to button press \n " , slot_name ( p_slot ) ) ;
2007-03-07 02:02:26 +03:00
p_slot - > state = STATIC_STATE ;
break ;
case POWEROFF_STATE :
case POWERON_STATE :
/*
* Ignore if the slot is on power - on or power - off state ;
* this means that the previous attention button action
* to hot - add or hot - remove is undergoing
*/
2008-10-21 03:41:38 +04:00
ctrl_info ( ctrl , " Button ignore on Slot(%s) \n " , slot_name ( p_slot ) ) ;
2007-03-07 02:02:26 +03:00
break ;
default :
2008-09-05 07:11:26 +04:00
ctrl_warn ( ctrl , " Not a valid state \n " ) ;
2007-03-07 02:02:26 +03:00
break ;
2005-04-17 02:20:36 +04:00
}
}
2007-03-07 02:02:26 +03:00
/*
* Note : This function must be called with slot - > lock held
*/
static void handle_surprise_event ( struct slot * p_slot )
{
u8 getstatus ;
struct power_work_info * info ;
info = kmalloc ( sizeof ( * info ) , GFP_KERNEL ) ;
if ( ! info ) {
2008-09-05 07:11:26 +04:00
ctrl_err ( p_slot - > ctrl , " %s: Cannot allocate memory \n " ,
__func__ ) ;
2007-03-07 02:02:26 +03:00
return ;
}
info - > p_slot = p_slot ;
INIT_WORK ( & info - > work , pciehp_power_thread ) ;
2009-09-15 12:30:48 +04:00
pciehp_get_adapter_status ( p_slot , & getstatus ) ;
2007-03-07 02:02:26 +03:00
if ( ! getstatus )
p_slot - > state = POWEROFF_STATE ;
else
p_slot - > state = POWERON_STATE ;
queue_work ( pciehp_wq , & info - > work ) ;
}
static void interrupt_event_handler ( struct work_struct * work )
{
struct event_info * info = container_of ( work , struct event_info , work ) ;
struct slot * p_slot = info - > p_slot ;
struct controller * ctrl = p_slot - > ctrl ;
mutex_lock ( & p_slot - > lock ) ;
switch ( info - > event_type ) {
case INT_BUTTON_PRESS :
handle_button_press_event ( p_slot ) ;
break ;
case INT_POWER_FAULT :
2008-04-26 01:39:06 +04:00
if ( ! POWER_CTRL ( ctrl ) )
2007-03-07 02:02:26 +03:00
break ;
2008-04-26 01:39:06 +04:00
if ( ATTN_LED ( ctrl ) )
2009-09-15 12:30:48 +04:00
pciehp_set_attention_status ( p_slot , 1 ) ;
2008-04-26 01:39:06 +04:00
if ( PWR_LED ( ctrl ) )
2009-09-15 12:30:48 +04:00
pciehp_green_led_off ( p_slot ) ;
2007-03-07 02:02:26 +03:00
break ;
case INT_PRESENCE_ON :
case INT_PRESENCE_OFF :
2008-04-26 01:39:06 +04:00
if ( ! HP_SUPR_RM ( ctrl ) )
2007-03-07 02:02:26 +03:00
break ;
2008-09-05 07:11:26 +04:00
ctrl_dbg ( ctrl , " Surprise Removal \n " ) ;
2007-03-07 02:02:26 +03:00
handle_surprise_event ( p_slot ) ;
break ;
default :
break ;
}
mutex_unlock ( & p_slot - > lock ) ;
kfree ( info ) ;
}
2005-04-17 02:20:36 +04:00
int pciehp_enable_slot ( struct slot * p_slot )
{
u8 getstatus = 0 ;
int rc ;
2008-09-05 07:11:26 +04:00
struct controller * ctrl = p_slot - > ctrl ;
2005-04-17 02:20:36 +04:00
2009-09-15 12:30:48 +04:00
rc = pciehp_get_adapter_status ( p_slot , & getstatus ) ;
2005-04-17 02:20:36 +04:00
if ( rc | | ! getstatus ) {
2008-10-23 06:47:32 +04:00
ctrl_info ( ctrl , " No adapter on slot(%s) \n " , slot_name ( p_slot ) ) ;
2006-09-20 04:04:33 +04:00
return - ENODEV ;
2005-04-17 02:20:36 +04:00
}
2008-04-26 01:39:06 +04:00
if ( MRL_SENS ( p_slot - > ctrl ) ) {
2009-09-15 12:30:48 +04:00
rc = pciehp_get_latch_status ( p_slot , & getstatus ) ;
2005-04-17 02:20:36 +04:00
if ( rc | | getstatus ) {
2008-10-23 06:47:32 +04:00
ctrl_info ( ctrl , " Latch open on slot(%s) \n " ,
slot_name ( p_slot ) ) ;
2006-09-20 04:04:33 +04:00
return - ENODEV ;
2005-04-17 02:20:36 +04:00
}
}
2007-08-10 03:09:36 +04:00
2008-04-26 01:39:06 +04:00
if ( POWER_CTRL ( p_slot - > ctrl ) ) {
2009-09-15 12:30:48 +04:00
rc = pciehp_get_power_status ( p_slot , & getstatus ) ;
2005-04-17 02:20:36 +04:00
if ( rc | | getstatus ) {
2008-10-23 06:47:32 +04:00
ctrl_info ( ctrl , " Already enabled on slot(%s) \n " ,
slot_name ( p_slot ) ) ;
2006-09-20 04:04:33 +04:00
return - EINVAL ;
2005-04-17 02:20:36 +04:00
}
}
2009-09-15 12:30:48 +04:00
pciehp_get_latch_status ( p_slot , & getstatus ) ;
2005-04-17 02:20:36 +04:00
2005-11-01 03:20:08 +03:00
rc = board_added ( p_slot ) ;
2005-04-17 02:20:36 +04:00
if ( rc ) {
2009-09-15 12:30:48 +04:00
pciehp_get_latch_status ( p_slot , & getstatus ) ;
2005-04-17 02:20:36 +04:00
}
return rc ;
}
int pciehp_disable_slot ( struct slot * p_slot )
{
u8 getstatus = 0 ;
int ret = 0 ;
2008-09-05 07:11:26 +04:00
struct controller * ctrl = p_slot - > ctrl ;
2005-04-17 02:20:36 +04:00
if ( ! p_slot - > ctrl )
return 1 ;
2008-04-26 01:39:06 +04:00
if ( ! HP_SUPR_RM ( p_slot - > ctrl ) ) {
2009-09-15 12:30:48 +04:00
ret = pciehp_get_adapter_status ( p_slot , & getstatus ) ;
2005-04-17 02:20:36 +04:00
if ( ret | | ! getstatus ) {
2008-10-23 06:47:32 +04:00
ctrl_info ( ctrl , " No adapter on slot(%s) \n " ,
slot_name ( p_slot ) ) ;
2006-09-20 04:04:33 +04:00
return - ENODEV ;
2005-04-17 02:20:36 +04:00
}
}
2008-04-26 01:39:06 +04:00
if ( MRL_SENS ( p_slot - > ctrl ) ) {
2009-09-15 12:30:48 +04:00
ret = pciehp_get_latch_status ( p_slot , & getstatus ) ;
2005-04-17 02:20:36 +04:00
if ( ret | | getstatus ) {
2008-10-23 06:47:32 +04:00
ctrl_info ( ctrl , " Latch open on slot(%s) \n " ,
slot_name ( p_slot ) ) ;
2006-09-20 04:04:33 +04:00
return - ENODEV ;
2005-04-17 02:20:36 +04:00
}
}
2008-04-26 01:39:06 +04:00
if ( POWER_CTRL ( p_slot - > ctrl ) ) {
2009-09-15 12:30:48 +04:00
ret = pciehp_get_power_status ( p_slot , & getstatus ) ;
2005-04-17 02:20:36 +04:00
if ( ret | | ! getstatus ) {
2008-10-23 06:47:32 +04:00
ctrl_info ( ctrl , " Already disabled on slot(%s) \n " ,
slot_name ( p_slot ) ) ;
2006-09-20 04:04:33 +04:00
return - EINVAL ;
2005-04-17 02:20:36 +04:00
}
}
2009-10-05 12:42:59 +04:00
return remove_board ( p_slot ) ;
2005-04-17 02:20:36 +04:00
}
2007-03-07 02:02:26 +03:00
int pciehp_sysfs_enable_slot ( struct slot * p_slot )
{
int retval = - ENODEV ;
2008-09-05 07:11:26 +04:00
struct controller * ctrl = p_slot - > ctrl ;
2007-03-07 02:02:26 +03:00
mutex_lock ( & p_slot - > lock ) ;
switch ( p_slot - > state ) {
case BLINKINGON_STATE :
cancel_delayed_work ( & p_slot - > work ) ;
case STATIC_STATE :
p_slot - > state = POWERON_STATE ;
mutex_unlock ( & p_slot - > lock ) ;
retval = pciehp_enable_slot ( p_slot ) ;
mutex_lock ( & p_slot - > lock ) ;
p_slot - > state = STATIC_STATE ;
break ;
case POWERON_STATE :
2008-09-05 07:11:26 +04:00
ctrl_info ( ctrl , " Slot %s is already in powering on state \n " ,
2008-10-21 03:41:38 +04:00
slot_name ( p_slot ) ) ;
2007-03-07 02:02:26 +03:00
break ;
case BLINKINGOFF_STATE :
case POWEROFF_STATE :
2008-10-21 03:41:38 +04:00
ctrl_info ( ctrl , " Already enabled on slot %s \n " ,
slot_name ( p_slot ) ) ;
2007-03-07 02:02:26 +03:00
break ;
default :
2008-10-21 03:41:38 +04:00
ctrl_err ( ctrl , " Not a valid state on slot %s \n " ,
slot_name ( p_slot ) ) ;
2007-03-07 02:02:26 +03:00
break ;
}
mutex_unlock ( & p_slot - > lock ) ;
return retval ;
}
int pciehp_sysfs_disable_slot ( struct slot * p_slot )
{
int retval = - ENODEV ;
2008-09-05 07:11:26 +04:00
struct controller * ctrl = p_slot - > ctrl ;
2007-03-07 02:02:26 +03:00
mutex_lock ( & p_slot - > lock ) ;
switch ( p_slot - > state ) {
case BLINKINGOFF_STATE :
cancel_delayed_work ( & p_slot - > work ) ;
case STATIC_STATE :
p_slot - > state = POWEROFF_STATE ;
mutex_unlock ( & p_slot - > lock ) ;
retval = pciehp_disable_slot ( p_slot ) ;
mutex_lock ( & p_slot - > lock ) ;
p_slot - > state = STATIC_STATE ;
break ;
case POWEROFF_STATE :
2008-09-05 07:11:26 +04:00
ctrl_info ( ctrl , " Slot %s is already in powering off state \n " ,
2008-10-21 03:41:38 +04:00
slot_name ( p_slot ) ) ;
2007-03-07 02:02:26 +03:00
break ;
case BLINKINGON_STATE :
case POWERON_STATE :
2008-10-21 03:41:38 +04:00
ctrl_info ( ctrl , " Already disabled on slot %s \n " ,
slot_name ( p_slot ) ) ;
2007-03-07 02:02:26 +03:00
break ;
default :
2008-10-21 03:41:38 +04:00
ctrl_err ( ctrl , " Not a valid state on slot %s \n " ,
slot_name ( p_slot ) ) ;
2007-03-07 02:02:26 +03:00
break ;
}
mutex_unlock ( & p_slot - > lock ) ;
return retval ;
}