2005-04-16 15:20:36 -07:00
/*
* IBM Hot Plug Controller Driver
*
* Written By : Jyoti Shah , IBM Corporation
*
* Copyright ( C ) 2001 - 2003 IBM Corp .
*
* 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 .
*
* Send feedback to < gregkh @ us . ibm . com >
* < jshah @ us . ibm . com >
*
*/
# include <linux/wait.h>
# include <linux/time.h>
# include <linux/delay.h>
# include <linux/module.h>
# include <linux/pci.h>
# include <linux/init.h>
2006-01-13 16:02:15 +01:00
# include <linux/mutex.h>
Detach sched.h from mm.h
First thing mm.h does is including sched.h solely for can_do_mlock() inline
function which has "current" dereference inside. By dealing with can_do_mlock()
mm.h can be detached from sched.h which is good. See below, why.
This patch
a) removes unconditional inclusion of sched.h from mm.h
b) makes can_do_mlock() normal function in mm/mlock.c
c) exports can_do_mlock() to not break compilation
d) adds sched.h inclusions back to files that were getting it indirectly.
e) adds less bloated headers to some files (asm/signal.h, jiffies.h) that were
getting them indirectly
Net result is:
a) mm.h users would get less code to open, read, preprocess, parse, ... if
they don't need sched.h
b) sched.h stops being dependency for significant number of files:
on x86_64 allmodconfig touching sched.h results in recompile of 4083 files,
after patch it's only 3744 (-8.3%).
Cross-compile tested on
all arm defconfigs, all mips defconfigs, all powerpc defconfigs,
alpha alpha-up
arm
i386 i386-up i386-defconfig i386-allnoconfig
ia64 ia64-up
m68k
mips
parisc parisc-up
powerpc powerpc-up
s390 s390-up
sparc sparc-up
sparc64 sparc64-up
um-x86_64
x86_64 x86_64-up x86_64-defconfig x86_64-allnoconfig
as well as my two usual configs.
Signed-off-by: Alexey Dobriyan <adobriyan@gmail.com>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-05-21 01:22:52 +04:00
# include <linux/sched.h>
2010-02-04 12:12:23 -08:00
# include <linux/semaphore.h>
2007-08-14 16:07:00 -07:00
# include <linux/kthread.h>
2005-04-16 15:20:36 -07:00
# include "ibmphp.h"
2006-03-14 16:24:47 -08:00
static int to_debug = 0 ;
2015-12-27 13:21:11 -08:00
# define debug_polling(fmt, arg...) do { if (to_debug) debug(fmt, arg); } while (0)
2005-04-16 15:20:36 -07:00
//----------------------------------------------------------------------------
// timeout values
//----------------------------------------------------------------------------
# define CMD_COMPLETE_TOUT_SEC 60 // give HPC 60 sec to finish cmd
# define HPC_CTLR_WORKING_TOUT 60 // give HPC 60 sec to finish cmd
# define HPC_GETACCESS_TIMEOUT 60 // seconds
# define POLL_INTERVAL_SEC 2 // poll HPC every 2 seconds
# define POLL_LATCH_CNT 5 // poll latch 5 times, then poll slots
//----------------------------------------------------------------------------
// Winnipeg Architected Register Offsets
//----------------------------------------------------------------------------
# define WPG_I2CMBUFL_OFFSET 0x08 // I2C Message Buffer Low
# define WPG_I2CMOSUP_OFFSET 0x10 // I2C Master Operation Setup Reg
# define WPG_I2CMCNTL_OFFSET 0x20 // I2C Master Control Register
# define WPG_I2CPARM_OFFSET 0x40 // I2C Parameter Register
# define WPG_I2CSTAT_OFFSET 0x70 // I2C Status Register
//----------------------------------------------------------------------------
// Winnipeg Store Type commands (Add this commands to the register offset)
//----------------------------------------------------------------------------
# define WPG_I2C_AND 0x1000 // I2C AND operation
# define WPG_I2C_OR 0x2000 // I2C OR operation
//----------------------------------------------------------------------------
2005-05-03 18:38:30 -06:00
// Command set for I2C Master Operation Setup Register
2005-04-16 15:20:36 -07:00
//----------------------------------------------------------------------------
# define WPG_READATADDR_MASK 0x00010000 // read,bytes,I2C shifted,index
# define WPG_WRITEATADDR_MASK 0x40010000 // write,bytes,I2C shifted,index
# define WPG_READDIRECT_MASK 0x10010000
# define WPG_WRITEDIRECT_MASK 0x60010000
//----------------------------------------------------------------------------
// bit masks for I2C Master Control Register
//----------------------------------------------------------------------------
# define WPG_I2CMCNTL_STARTOP_MASK 0x00000002 // Start the Operation
//----------------------------------------------------------------------------
//
//----------------------------------------------------------------------------
# define WPG_I2C_IOREMAP_SIZE 0x2044 // size of linear address interval
//----------------------------------------------------------------------------
// command index
//----------------------------------------------------------------------------
# define WPG_1ST_SLOT_INDEX 0x01 // index - 1st slot for ctlr
# define WPG_CTLR_INDEX 0x0F // index - ctlr
# define WPG_1ST_EXTSLOT_INDEX 0x10 // index - 1st ext slot for ctlr
# define WPG_1ST_BUS_INDEX 0x1F // index - 1st bus for ctlr
//----------------------------------------------------------------------------
// macro utilities
//----------------------------------------------------------------------------
2006-03-14 16:24:47 -08:00
// if bits 20,22,25,26,27,29,30 are OFF return 1
# define HPC_I2CSTATUS_CHECK(s) ((u8)((s & 0x00000A76) ? 0 : 1))
2005-04-16 15:20:36 -07:00
//----------------------------------------------------------------------------
// global variables
//----------------------------------------------------------------------------
2006-01-13 16:02:15 +01:00
static struct mutex sem_hpcaccess ; // lock access to HPC
2005-04-16 15:20:36 -07:00
static struct semaphore semOperations ; // lock all operations and
// access to data structures
static struct semaphore sem_exit ; // make sure polling thread goes away
2007-08-14 16:07:00 -07:00
static struct task_struct * ibmphp_poll_thread ;
2005-04-16 15:20:36 -07:00
//----------------------------------------------------------------------------
// local function prototypes
//----------------------------------------------------------------------------
2015-12-27 13:21:11 -08:00
static u8 i2c_ctrl_read ( struct controller * , void __iomem * , u8 ) ;
static u8 i2c_ctrl_write ( struct controller * , void __iomem * , u8 , u8 ) ;
static u8 hpc_writecmdtoindex ( u8 , u8 ) ;
static u8 hpc_readcmdtoindex ( u8 , u8 ) ;
static void get_hpc_access ( void ) ;
static void free_hpc_access ( void ) ;
2007-08-14 16:07:00 -07:00
static int poll_hpc ( void * data ) ;
2015-12-27 13:21:11 -08:00
static int process_changeinstatus ( struct slot * , struct slot * ) ;
static int process_changeinlatch ( u8 , u8 , struct controller * ) ;
static int hpc_wait_ctlr_notworking ( int , struct controller * , void __iomem * , u8 * ) ;
2005-04-16 15:20:36 -07:00
//----------------------------------------------------------------------------
/*----------------------------------------------------------------------
* Name : ibmphp_hpc_initvars
*
* Action : initialize semaphores and variables
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2015-12-27 13:21:11 -08:00
void __init ibmphp_hpc_initvars ( void )
2005-04-16 15:20:36 -07:00
{
2015-12-27 13:21:11 -08:00
debug ( " %s - Entry \n " , __func__ ) ;
2005-04-16 15:20:36 -07:00
2006-01-13 16:02:15 +01:00
mutex_init ( & sem_hpcaccess ) ;
2010-09-07 14:32:38 +00:00
sema_init ( & semOperations , 1 ) ;
sema_init ( & sem_exit , 0 ) ;
2006-03-14 16:24:47 -08:00
to_debug = 0 ;
2005-04-16 15:20:36 -07:00
2015-12-27 13:21:11 -08:00
debug ( " %s - Exit \n " , __func__ ) ;
2005-04-16 15:20:36 -07:00
}
/*----------------------------------------------------------------------
* Name : i2c_ctrl_read
*
* Action : read from HPC over I2C
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2015-12-27 13:21:11 -08:00
static u8 i2c_ctrl_read ( struct controller * ctlr_ptr , void __iomem * WPGBbar , u8 index )
2005-04-16 15:20:36 -07:00
{
u8 status ;
int i ;
void __iomem * wpg_addr ; // base addr + offset
unsigned long wpg_data ; // data to/from WPG LOHI format
unsigned long ultemp ;
unsigned long data ; // actual data HILO format
2015-12-27 13:21:11 -08:00
debug_polling ( " %s - Entry WPGBbar[%p] index[%x] \n " , __func__ , WPGBbar , index ) ;
2005-04-16 15:20:36 -07:00
//--------------------------------------------------------------------
// READ - step 1
// read at address, byte length, I2C address (shifted), index
// or read direct, byte length, index
if ( ctlr_ptr - > ctlr_type = = 0x02 ) {
data = WPG_READATADDR_MASK ;
// fill in I2C address
ultemp = ( unsigned long ) ctlr_ptr - > u . wpeg_ctlr . i2c_addr ;
ultemp = ultemp > > 1 ;
data | = ( ultemp < < 8 ) ;
// fill in index
data | = ( unsigned long ) index ;
} else if ( ctlr_ptr - > ctlr_type = = 0x04 ) {
data = WPG_READDIRECT_MASK ;
// fill in index
ultemp = ( unsigned long ) index ;
ultemp = ultemp < < 8 ;
data | = ultemp ;
} else {
2015-12-27 13:21:11 -08:00
err ( " this controller type is not supported \n " ) ;
2005-04-16 15:20:36 -07:00
return HPC_ERROR ;
}
2015-12-27 13:21:11 -08:00
wpg_data = swab32 ( data ) ; // swap data before writing
2005-04-16 15:20:36 -07:00
wpg_addr = WPGBbar + WPG_I2CMOSUP_OFFSET ;
2015-12-27 13:21:11 -08:00
writel ( wpg_data , wpg_addr ) ;
2005-04-16 15:20:36 -07:00
//--------------------------------------------------------------------
// READ - step 2 : clear the message buffer
data = 0x00000000 ;
2015-12-27 13:21:11 -08:00
wpg_data = swab32 ( data ) ;
2005-04-16 15:20:36 -07:00
wpg_addr = WPGBbar + WPG_I2CMBUFL_OFFSET ;
2015-12-27 13:21:11 -08:00
writel ( wpg_data , wpg_addr ) ;
2005-04-16 15:20:36 -07:00
//--------------------------------------------------------------------
// READ - step 3 : issue start operation, I2C master control bit 30:ON
// 2020 : [20] OR operation at [20] offset 0x20
data = WPG_I2CMCNTL_STARTOP_MASK ;
2015-12-27 13:21:11 -08:00
wpg_data = swab32 ( data ) ;
2005-04-16 15:20:36 -07:00
wpg_addr = WPGBbar + WPG_I2CMCNTL_OFFSET + WPG_I2C_OR ;
2015-12-27 13:21:11 -08:00
writel ( wpg_data , wpg_addr ) ;
2005-04-16 15:20:36 -07:00
//--------------------------------------------------------------------
// READ - step 4 : wait until start operation bit clears
i = CMD_COMPLETE_TOUT_SEC ;
while ( i ) {
msleep ( 10 ) ;
wpg_addr = WPGBbar + WPG_I2CMCNTL_OFFSET ;
2015-12-27 13:21:11 -08:00
wpg_data = readl ( wpg_addr ) ;
data = swab32 ( wpg_data ) ;
2005-04-16 15:20:36 -07:00
if ( ! ( data & WPG_I2CMCNTL_STARTOP_MASK ) )
break ;
i - - ;
}
if ( i = = 0 ) {
2015-12-27 13:21:11 -08:00
debug ( " %s - Error : WPG timeout \n " , __func__ ) ;
2005-04-16 15:20:36 -07:00
return HPC_ERROR ;
}
//--------------------------------------------------------------------
// READ - step 5 : read I2C status register
i = CMD_COMPLETE_TOUT_SEC ;
while ( i ) {
msleep ( 10 ) ;
wpg_addr = WPGBbar + WPG_I2CSTAT_OFFSET ;
2015-12-27 13:21:11 -08:00
wpg_data = readl ( wpg_addr ) ;
data = swab32 ( wpg_data ) ;
if ( HPC_I2CSTATUS_CHECK ( data ) )
2005-04-16 15:20:36 -07:00
break ;
i - - ;
}
if ( i = = 0 ) {
2015-12-27 13:21:11 -08:00
debug ( " ctrl_read - Exit Error:I2C timeout \n " ) ;
2005-04-16 15:20:36 -07:00
return HPC_ERROR ;
}
//--------------------------------------------------------------------
// READ - step 6 : get DATA
wpg_addr = WPGBbar + WPG_I2CMBUFL_OFFSET ;
2015-12-27 13:21:11 -08:00
wpg_data = readl ( wpg_addr ) ;
data = swab32 ( wpg_data ) ;
2005-04-16 15:20:36 -07:00
status = ( u8 ) data ;
2015-12-27 13:21:11 -08:00
debug_polling ( " %s - Exit index[%x] status[%x] \n " , __func__ , index , status ) ;
2005-04-16 15:20:36 -07:00
return ( status ) ;
}
/*----------------------------------------------------------------------
* Name : i2c_ctrl_write
*
* Action : write to HPC over I2C
*
* Return 0 or error codes
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2015-12-27 13:21:11 -08:00
static u8 i2c_ctrl_write ( struct controller * ctlr_ptr , void __iomem * WPGBbar , u8 index , u8 cmd )
2005-04-16 15:20:36 -07:00
{
u8 rc ;
void __iomem * wpg_addr ; // base addr + offset
2013-11-14 11:28:18 -07:00
unsigned long wpg_data ; // data to/from WPG LOHI format
2005-04-16 15:20:36 -07:00
unsigned long ultemp ;
unsigned long data ; // actual data HILO format
int i ;
2015-12-27 13:21:11 -08:00
debug_polling ( " %s - Entry WPGBbar[%p] index[%x] cmd[%x] \n " , __func__ , WPGBbar , index , cmd ) ;
2005-04-16 15:20:36 -07:00
rc = 0 ;
//--------------------------------------------------------------------
// WRITE - step 1
// write at address, byte length, I2C address (shifted), index
// or write direct, byte length, index
data = 0x00000000 ;
if ( ctlr_ptr - > ctlr_type = = 0x02 ) {
data = WPG_WRITEATADDR_MASK ;
// fill in I2C address
ultemp = ( unsigned long ) ctlr_ptr - > u . wpeg_ctlr . i2c_addr ;
ultemp = ultemp > > 1 ;
data | = ( ultemp < < 8 ) ;
// fill in index
data | = ( unsigned long ) index ;
} else if ( ctlr_ptr - > ctlr_type = = 0x04 ) {
data = WPG_WRITEDIRECT_MASK ;
// fill in index
ultemp = ( unsigned long ) index ;
ultemp = ultemp < < 8 ;
data | = ultemp ;
} else {
2015-12-27 13:21:11 -08:00
err ( " this controller type is not supported \n " ) ;
2005-04-16 15:20:36 -07:00
return HPC_ERROR ;
}
2015-12-27 13:21:11 -08:00
wpg_data = swab32 ( data ) ; // swap data before writing
2005-04-16 15:20:36 -07:00
wpg_addr = WPGBbar + WPG_I2CMOSUP_OFFSET ;
2015-12-27 13:21:11 -08:00
writel ( wpg_data , wpg_addr ) ;
2005-04-16 15:20:36 -07:00
//--------------------------------------------------------------------
// WRITE - step 2 : clear the message buffer
data = 0x00000000 | ( unsigned long ) cmd ;
2015-12-27 13:21:11 -08:00
wpg_data = swab32 ( data ) ;
2005-04-16 15:20:36 -07:00
wpg_addr = WPGBbar + WPG_I2CMBUFL_OFFSET ;
2015-12-27 13:21:11 -08:00
writel ( wpg_data , wpg_addr ) ;
2005-04-16 15:20:36 -07:00
//--------------------------------------------------------------------
// WRITE - step 3 : issue start operation,I2C master control bit 30:ON
// 2020 : [20] OR operation at [20] offset 0x20
data = WPG_I2CMCNTL_STARTOP_MASK ;
2015-12-27 13:21:11 -08:00
wpg_data = swab32 ( data ) ;
2005-04-16 15:20:36 -07:00
wpg_addr = WPGBbar + WPG_I2CMCNTL_OFFSET + WPG_I2C_OR ;
2015-12-27 13:21:11 -08:00
writel ( wpg_data , wpg_addr ) ;
2005-04-16 15:20:36 -07:00
//--------------------------------------------------------------------
// WRITE - step 4 : wait until start operation bit clears
i = CMD_COMPLETE_TOUT_SEC ;
while ( i ) {
msleep ( 10 ) ;
wpg_addr = WPGBbar + WPG_I2CMCNTL_OFFSET ;
2015-12-27 13:21:11 -08:00
wpg_data = readl ( wpg_addr ) ;
data = swab32 ( wpg_data ) ;
2005-04-16 15:20:36 -07:00
if ( ! ( data & WPG_I2CMCNTL_STARTOP_MASK ) )
break ;
i - - ;
}
if ( i = = 0 ) {
2015-12-27 13:21:11 -08:00
debug ( " %s - Exit Error:WPG timeout \n " , __func__ ) ;
2005-04-16 15:20:36 -07:00
rc = HPC_ERROR ;
}
//--------------------------------------------------------------------
// WRITE - step 5 : read I2C status register
i = CMD_COMPLETE_TOUT_SEC ;
while ( i ) {
msleep ( 10 ) ;
wpg_addr = WPGBbar + WPG_I2CSTAT_OFFSET ;
2015-12-27 13:21:11 -08:00
wpg_data = readl ( wpg_addr ) ;
data = swab32 ( wpg_data ) ;
if ( HPC_I2CSTATUS_CHECK ( data ) )
2005-04-16 15:20:36 -07:00
break ;
i - - ;
}
if ( i = = 0 ) {
2015-12-27 13:21:11 -08:00
debug ( " ctrl_read - Error : I2C timeout \n " ) ;
2005-04-16 15:20:36 -07:00
rc = HPC_ERROR ;
}
2015-12-27 13:21:11 -08:00
debug_polling ( " %s Exit rc[%x] \n " , __func__ , rc ) ;
2005-04-16 15:20:36 -07:00
return ( rc ) ;
}
//------------------------------------------------------------
2013-11-14 11:28:18 -07:00
// Read from ISA type HPC
2005-04-16 15:20:36 -07:00
//------------------------------------------------------------
2015-12-27 13:21:11 -08:00
static u8 isa_ctrl_read ( struct controller * ctlr_ptr , u8 offset )
2005-04-16 15:20:36 -07:00
{
u16 start_address ;
u16 end_address ;
u8 data ;
start_address = ctlr_ptr - > u . isa_ctlr . io_start ;
end_address = ctlr_ptr - > u . isa_ctlr . io_end ;
2015-12-27 13:21:11 -08:00
data = inb ( start_address + offset ) ;
2005-04-16 15:20:36 -07:00
return data ;
}
//--------------------------------------------------------------
// Write to ISA type HPC
//--------------------------------------------------------------
2015-12-27 13:21:11 -08:00
static void isa_ctrl_write ( struct controller * ctlr_ptr , u8 offset , u8 data )
2005-04-16 15:20:36 -07:00
{
u16 start_address ;
u16 port_address ;
2013-11-14 11:28:18 -07:00
2005-04-16 15:20:36 -07:00
start_address = ctlr_ptr - > u . isa_ctlr . io_start ;
port_address = start_address + ( u16 ) offset ;
2015-12-27 13:21:11 -08:00
outb ( data , port_address ) ;
2005-04-16 15:20:36 -07:00
}
2015-12-27 13:21:11 -08:00
static u8 pci_ctrl_read ( struct controller * ctrl , u8 offset )
2005-04-16 15:20:36 -07:00
{
u8 data = 0x00 ;
2015-12-27 13:21:11 -08:00
debug ( " inside pci_ctrl_read \n " ) ;
2005-04-16 15:20:36 -07:00
if ( ctrl - > ctrl_dev )
2015-12-27 13:21:11 -08:00
pci_read_config_byte ( ctrl - > ctrl_dev , HPC_PCI_OFFSET + offset , & data ) ;
2005-04-16 15:20:36 -07:00
return data ;
}
2015-12-27 13:21:11 -08:00
static u8 pci_ctrl_write ( struct controller * ctrl , u8 offset , u8 data )
2005-04-16 15:20:36 -07:00
{
u8 rc = - ENODEV ;
2015-12-27 13:21:11 -08:00
debug ( " inside pci_ctrl_write \n " ) ;
2005-04-16 15:20:36 -07:00
if ( ctrl - > ctrl_dev ) {
2015-12-27 13:21:11 -08:00
pci_write_config_byte ( ctrl - > ctrl_dev , HPC_PCI_OFFSET + offset , data ) ;
2005-04-16 15:20:36 -07:00
rc = 0 ;
}
return rc ;
}
2015-12-27 13:21:11 -08:00
static u8 ctrl_read ( struct controller * ctlr , void __iomem * base , u8 offset )
2005-04-16 15:20:36 -07:00
{
u8 rc ;
switch ( ctlr - > ctlr_type ) {
case 0 :
2015-12-27 13:21:11 -08:00
rc = isa_ctrl_read ( ctlr , offset ) ;
2005-04-16 15:20:36 -07:00
break ;
case 1 :
2015-12-27 13:21:11 -08:00
rc = pci_ctrl_read ( ctlr , offset ) ;
2005-04-16 15:20:36 -07:00
break ;
case 2 :
case 4 :
2015-12-27 13:21:11 -08:00
rc = i2c_ctrl_read ( ctlr , base , offset ) ;
2005-04-16 15:20:36 -07:00
break ;
default :
return - ENODEV ;
}
return rc ;
}
2015-12-27 13:21:11 -08:00
static u8 ctrl_write ( struct controller * ctlr , void __iomem * base , u8 offset , u8 data )
2005-04-16 15:20:36 -07:00
{
u8 rc = 0 ;
switch ( ctlr - > ctlr_type ) {
case 0 :
isa_ctrl_write ( ctlr , offset , data ) ;
break ;
case 1 :
2015-12-27 13:21:11 -08:00
rc = pci_ctrl_write ( ctlr , offset , data ) ;
2005-04-16 15:20:36 -07:00
break ;
case 2 :
case 4 :
rc = i2c_ctrl_write ( ctlr , base , offset , data ) ;
break ;
default :
return - ENODEV ;
}
return rc ;
}
/*----------------------------------------------------------------------
* Name : hpc_writecmdtoindex ( )
*
* Action : convert a write command to proper index within a controller
*
* Return index , HPC_ERROR
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2015-12-27 13:21:11 -08:00
static u8 hpc_writecmdtoindex ( u8 cmd , u8 index )
2005-04-16 15:20:36 -07:00
{
u8 rc ;
switch ( cmd ) {
case HPC_CTLR_ENABLEIRQ : // 0x00.N.15
case HPC_CTLR_CLEARIRQ : // 0x06.N.15
case HPC_CTLR_RESET : // 0x07.N.15
case HPC_CTLR_IRQSTEER : // 0x08.N.15
case HPC_CTLR_DISABLEIRQ : // 0x01.N.15
case HPC_ALLSLOT_ON : // 0x11.N.15
case HPC_ALLSLOT_OFF : // 0x12.N.15
rc = 0x0F ;
break ;
case HPC_SLOT_OFF : // 0x02.Y.0-14
case HPC_SLOT_ON : // 0x03.Y.0-14
case HPC_SLOT_ATTNOFF : // 0x04.N.0-14
case HPC_SLOT_ATTNON : // 0x05.N.0-14
case HPC_SLOT_BLINKLED : // 0x13.N.0-14
rc = index ;
break ;
case HPC_BUS_33CONVMODE :
case HPC_BUS_66CONVMODE :
case HPC_BUS_66PCIXMODE :
case HPC_BUS_100PCIXMODE :
case HPC_BUS_133PCIXMODE :
rc = index + WPG_1ST_BUS_INDEX - 1 ;
break ;
default :
2015-12-27 13:21:11 -08:00
err ( " hpc_writecmdtoindex - Error invalid cmd[%x] \n " , cmd ) ;
2005-04-16 15:20:36 -07:00
rc = HPC_ERROR ;
}
return rc ;
}
/*----------------------------------------------------------------------
* Name : hpc_readcmdtoindex ( )
*
* Action : convert a read command to proper index within a controller
*
* Return index , HPC_ERROR
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2015-12-27 13:21:11 -08:00
static u8 hpc_readcmdtoindex ( u8 cmd , u8 index )
2005-04-16 15:20:36 -07:00
{
u8 rc ;
switch ( cmd ) {
case READ_CTLRSTATUS :
rc = 0x0F ;
break ;
case READ_SLOTSTATUS :
case READ_ALLSTAT :
rc = index ;
break ;
case READ_EXTSLOTSTATUS :
rc = index + WPG_1ST_EXTSLOT_INDEX ;
break ;
case READ_BUSSTATUS :
rc = index + WPG_1ST_BUS_INDEX - 1 ;
break ;
case READ_SLOTLATCHLOWREG :
rc = 0x28 ;
break ;
case READ_REVLEVEL :
rc = 0x25 ;
break ;
case READ_HPCOPTIONS :
rc = 0x27 ;
break ;
default :
rc = HPC_ERROR ;
}
return rc ;
}
/*----------------------------------------------------------------------
* Name : HPCreadslot ( )
*
* Action : issue a READ command to HPC
*
2006-11-30 05:24:39 +01:00
* Input : pslot - cannot be NULL for READ_ALLSTAT
2005-04-16 15:20:36 -07:00
* pstatus - can be NULL for READ_ALLSTAT
*
* Return 0 or error codes
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2015-12-27 13:21:11 -08:00
int ibmphp_hpc_readslot ( struct slot * pslot , u8 cmd , u8 * pstatus )
2005-04-16 15:20:36 -07:00
{
void __iomem * wpg_bbar = NULL ;
struct controller * ctlr_ptr ;
u8 index , status ;
int rc = 0 ;
int busindex ;
2015-12-27 13:21:11 -08:00
debug_polling ( " %s - Entry pslot[%p] cmd[%x] pstatus[%p] \n " , __func__ , pslot , cmd , pstatus ) ;
2005-04-16 15:20:36 -07:00
if ( ( pslot = = NULL )
| | ( ( pstatus = = NULL ) & & ( cmd ! = READ_ALLSTAT ) & & ( cmd ! = READ_BUSSTATUS ) ) ) {
rc = - EINVAL ;
2015-12-27 13:21:11 -08:00
err ( " %s - Error invalid pointer, rc[%d] \n " , __func__ , rc ) ;
2005-04-16 15:20:36 -07:00
return rc ;
}
if ( cmd = = READ_BUSSTATUS ) {
2015-12-27 13:21:11 -08:00
busindex = ibmphp_get_bus_index ( pslot - > bus ) ;
2005-04-16 15:20:36 -07:00
if ( busindex < 0 ) {
rc = - EINVAL ;
2015-12-27 13:21:11 -08:00
err ( " %s - Exit Error:invalid bus, rc[%d] \n " , __func__ , rc ) ;
2005-04-16 15:20:36 -07:00
return rc ;
} else
index = ( u8 ) busindex ;
} else
index = pslot - > ctlr_index ;
2015-12-27 13:21:11 -08:00
index = hpc_readcmdtoindex ( cmd , index ) ;
2005-04-16 15:20:36 -07:00
if ( index = = HPC_ERROR ) {
rc = - EINVAL ;
2015-12-27 13:21:11 -08:00
err ( " %s - Exit Error:invalid index, rc[%d] \n " , __func__ , rc ) ;
2005-04-16 15:20:36 -07:00
return rc ;
}
ctlr_ptr = pslot - > ctrl ;
2015-12-27 13:21:11 -08:00
get_hpc_access ( ) ;
2005-04-16 15:20:36 -07:00
//--------------------------------------------------------------------
// map physical address to logical address
//--------------------------------------------------------------------
if ( ( ctlr_ptr - > ctlr_type = = 2 ) | | ( ctlr_ptr - > ctlr_type = = 4 ) )
2015-12-27 13:21:11 -08:00
wpg_bbar = ioremap ( ctlr_ptr - > u . wpeg_ctlr . wpegbbar , WPG_I2C_IOREMAP_SIZE ) ;
2005-04-16 15:20:36 -07:00
//--------------------------------------------------------------------
// check controller status before reading
//--------------------------------------------------------------------
2015-12-27 13:21:11 -08:00
rc = hpc_wait_ctlr_notworking ( HPC_CTLR_WORKING_TOUT , ctlr_ptr , wpg_bbar , & status ) ;
2005-04-16 15:20:36 -07:00
if ( ! rc ) {
switch ( cmd ) {
case READ_ALLSTAT :
// update the slot structure
pslot - > ctrl - > status = status ;
2015-12-27 13:21:11 -08:00
pslot - > status = ctrl_read ( ctlr_ptr , wpg_bbar , index ) ;
rc = hpc_wait_ctlr_notworking ( HPC_CTLR_WORKING_TOUT , ctlr_ptr , wpg_bbar ,
2005-04-16 15:20:36 -07:00
& status ) ;
if ( ! rc )
2015-12-27 13:21:11 -08:00
pslot - > ext_status = ctrl_read ( ctlr_ptr , wpg_bbar , index + WPG_1ST_EXTSLOT_INDEX ) ;
2005-04-16 15:20:36 -07:00
break ;
case READ_SLOTSTATUS :
// DO NOT update the slot structure
2015-12-27 13:21:11 -08:00
* pstatus = ctrl_read ( ctlr_ptr , wpg_bbar , index ) ;
2005-04-16 15:20:36 -07:00
break ;
case READ_EXTSLOTSTATUS :
// DO NOT update the slot structure
2015-12-27 13:21:11 -08:00
* pstatus = ctrl_read ( ctlr_ptr , wpg_bbar , index ) ;
2005-04-16 15:20:36 -07:00
break ;
case READ_CTLRSTATUS :
// DO NOT update the slot structure
* pstatus = status ;
break ;
case READ_BUSSTATUS :
2015-12-27 13:21:11 -08:00
pslot - > busstatus = ctrl_read ( ctlr_ptr , wpg_bbar , index ) ;
2005-04-16 15:20:36 -07:00
break ;
case READ_REVLEVEL :
2015-12-27 13:21:11 -08:00
* pstatus = ctrl_read ( ctlr_ptr , wpg_bbar , index ) ;
2005-04-16 15:20:36 -07:00
break ;
case READ_HPCOPTIONS :
2015-12-27 13:21:11 -08:00
* pstatus = ctrl_read ( ctlr_ptr , wpg_bbar , index ) ;
2005-04-16 15:20:36 -07:00
break ;
case READ_SLOTLATCHLOWREG :
// DO NOT update the slot structure
2015-12-27 13:21:11 -08:00
* pstatus = ctrl_read ( ctlr_ptr , wpg_bbar , index ) ;
2005-04-16 15:20:36 -07:00
break ;
// Not used
case READ_ALLSLOT :
2015-12-12 21:36:57 +08:00
list_for_each_entry ( pslot , & ibmphp_slot_head ,
ibm_slot_list ) {
2005-04-16 15:20:36 -07:00
index = pslot - > ctlr_index ;
2015-12-27 13:21:11 -08:00
rc = hpc_wait_ctlr_notworking ( HPC_CTLR_WORKING_TOUT , ctlr_ptr ,
2005-04-16 15:20:36 -07:00
wpg_bbar , & status ) ;
if ( ! rc ) {
2015-12-27 13:21:11 -08:00
pslot - > status = ctrl_read ( ctlr_ptr , wpg_bbar , index ) ;
rc = hpc_wait_ctlr_notworking ( HPC_CTLR_WORKING_TOUT ,
2005-04-16 15:20:36 -07:00
ctlr_ptr , wpg_bbar , & status ) ;
if ( ! rc )
pslot - > ext_status =
2015-12-27 13:21:11 -08:00
ctrl_read ( ctlr_ptr , wpg_bbar ,
2005-04-16 15:20:36 -07:00
index + WPG_1ST_EXTSLOT_INDEX ) ;
} else {
2015-12-27 13:21:11 -08:00
err ( " %s - Error ctrl_read failed \n " , __func__ ) ;
2005-04-16 15:20:36 -07:00
rc = - EINVAL ;
break ;
}
}
break ;
default :
rc = - EINVAL ;
break ;
}
}
//--------------------------------------------------------------------
// cleanup
//--------------------------------------------------------------------
2013-11-14 11:28:18 -07:00
2005-04-16 15:20:36 -07:00
// remove physical to logical address mapping
if ( ( ctlr_ptr - > ctlr_type = = 2 ) | | ( ctlr_ptr - > ctlr_type = = 4 ) )
2015-12-27 13:21:11 -08:00
iounmap ( wpg_bbar ) ;
2013-11-14 11:28:18 -07:00
2015-12-27 13:21:11 -08:00
free_hpc_access ( ) ;
2005-04-16 15:20:36 -07:00
2015-12-27 13:21:11 -08:00
debug_polling ( " %s - Exit rc[%d] \n " , __func__ , rc ) ;
2005-04-16 15:20:36 -07:00
return rc ;
}
/*----------------------------------------------------------------------
* Name : ibmphp_hpc_writeslot ( )
*
* Action : issue a WRITE command to HPC
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2015-12-27 13:21:11 -08:00
int ibmphp_hpc_writeslot ( struct slot * pslot , u8 cmd )
2005-04-16 15:20:36 -07:00
{
void __iomem * wpg_bbar = NULL ;
struct controller * ctlr_ptr ;
u8 index , status ;
int busindex ;
u8 done ;
int rc = 0 ;
int timeout ;
2015-12-27 13:21:11 -08:00
debug_polling ( " %s - Entry pslot[%p] cmd[%x] \n " , __func__ , pslot , cmd ) ;
2005-04-16 15:20:36 -07:00
if ( pslot = = NULL ) {
rc = - EINVAL ;
2015-12-27 13:21:11 -08:00
err ( " %s - Error Exit rc[%d] \n " , __func__ , rc ) ;
2005-04-16 15:20:36 -07:00
return rc ;
}
if ( ( cmd = = HPC_BUS_33CONVMODE ) | | ( cmd = = HPC_BUS_66CONVMODE ) | |
( cmd = = HPC_BUS_66PCIXMODE ) | | ( cmd = = HPC_BUS_100PCIXMODE ) | |
( cmd = = HPC_BUS_133PCIXMODE ) ) {
2015-12-27 13:21:11 -08:00
busindex = ibmphp_get_bus_index ( pslot - > bus ) ;
2005-04-16 15:20:36 -07:00
if ( busindex < 0 ) {
rc = - EINVAL ;
2015-12-27 13:21:11 -08:00
err ( " %s - Exit Error:invalid bus, rc[%d] \n " , __func__ , rc ) ;
2005-04-16 15:20:36 -07:00
return rc ;
} else
index = ( u8 ) busindex ;
} else
index = pslot - > ctlr_index ;
2015-12-27 13:21:11 -08:00
index = hpc_writecmdtoindex ( cmd , index ) ;
2005-04-16 15:20:36 -07:00
if ( index = = HPC_ERROR ) {
rc = - EINVAL ;
2015-12-27 13:21:11 -08:00
err ( " %s - Error Exit rc[%d] \n " , __func__ , rc ) ;
2005-04-16 15:20:36 -07:00
return rc ;
}
ctlr_ptr = pslot - > ctrl ;
2015-12-27 13:21:11 -08:00
get_hpc_access ( ) ;
2005-04-16 15:20:36 -07:00
//--------------------------------------------------------------------
// map physical address to logical address
//--------------------------------------------------------------------
if ( ( ctlr_ptr - > ctlr_type = = 2 ) | | ( ctlr_ptr - > ctlr_type = = 4 ) ) {
2015-12-27 13:21:11 -08:00
wpg_bbar = ioremap ( ctlr_ptr - > u . wpeg_ctlr . wpegbbar , WPG_I2C_IOREMAP_SIZE ) ;
2005-04-16 15:20:36 -07:00
2015-12-27 13:21:11 -08:00
debug ( " %s - ctlr id[%x] physical[%lx] logical[%lx] i2c[%x] \n " , __func__ ,
2005-04-16 15:20:36 -07:00
ctlr_ptr - > ctlr_id , ( ulong ) ( ctlr_ptr - > u . wpeg_ctlr . wpegbbar ) , ( ulong ) wpg_bbar ,
ctlr_ptr - > u . wpeg_ctlr . i2c_addr ) ;
}
//--------------------------------------------------------------------
// check controller status before writing
//--------------------------------------------------------------------
2015-12-27 13:21:11 -08:00
rc = hpc_wait_ctlr_notworking ( HPC_CTLR_WORKING_TOUT , ctlr_ptr , wpg_bbar , & status ) ;
2005-04-16 15:20:36 -07:00
if ( ! rc ) {
2015-12-27 13:21:11 -08:00
ctrl_write ( ctlr_ptr , wpg_bbar , index , cmd ) ;
2005-04-16 15:20:36 -07:00
//--------------------------------------------------------------------
// check controller is still not working on the command
//--------------------------------------------------------------------
timeout = CMD_COMPLETE_TOUT_SEC ;
2006-03-14 16:24:47 -08:00
done = 0 ;
2005-04-16 15:20:36 -07:00
while ( ! done ) {
2015-12-27 13:21:11 -08:00
rc = hpc_wait_ctlr_notworking ( HPC_CTLR_WORKING_TOUT , ctlr_ptr , wpg_bbar ,
2005-04-16 15:20:36 -07:00
& status ) ;
if ( ! rc ) {
2015-12-27 13:21:11 -08:00
if ( NEEDTOCHECK_CMDSTATUS ( cmd ) ) {
if ( CTLR_FINISHED ( status ) = = HPC_CTLR_FINISHED_YES )
2006-03-14 16:24:47 -08:00
done = 1 ;
2005-04-16 15:20:36 -07:00
} else
2006-03-14 16:24:47 -08:00
done = 1 ;
2005-04-16 15:20:36 -07:00
}
if ( ! done ) {
msleep ( 1000 ) ;
if ( timeout < 1 ) {
2006-03-14 16:24:47 -08:00
done = 1 ;
2015-12-27 13:21:11 -08:00
err ( " %s - Error command complete timeout \n " , __func__ ) ;
2005-04-16 15:20:36 -07:00
rc = - EFAULT ;
} else
timeout - - ;
}
}
ctlr_ptr - > status = status ;
}
// cleanup
// remove physical to logical address mapping
if ( ( ctlr_ptr - > ctlr_type = = 2 ) | | ( ctlr_ptr - > ctlr_type = = 4 ) )
2015-12-27 13:21:11 -08:00
iounmap ( wpg_bbar ) ;
free_hpc_access ( ) ;
2005-04-16 15:20:36 -07:00
2015-12-27 13:21:11 -08:00
debug_polling ( " %s - Exit rc[%d] \n " , __func__ , rc ) ;
2005-04-16 15:20:36 -07:00
return rc ;
}
/*----------------------------------------------------------------------
* Name : get_hpc_access ( )
*
* Action : make sure only one process can access HPC at one time
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2015-12-27 13:21:11 -08:00
static void get_hpc_access ( void )
2005-04-16 15:20:36 -07:00
{
2006-01-13 16:02:15 +01:00
mutex_lock ( & sem_hpcaccess ) ;
2005-04-16 15:20:36 -07:00
}
/*----------------------------------------------------------------------
* Name : free_hpc_access ( )
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2015-12-27 13:21:11 -08:00
void free_hpc_access ( void )
2005-04-16 15:20:36 -07:00
{
2006-01-13 16:02:15 +01:00
mutex_unlock ( & sem_hpcaccess ) ;
2005-04-16 15:20:36 -07:00
}
/*----------------------------------------------------------------------
* Name : ibmphp_lock_operations ( )
*
* Action : make sure only one process can change the data structure
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2015-12-27 13:21:11 -08:00
void ibmphp_lock_operations ( void )
2005-04-16 15:20:36 -07:00
{
2015-12-27 13:21:11 -08:00
down ( & semOperations ) ;
2006-03-14 16:24:47 -08:00
to_debug = 1 ;
2005-04-16 15:20:36 -07:00
}
/*----------------------------------------------------------------------
* Name : ibmphp_unlock_operations ( )
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2015-12-27 13:21:11 -08:00
void ibmphp_unlock_operations ( void )
2005-04-16 15:20:36 -07:00
{
2015-12-27 13:21:11 -08:00
debug ( " %s - Entry \n " , __func__ ) ;
up ( & semOperations ) ;
2006-03-14 16:24:47 -08:00
to_debug = 0 ;
2015-12-27 13:21:11 -08:00
debug ( " %s - Exit \n " , __func__ ) ;
2005-04-16 15:20:36 -07:00
}
/*----------------------------------------------------------------------
* Name : poll_hpc ( )
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
# define POLL_LATCH_REGISTER 0
# define POLL_SLOTS 1
# define POLL_SLEEP 2
2007-08-14 16:07:00 -07:00
static int poll_hpc ( void * data )
2005-04-16 15:20:36 -07:00
{
struct slot myslot ;
struct slot * pslot = NULL ;
int rc ;
int poll_state = POLL_LATCH_REGISTER ;
u8 oldlatchlow = 0x00 ;
u8 curlatchlow = 0x00 ;
int poll_count = 0 ;
u8 ctrl_count = 0x00 ;
2015-12-27 13:21:11 -08:00
debug ( " %s - Entry \n " , __func__ ) ;
2005-04-16 15:20:36 -07:00
2007-08-14 16:07:00 -07:00
while ( ! kthread_should_stop ( ) ) {
2005-05-03 18:38:30 -06:00
/* try to get the lock to do some kind of hardware access */
2015-12-27 13:21:11 -08:00
down ( & semOperations ) ;
2005-04-16 15:20:36 -07:00
switch ( poll_state ) {
2013-11-14 11:28:18 -07:00
case POLL_LATCH_REGISTER :
2005-04-16 15:20:36 -07:00
oldlatchlow = curlatchlow ;
ctrl_count = 0x00 ;
2015-12-12 21:36:57 +08:00
list_for_each_entry ( pslot , & ibmphp_slot_head ,
ibm_slot_list ) {
2005-04-16 15:20:36 -07:00
if ( ctrl_count > = ibmphp_get_total_controllers ( ) )
break ;
if ( pslot - > ctrl - > ctlr_relative_id = = ctrl_count ) {
ctrl_count + + ;
2015-12-27 13:21:11 -08:00
if ( READ_SLOT_LATCH ( pslot - > ctrl ) ) {
rc = ibmphp_hpc_readslot ( pslot ,
2005-04-16 15:20:36 -07:00
READ_SLOTLATCHLOWREG ,
& curlatchlow ) ;
if ( oldlatchlow ! = curlatchlow )
2015-12-27 13:21:11 -08:00
process_changeinlatch ( oldlatchlow ,
2005-04-16 15:20:36 -07:00
curlatchlow ,
pslot - > ctrl ) ;
}
}
}
+ + poll_count ;
poll_state = POLL_SLEEP ;
break ;
case POLL_SLOTS :
2015-12-12 21:36:57 +08:00
list_for_each_entry ( pslot , & ibmphp_slot_head ,
ibm_slot_list ) {
2005-04-16 15:20:36 -07:00
// make a copy of the old status
2015-12-27 13:21:11 -08:00
memcpy ( ( void * ) & myslot , ( void * ) pslot ,
sizeof ( struct slot ) ) ;
rc = ibmphp_hpc_readslot ( pslot , READ_ALLSTAT , NULL ) ;
2005-04-16 15:20:36 -07:00
if ( ( myslot . status ! = pslot - > status )
| | ( myslot . ext_status ! = pslot - > ext_status ) )
2015-12-27 13:21:11 -08:00
process_changeinstatus ( pslot , & myslot ) ;
2005-04-16 15:20:36 -07:00
}
ctrl_count = 0x00 ;
2015-12-12 21:36:57 +08:00
list_for_each_entry ( pslot , & ibmphp_slot_head ,
ibm_slot_list ) {
2005-04-16 15:20:36 -07:00
if ( ctrl_count > = ibmphp_get_total_controllers ( ) )
break ;
if ( pslot - > ctrl - > ctlr_relative_id = = ctrl_count ) {
ctrl_count + + ;
2015-12-27 13:21:11 -08:00
if ( READ_SLOT_LATCH ( pslot - > ctrl ) )
rc = ibmphp_hpc_readslot ( pslot ,
2005-04-16 15:20:36 -07:00
READ_SLOTLATCHLOWREG ,
& curlatchlow ) ;
}
}
+ + poll_count ;
poll_state = POLL_SLEEP ;
break ;
case POLL_SLEEP :
/* don't sleep with a lock on the hardware */
2015-12-27 13:21:11 -08:00
up ( & semOperations ) ;
2005-04-16 15:20:36 -07:00
msleep ( POLL_INTERVAL_SEC * 1000 ) ;
2007-08-14 16:07:00 -07:00
if ( kthread_should_stop ( ) )
2009-12-04 15:18:01 -08:00
goto out_sleep ;
2013-11-14 11:28:18 -07:00
2015-12-27 13:21:11 -08:00
down ( & semOperations ) ;
2013-11-14 11:28:18 -07:00
2005-04-16 15:20:36 -07:00
if ( poll_count > = POLL_LATCH_CNT ) {
poll_count = 0 ;
poll_state = POLL_SLOTS ;
} else
poll_state = POLL_LATCH_REGISTER ;
break ;
2013-11-14 11:28:18 -07:00
}
2005-05-03 18:38:30 -06:00
/* give up the hardware semaphore */
2015-12-27 13:21:11 -08:00
up ( & semOperations ) ;
2005-04-16 15:20:36 -07:00
/* sleep for a short time just for good measure */
2009-12-04 15:18:01 -08:00
out_sleep :
2005-04-16 15:20:36 -07:00
msleep ( 100 ) ;
}
2015-12-27 13:21:11 -08:00
up ( & sem_exit ) ;
debug ( " %s - Exit \n " , __func__ ) ;
2007-08-14 16:07:00 -07:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
/*----------------------------------------------------------------------
* Name : process_changeinstatus
*
* Action : compare old and new slot status , process the change in status
*
* Input : pointer to slot struct , old slot struct
*
* Return 0 or error codes
* Value :
*
* Side
* Effects : None .
*
* Notes :
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2015-12-27 13:21:11 -08:00
static int process_changeinstatus ( struct slot * pslot , struct slot * poldslot )
2005-04-16 15:20:36 -07:00
{
u8 status ;
int rc = 0 ;
2006-03-14 16:24:47 -08:00
u8 disable = 0 ;
u8 update = 0 ;
2005-04-16 15:20:36 -07:00
2015-12-27 13:21:11 -08:00
debug ( " process_changeinstatus - Entry pslot[%p], poldslot[%p] \n " , pslot , poldslot ) ;
2005-04-16 15:20:36 -07:00
// bit 0 - HPC_SLOT_POWER
if ( ( pslot - > status & 0x01 ) ! = ( poldslot - > status & 0x01 ) )
2006-03-14 16:24:47 -08:00
update = 1 ;
2005-04-16 15:20:36 -07:00
// bit 1 - HPC_SLOT_CONNECT
// ignore
// bit 2 - HPC_SLOT_ATTN
if ( ( pslot - > status & 0x04 ) ! = ( poldslot - > status & 0x04 ) )
2006-03-14 16:24:47 -08:00
update = 1 ;
2005-04-16 15:20:36 -07:00
// bit 3 - HPC_SLOT_PRSNT2
// bit 4 - HPC_SLOT_PRSNT1
if ( ( ( pslot - > status & 0x08 ) ! = ( poldslot - > status & 0x08 ) )
| | ( ( pslot - > status & 0x10 ) ! = ( poldslot - > status & 0x10 ) ) )
2006-03-14 16:24:47 -08:00
update = 1 ;
2005-04-16 15:20:36 -07:00
// bit 5 - HPC_SLOT_PWRGD
if ( ( pslot - > status & 0x20 ) ! = ( poldslot - > status & 0x20 ) )
// OFF -> ON: ignore, ON -> OFF: disable slot
2015-12-27 13:21:11 -08:00
if ( ( poldslot - > status & 0x20 ) & & ( SLOT_CONNECT ( poldslot - > status ) = = HPC_SLOT_CONNECTED ) & & ( SLOT_PRESENT ( poldslot - > status ) ) )
2006-03-14 16:24:47 -08:00
disable = 1 ;
2005-04-16 15:20:36 -07:00
// bit 6 - HPC_SLOT_BUS_SPEED
// ignore
// bit 7 - HPC_SLOT_LATCH
if ( ( pslot - > status & 0x80 ) ! = ( poldslot - > status & 0x80 ) ) {
2006-03-14 16:24:47 -08:00
update = 1 ;
2005-04-16 15:20:36 -07:00
// OPEN -> CLOSE
if ( pslot - > status & 0x80 ) {
2015-12-27 13:21:11 -08:00
if ( SLOT_PWRGD ( pslot - > status ) ) {
2005-04-16 15:20:36 -07:00
// power goes on and off after closing latch
// check again to make sure power is still ON
msleep ( 1000 ) ;
2015-12-27 13:21:11 -08:00
rc = ibmphp_hpc_readslot ( pslot , READ_SLOTSTATUS , & status ) ;
if ( SLOT_PWRGD ( status ) )
2006-03-14 16:24:47 -08:00
update = 1 ;
2005-04-16 15:20:36 -07:00
else // overwrite power in pslot to OFF
pslot - > status & = ~ HPC_SLOT_POWER ;
}
}
2013-11-14 11:28:18 -07:00
// CLOSE -> OPEN
2015-12-27 13:21:11 -08:00
else if ( ( SLOT_PWRGD ( poldslot - > status ) = = HPC_SLOT_PWRGD_GOOD )
& & ( SLOT_CONNECT ( poldslot - > status ) = = HPC_SLOT_CONNECTED ) & & ( SLOT_PRESENT ( poldslot - > status ) ) ) {
2006-03-14 16:24:47 -08:00
disable = 1 ;
2005-04-16 15:20:36 -07:00
}
// else - ignore
}
// bit 4 - HPC_SLOT_BLINK_ATTN
if ( ( pslot - > ext_status & 0x08 ) ! = ( poldslot - > ext_status & 0x08 ) )
2006-03-14 16:24:47 -08:00
update = 1 ;
2005-04-16 15:20:36 -07:00
if ( disable ) {
2015-12-27 13:21:11 -08:00
debug ( " process_changeinstatus - disable slot \n " ) ;
2006-03-14 16:24:47 -08:00
pslot - > flag = 0 ;
2015-12-27 13:21:11 -08:00
rc = ibmphp_do_disable_slot ( pslot ) ;
2005-04-16 15:20:36 -07:00
}
2014-09-07 20:02:47 +02:00
if ( update | | disable )
2015-12-27 13:21:11 -08:00
ibmphp_update_slot_info ( pslot ) ;
2005-04-16 15:20:36 -07:00
2015-12-27 13:21:11 -08:00
debug ( " %s - Exit rc[%d] disable[%x] update[%x] \n " , __func__ , rc , disable , update ) ;
2005-04-16 15:20:36 -07:00
return rc ;
}
/*----------------------------------------------------------------------
* Name : process_changeinlatch
*
* Action : compare old and new latch reg status , process the change
*
* Input : old and current latch register status
*
* Return 0 or error codes
* Value :
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2015-12-27 13:21:11 -08:00
static int process_changeinlatch ( u8 old , u8 new , struct controller * ctrl )
2005-04-16 15:20:36 -07:00
{
struct slot myslot , * pslot ;
u8 i ;
u8 mask ;
int rc = 0 ;
2015-12-27 13:21:11 -08:00
debug ( " %s - Entry old[%x], new[%x] \n " , __func__ , old , new ) ;
2005-04-16 15:20:36 -07:00
// bit 0 reserved, 0 is LSB, check bit 1-6 for 6 slots
for ( i = ctrl - > starting_slot_num ; i < = ctrl - > ending_slot_num ; i + + ) {
mask = 0x01 < < i ;
if ( ( mask & old ) ! = ( mask & new ) ) {
2015-12-27 13:21:11 -08:00
pslot = ibmphp_get_slot_from_physical_num ( i ) ;
2005-04-16 15:20:36 -07:00
if ( pslot ) {
2015-12-27 13:21:11 -08:00
memcpy ( ( void * ) & myslot , ( void * ) pslot , sizeof ( struct slot ) ) ;
rc = ibmphp_hpc_readslot ( pslot , READ_ALLSTAT , NULL ) ;
debug ( " %s - call process_changeinstatus for slot[%d] \n " , __func__ , i ) ;
process_changeinstatus ( pslot , & myslot ) ;
2005-04-16 15:20:36 -07:00
} else {
rc = - EINVAL ;
2015-12-27 13:21:11 -08:00
err ( " %s - Error bad pointer for slot[%d] \n " , __func__ , i ) ;
2005-04-16 15:20:36 -07:00
}
}
}
2015-12-27 13:21:11 -08:00
debug ( " %s - Exit rc[%d] \n " , __func__ , rc ) ;
2005-04-16 15:20:36 -07:00
return rc ;
}
/*----------------------------------------------------------------------
* Name : ibmphp_hpc_start_poll_thread
*
* Action : start polling thread
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2015-12-27 13:21:11 -08:00
int __init ibmphp_hpc_start_poll_thread ( void )
2005-04-16 15:20:36 -07:00
{
2015-12-27 13:21:11 -08:00
debug ( " %s - Entry \n " , __func__ ) ;
2005-04-16 15:20:36 -07:00
2007-08-14 16:07:00 -07:00
ibmphp_poll_thread = kthread_run ( poll_hpc , NULL , " hpc_poll " ) ;
if ( IS_ERR ( ibmphp_poll_thread ) ) {
2015-12-27 13:21:11 -08:00
err ( " %s - Error, thread not started \n " , __func__ ) ;
2007-08-14 16:07:00 -07:00
return PTR_ERR ( ibmphp_poll_thread ) ;
2005-04-16 15:20:36 -07:00
}
2007-08-14 16:07:00 -07:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
/*----------------------------------------------------------------------
* Name : ibmphp_hpc_stop_poll_thread
*
* Action : stop polling thread and cleanup
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2015-12-27 13:21:11 -08:00
void __exit ibmphp_hpc_stop_poll_thread ( void )
2005-04-16 15:20:36 -07:00
{
2015-12-27 13:21:11 -08:00
debug ( " %s - Entry \n " , __func__ ) ;
2005-04-16 15:20:36 -07:00
2007-08-14 16:07:00 -07:00
kthread_stop ( ibmphp_poll_thread ) ;
2015-12-27 13:21:11 -08:00
debug ( " before locking operations \n " ) ;
ibmphp_lock_operations ( ) ;
debug ( " after locking operations \n " ) ;
2013-11-14 11:28:18 -07:00
2005-04-16 15:20:36 -07:00
// wait for poll thread to exit
2015-12-27 13:21:11 -08:00
debug ( " before sem_exit down \n " ) ;
down ( & sem_exit ) ;
debug ( " after sem_exit down \n " ) ;
2005-04-16 15:20:36 -07:00
// cleanup
2015-12-27 13:21:11 -08:00
debug ( " before free_hpc_access \n " ) ;
free_hpc_access ( ) ;
debug ( " after free_hpc_access \n " ) ;
ibmphp_unlock_operations ( ) ;
debug ( " after unlock operations \n " ) ;
up ( & sem_exit ) ;
debug ( " after sem exit up \n " ) ;
debug ( " %s - Exit \n " , __func__ ) ;
2005-04-16 15:20:36 -07:00
}
/*----------------------------------------------------------------------
* Name : hpc_wait_ctlr_notworking
*
* Action : wait until the controller is in a not working state
*
* Return 0 , HPC_ERROR
* Value :
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2015-12-27 13:21:11 -08:00
static int hpc_wait_ctlr_notworking ( int timeout , struct controller * ctlr_ptr , void __iomem * wpg_bbar ,
2014-04-18 20:13:49 -04:00
u8 * pstatus )
2005-04-16 15:20:36 -07:00
{
int rc = 0 ;
2006-03-14 16:24:47 -08:00
u8 done = 0 ;
2005-04-16 15:20:36 -07:00
2015-12-27 13:21:11 -08:00
debug_polling ( " hpc_wait_ctlr_notworking - Entry timeout[%d] \n " , timeout ) ;
2005-04-16 15:20:36 -07:00
while ( ! done ) {
2015-12-27 13:21:11 -08:00
* pstatus = ctrl_read ( ctlr_ptr , wpg_bbar , WPG_CTLR_INDEX ) ;
2005-04-16 15:20:36 -07:00
if ( * pstatus = = HPC_ERROR ) {
rc = HPC_ERROR ;
2006-03-14 16:24:47 -08:00
done = 1 ;
2005-04-16 15:20:36 -07:00
}
2015-12-27 13:21:11 -08:00
if ( CTLR_WORKING ( * pstatus ) = = HPC_CTLR_WORKING_NO )
2006-03-14 16:24:47 -08:00
done = 1 ;
2005-04-16 15:20:36 -07:00
if ( ! done ) {
msleep ( 1000 ) ;
if ( timeout < 1 ) {
2006-03-14 16:24:47 -08:00
done = 1 ;
2015-12-27 13:21:11 -08:00
err ( " HPCreadslot - Error ctlr timeout \n " ) ;
2005-04-16 15:20:36 -07:00
rc = HPC_ERROR ;
} else
timeout - - ;
}
}
2015-12-27 13:21:11 -08:00
debug_polling ( " hpc_wait_ctlr_notworking - Exit rc[%x] status[%x] \n " , rc , * pstatus ) ;
2005-04-16 15:20:36 -07:00
return rc ;
}