2007-04-27 08:08:21 +04:00
/* pci_sabre.c: Sabre specific PCI controller support.
2005-04-17 02:20:36 +04:00
*
2007-04-27 08:08:21 +04:00
* Copyright ( C ) 1997 , 1998 , 1999 , 2007 David S . Miller ( davem @ davemloft . net )
2005-04-17 02:20:36 +04:00
* Copyright ( C ) 1998 , 1999 Eddie C . Dost ( ecd @ skynet . be )
* Copyright ( C ) 1999 Jakub Jelinek ( jakub @ redhat . com )
*/
# include <linux/kernel.h>
# include <linux/types.h>
# include <linux/pci.h>
# include <linux/init.h>
# include <linux/slab.h>
# include <linux/interrupt.h>
2008-08-08 02:33:36 +04:00
# include <linux/of_device.h>
2005-04-17 02:20:36 +04:00
# include <asm/apb.h>
# include <asm/iommu.h>
# include <asm/irq.h>
2006-06-22 05:18:47 +04:00
# include <asm/prom.h>
2008-09-10 15:13:10 +04:00
# include <asm/upa.h>
2005-04-17 02:20:36 +04:00
# include "pci_impl.h"
# include "iommu_common.h"
2008-09-10 14:07:03 +04:00
# include "psycho_common.h"
2005-04-17 02:20:36 +04:00
2008-08-30 14:14:01 +04:00
# define DRIVER_NAME "sabre"
# define PFX DRIVER_NAME ": "
2005-04-17 02:20:36 +04:00
/* SABRE PCI controller register offsets and definitions. */
# define SABRE_UE_AFSR 0x0030UL
# define SABRE_UEAFSR_PDRD 0x4000000000000000UL /* Primary PCI DMA Read */
# define SABRE_UEAFSR_PDWR 0x2000000000000000UL /* Primary PCI DMA Write */
# define SABRE_UEAFSR_SDRD 0x0800000000000000UL /* Secondary PCI DMA Read */
# define SABRE_UEAFSR_SDWR 0x0400000000000000UL /* Secondary PCI DMA Write */
# define SABRE_UEAFSR_SDTE 0x0200000000000000UL /* Secondary DMA Translation Error */
# define SABRE_UEAFSR_PDTE 0x0100000000000000UL /* Primary DMA Translation Error */
# define SABRE_UEAFSR_BMSK 0x0000ffff00000000UL /* Bytemask */
# define SABRE_UEAFSR_OFF 0x00000000e0000000UL /* Offset (AFAR bits [5:3] */
# define SABRE_UEAFSR_BLK 0x0000000000800000UL /* Was block operation */
# define SABRE_UECE_AFAR 0x0038UL
# define SABRE_CE_AFSR 0x0040UL
# define SABRE_CEAFSR_PDRD 0x4000000000000000UL /* Primary PCI DMA Read */
# define SABRE_CEAFSR_PDWR 0x2000000000000000UL /* Primary PCI DMA Write */
# define SABRE_CEAFSR_SDRD 0x0800000000000000UL /* Secondary PCI DMA Read */
# define SABRE_CEAFSR_SDWR 0x0400000000000000UL /* Secondary PCI DMA Write */
# define SABRE_CEAFSR_ESYND 0x00ff000000000000UL /* ECC Syndrome */
# define SABRE_CEAFSR_BMSK 0x0000ffff00000000UL /* Bytemask */
# define SABRE_CEAFSR_OFF 0x00000000e0000000UL /* Offset */
# define SABRE_CEAFSR_BLK 0x0000000000800000UL /* Was block operation */
# define SABRE_UECE_AFAR_ALIAS 0x0048UL /* Aliases to 0x0038 */
# define SABRE_IOMMU_CONTROL 0x0200UL
# define SABRE_IOMMUCTRL_ERRSTS 0x0000000006000000UL /* Error status bits */
# define SABRE_IOMMUCTRL_ERR 0x0000000001000000UL /* Error present in IOTLB */
# define SABRE_IOMMUCTRL_LCKEN 0x0000000000800000UL /* IOTLB lock enable */
# define SABRE_IOMMUCTRL_LCKPTR 0x0000000000780000UL /* IOTLB lock pointer */
# define SABRE_IOMMUCTRL_TSBSZ 0x0000000000070000UL /* TSB Size */
# define SABRE_IOMMU_TSBSZ_1K 0x0000000000000000
# define SABRE_IOMMU_TSBSZ_2K 0x0000000000010000
# define SABRE_IOMMU_TSBSZ_4K 0x0000000000020000
# define SABRE_IOMMU_TSBSZ_8K 0x0000000000030000
# define SABRE_IOMMU_TSBSZ_16K 0x0000000000040000
# define SABRE_IOMMU_TSBSZ_32K 0x0000000000050000
# define SABRE_IOMMU_TSBSZ_64K 0x0000000000060000
# define SABRE_IOMMU_TSBSZ_128K 0x0000000000070000
# define SABRE_IOMMUCTRL_TBWSZ 0x0000000000000004UL /* TSB assumed page size */
# define SABRE_IOMMUCTRL_DENAB 0x0000000000000002UL /* Diagnostic Mode Enable */
# define SABRE_IOMMUCTRL_ENAB 0x0000000000000001UL /* IOMMU Enable */
# define SABRE_IOMMU_TSBBASE 0x0208UL
# define SABRE_IOMMU_FLUSH 0x0210UL
# define SABRE_IMAP_A_SLOT0 0x0c00UL
# define SABRE_IMAP_B_SLOT0 0x0c20UL
# define SABRE_IMAP_SCSI 0x1000UL
# define SABRE_IMAP_ETH 0x1008UL
# define SABRE_IMAP_BPP 0x1010UL
# define SABRE_IMAP_AU_REC 0x1018UL
# define SABRE_IMAP_AU_PLAY 0x1020UL
# define SABRE_IMAP_PFAIL 0x1028UL
# define SABRE_IMAP_KMS 0x1030UL
# define SABRE_IMAP_FLPY 0x1038UL
# define SABRE_IMAP_SHW 0x1040UL
# define SABRE_IMAP_KBD 0x1048UL
# define SABRE_IMAP_MS 0x1050UL
# define SABRE_IMAP_SER 0x1058UL
# define SABRE_IMAP_UE 0x1070UL
# define SABRE_IMAP_CE 0x1078UL
# define SABRE_IMAP_PCIERR 0x1080UL
# define SABRE_IMAP_GFX 0x1098UL
# define SABRE_IMAP_EUPA 0x10a0UL
# define SABRE_ICLR_A_SLOT0 0x1400UL
# define SABRE_ICLR_B_SLOT0 0x1480UL
# define SABRE_ICLR_SCSI 0x1800UL
# define SABRE_ICLR_ETH 0x1808UL
# define SABRE_ICLR_BPP 0x1810UL
# define SABRE_ICLR_AU_REC 0x1818UL
# define SABRE_ICLR_AU_PLAY 0x1820UL
# define SABRE_ICLR_PFAIL 0x1828UL
# define SABRE_ICLR_KMS 0x1830UL
# define SABRE_ICLR_FLPY 0x1838UL
# define SABRE_ICLR_SHW 0x1840UL
# define SABRE_ICLR_KBD 0x1848UL
# define SABRE_ICLR_MS 0x1850UL
# define SABRE_ICLR_SER 0x1858UL
# define SABRE_ICLR_UE 0x1870UL
# define SABRE_ICLR_CE 0x1878UL
# define SABRE_ICLR_PCIERR 0x1880UL
# define SABRE_WRSYNC 0x1c20UL
# define SABRE_PCICTRL 0x2000UL
# define SABRE_PCICTRL_MRLEN 0x0000001000000000UL /* Use MemoryReadLine for block loads/stores */
# define SABRE_PCICTRL_SERR 0x0000000400000000UL /* Set when SERR asserted on PCI bus */
# define SABRE_PCICTRL_ARBPARK 0x0000000000200000UL /* Bus Parking 0=Ultra-IIi 1=prev-bus-owner */
# define SABRE_PCICTRL_CPUPRIO 0x0000000000100000UL /* Ultra-IIi granted every other bus cycle */
# define SABRE_PCICTRL_ARBPRIO 0x00000000000f0000UL /* Slot which is granted every other bus cycle */
# define SABRE_PCICTRL_ERREN 0x0000000000000100UL /* PCI Error Interrupt Enable */
# define SABRE_PCICTRL_RTRYWE 0x0000000000000080UL /* DMA Flow Control 0=wait-if-possible 1=retry */
# define SABRE_PCICTRL_AEN 0x000000000000000fUL /* Slot PCI arbitration enables */
# define SABRE_PIOAFSR 0x2010UL
# define SABRE_PIOAFSR_PMA 0x8000000000000000UL /* Primary Master Abort */
# define SABRE_PIOAFSR_PTA 0x4000000000000000UL /* Primary Target Abort */
# define SABRE_PIOAFSR_PRTRY 0x2000000000000000UL /* Primary Excessive Retries */
# define SABRE_PIOAFSR_PPERR 0x1000000000000000UL /* Primary Parity Error */
# define SABRE_PIOAFSR_SMA 0x0800000000000000UL /* Secondary Master Abort */
# define SABRE_PIOAFSR_STA 0x0400000000000000UL /* Secondary Target Abort */
# define SABRE_PIOAFSR_SRTRY 0x0200000000000000UL /* Secondary Excessive Retries */
# define SABRE_PIOAFSR_SPERR 0x0100000000000000UL /* Secondary Parity Error */
# define SABRE_PIOAFSR_BMSK 0x0000ffff00000000UL /* Byte Mask */
# define SABRE_PIOAFSR_BLK 0x0000000080000000UL /* Was Block Operation */
# define SABRE_PIOAFAR 0x2018UL
# define SABRE_PCIDIAG 0x2020UL
# define SABRE_PCIDIAG_DRTRY 0x0000000000000040UL /* Disable PIO Retry Limit */
# define SABRE_PCIDIAG_IPAPAR 0x0000000000000008UL /* Invert PIO Address Parity */
# define SABRE_PCIDIAG_IPDPAR 0x0000000000000004UL /* Invert PIO Data Parity */
# define SABRE_PCIDIAG_IDDPAR 0x0000000000000002UL /* Invert DMA Data Parity */
# define SABRE_PCIDIAG_ELPBK 0x0000000000000001UL /* Loopback Enable - not supported */
# define SABRE_PCITASR 0x2028UL
# define SABRE_PCITASR_EF 0x0000000000000080UL /* Respond to 0xe0000000-0xffffffff */
# define SABRE_PCITASR_CD 0x0000000000000040UL /* Respond to 0xc0000000-0xdfffffff */
# define SABRE_PCITASR_AB 0x0000000000000020UL /* Respond to 0xa0000000-0xbfffffff */
# define SABRE_PCITASR_89 0x0000000000000010UL /* Respond to 0x80000000-0x9fffffff */
# define SABRE_PCITASR_67 0x0000000000000008UL /* Respond to 0x60000000-0x7fffffff */
# define SABRE_PCITASR_45 0x0000000000000004UL /* Respond to 0x40000000-0x5fffffff */
# define SABRE_PCITASR_23 0x0000000000000002UL /* Respond to 0x20000000-0x3fffffff */
# define SABRE_PCITASR_01 0x0000000000000001UL /* Respond to 0x00000000-0x1fffffff */
# define SABRE_PIOBUF_DIAG 0x5000UL
# define SABRE_DMABUF_DIAGLO 0x5100UL
# define SABRE_DMABUF_DIAGHI 0x51c0UL
# define SABRE_IMAP_GFX_ALIAS 0x6000UL /* Aliases to 0x1098 */
# define SABRE_IMAP_EUPA_ALIAS 0x8000UL /* Aliases to 0x10a0 */
# define SABRE_IOMMU_VADIAG 0xa400UL
# define SABRE_IOMMU_TCDIAG 0xa408UL
# define SABRE_IOMMU_TAG 0xa580UL
# define SABRE_IOMMUTAG_ERRSTS 0x0000000001800000UL /* Error status bits */
# define SABRE_IOMMUTAG_ERR 0x0000000000400000UL /* Error present */
# define SABRE_IOMMUTAG_WRITE 0x0000000000200000UL /* Page is writable */
# define SABRE_IOMMUTAG_STREAM 0x0000000000100000UL /* Streamable bit - unused */
# define SABRE_IOMMUTAG_SIZE 0x0000000000080000UL /* 0=8k 1=16k */
# define SABRE_IOMMUTAG_VPN 0x000000000007ffffUL /* Virtual Page Number [31:13] */
# define SABRE_IOMMU_DATA 0xa600UL
# define SABRE_IOMMUDATA_VALID 0x0000000040000000UL /* Valid */
# define SABRE_IOMMUDATA_USED 0x0000000020000000UL /* Used (for LRU algorithm) */
# define SABRE_IOMMUDATA_CACHE 0x0000000010000000UL /* Cacheable */
# define SABRE_IOMMUDATA_PPN 0x00000000001fffffUL /* Physical Page Number [33:13] */
# define SABRE_PCI_IRQSTATE 0xa800UL
# define SABRE_OBIO_IRQSTATE 0xa808UL
# define SABRE_FFBCFG 0xf000UL
# define SABRE_FFBCFG_SPRQS 0x000000000f000000 /* Slave P_RQST queue size */
# define SABRE_FFBCFG_ONEREAD 0x0000000000004000 /* Slave supports one outstanding read */
# define SABRE_MCCTRL0 0xf010UL
# define SABRE_MCCTRL0_RENAB 0x0000000080000000 /* Refresh Enable */
# define SABRE_MCCTRL0_EENAB 0x0000000010000000 /* Enable all ECC functions */
# define SABRE_MCCTRL0_11BIT 0x0000000000001000 /* Enable 11-bit column addressing */
# define SABRE_MCCTRL0_DPP 0x0000000000000f00 /* DIMM Pair Present Bits */
# define SABRE_MCCTRL0_RINTVL 0x00000000000000ff /* Refresh Interval */
# define SABRE_MCCTRL1 0xf018UL
# define SABRE_MCCTRL1_AMDC 0x0000000038000000 /* Advance Memdata Clock */
# define SABRE_MCCTRL1_ARDC 0x0000000007000000 /* Advance DRAM Read Data Clock */
# define SABRE_MCCTRL1_CSR 0x0000000000e00000 /* CAS to RAS delay for CBR refresh */
# define SABRE_MCCTRL1_CASRW 0x00000000001c0000 /* CAS length for read/write */
# define SABRE_MCCTRL1_RCD 0x0000000000038000 /* RAS to CAS delay */
# define SABRE_MCCTRL1_CP 0x0000000000007000 /* CAS Precharge */
# define SABRE_MCCTRL1_RP 0x0000000000000e00 /* RAS Precharge */
# define SABRE_MCCTRL1_RAS 0x00000000000001c0 /* Length of RAS for refresh */
# define SABRE_MCCTRL1_CASRW2 0x0000000000000038 /* Must be same as CASRW */
# define SABRE_MCCTRL1_RSC 0x0000000000000007 /* RAS after CAS hold time */
# define SABRE_RESETCTRL 0xf020UL
# define SABRE_CONFIGSPACE 0x001000000UL
# define SABRE_IOSPACE 0x002000000UL
# define SABRE_IOSPACE_SIZE 0x000ffffffUL
# define SABRE_MEMSPACE 0x100000000UL
# define SABRE_MEMSPACE_SIZE 0x07fffffffUL
static int hummingbird_p ;
static struct pci_bus * sabre_root_bus ;
2006-10-08 16:23:28 +04:00
static irqreturn_t sabre_ue_intr ( int irq , void * dev_id )
2005-04-17 02:20:36 +04:00
{
2007-05-08 10:49:01 +04:00
struct pci_pbm_info * pbm = dev_id ;
unsigned long afsr_reg = pbm - > controller_regs + SABRE_UE_AFSR ;
unsigned long afar_reg = pbm - > controller_regs + SABRE_UECE_AFAR ;
2005-04-17 02:20:36 +04:00
unsigned long afsr , afar , error_bits ;
int reported ;
/* Latch uncorrectable error status. */
2008-09-10 15:13:10 +04:00
afar = upa_readq ( afar_reg ) ;
afsr = upa_readq ( afsr_reg ) ;
2005-04-17 02:20:36 +04:00
/* Clear the primary/secondary error status bits. */
error_bits = afsr &
( SABRE_UEAFSR_PDRD | SABRE_UEAFSR_PDWR |
SABRE_UEAFSR_SDRD | SABRE_UEAFSR_SDWR |
SABRE_UEAFSR_SDTE | SABRE_UEAFSR_PDTE ) ;
if ( ! error_bits )
return IRQ_NONE ;
2008-09-10 15:13:10 +04:00
upa_writeq ( error_bits , afsr_reg ) ;
2005-04-17 02:20:36 +04:00
/* Log the error. */
2007-05-08 10:49:01 +04:00
printk ( " %s: Uncorrectable Error, primary error type[%s%s] \n " ,
pbm - > name ,
2005-04-17 02:20:36 +04:00
( ( error_bits & SABRE_UEAFSR_PDRD ) ?
" DMA Read " :
( ( error_bits & SABRE_UEAFSR_PDWR ) ?
" DMA Write " : " ??? " ) ) ,
( ( error_bits & SABRE_UEAFSR_PDTE ) ?
" :Translation Error " : " " ) ) ;
2007-05-08 10:49:01 +04:00
printk ( " %s: bytemask[%04lx] dword_offset[%lx] was_block(%d) \n " ,
pbm - > name ,
2005-04-17 02:20:36 +04:00
( afsr & SABRE_UEAFSR_BMSK ) > > 32UL ,
( afsr & SABRE_UEAFSR_OFF ) > > 29UL ,
( ( afsr & SABRE_UEAFSR_BLK ) ? 1 : 0 ) ) ;
2007-05-08 10:49:01 +04:00
printk ( " %s: UE AFAR [%016lx] \n " , pbm - > name , afar ) ;
printk ( " %s: UE Secondary errors [ " , pbm - > name ) ;
2005-04-17 02:20:36 +04:00
reported = 0 ;
if ( afsr & SABRE_UEAFSR_SDRD ) {
reported + + ;
printk ( " (DMA Read) " ) ;
}
if ( afsr & SABRE_UEAFSR_SDWR ) {
reported + + ;
printk ( " (DMA Write) " ) ;
}
if ( afsr & SABRE_UEAFSR_SDTE ) {
reported + + ;
printk ( " (Translation Error) " ) ;
}
if ( ! reported )
printk ( " (none) " ) ;
printk ( " ] \n " ) ;
/* Interrogate IOMMU for error status. */
2008-09-10 14:52:51 +04:00
psycho_check_iommu_error ( pbm , afsr , afar , UE_ERR ) ;
2005-04-17 02:20:36 +04:00
return IRQ_HANDLED ;
}
2006-10-08 16:23:28 +04:00
static irqreturn_t sabre_ce_intr ( int irq , void * dev_id )
2005-04-17 02:20:36 +04:00
{
2007-05-08 10:49:01 +04:00
struct pci_pbm_info * pbm = dev_id ;
unsigned long afsr_reg = pbm - > controller_regs + SABRE_CE_AFSR ;
unsigned long afar_reg = pbm - > controller_regs + SABRE_UECE_AFAR ;
2005-04-17 02:20:36 +04:00
unsigned long afsr , afar , error_bits ;
int reported ;
/* Latch error status. */
2008-09-10 15:13:10 +04:00
afar = upa_readq ( afar_reg ) ;
afsr = upa_readq ( afsr_reg ) ;
2005-04-17 02:20:36 +04:00
/* Clear primary/secondary error status bits. */
error_bits = afsr &
( SABRE_CEAFSR_PDRD | SABRE_CEAFSR_PDWR |
SABRE_CEAFSR_SDRD | SABRE_CEAFSR_SDWR ) ;
if ( ! error_bits )
return IRQ_NONE ;
2008-09-10 15:13:10 +04:00
upa_writeq ( error_bits , afsr_reg ) ;
2005-04-17 02:20:36 +04:00
/* Log the error. */
2007-05-08 10:49:01 +04:00
printk ( " %s: Correctable Error, primary error type[%s] \n " ,
pbm - > name ,
2005-04-17 02:20:36 +04:00
( ( error_bits & SABRE_CEAFSR_PDRD ) ?
" DMA Read " :
( ( error_bits & SABRE_CEAFSR_PDWR ) ?
" DMA Write " : " ??? " ) ) ) ;
/* XXX Use syndrome and afar to print out module string just like
* XXX UDB CE trap handler does . . . - DaveM
*/
2007-05-08 10:49:01 +04:00
printk ( " %s: syndrome[%02lx] bytemask[%04lx] dword_offset[%lx] "
2005-04-17 02:20:36 +04:00
" was_block(%d) \n " ,
2007-05-08 10:49:01 +04:00
pbm - > name ,
2005-04-17 02:20:36 +04:00
( afsr & SABRE_CEAFSR_ESYND ) > > 48UL ,
( afsr & SABRE_CEAFSR_BMSK ) > > 32UL ,
( afsr & SABRE_CEAFSR_OFF ) > > 29UL ,
( ( afsr & SABRE_CEAFSR_BLK ) ? 1 : 0 ) ) ;
2007-05-08 10:49:01 +04:00
printk ( " %s: CE AFAR [%016lx] \n " , pbm - > name , afar ) ;
printk ( " %s: CE Secondary errors [ " , pbm - > name ) ;
2005-04-17 02:20:36 +04:00
reported = 0 ;
if ( afsr & SABRE_CEAFSR_SDRD ) {
reported + + ;
printk ( " (DMA Read) " ) ;
}
if ( afsr & SABRE_CEAFSR_SDWR ) {
reported + + ;
printk ( " (DMA Write) " ) ;
}
if ( ! reported )
printk ( " (none) " ) ;
printk ( " ] \n " ) ;
return IRQ_HANDLED ;
}
2007-05-08 10:06:27 +04:00
static void sabre_register_error_handlers ( struct pci_pbm_info * pbm )
2005-04-17 02:20:36 +04:00
{
2010-04-14 03:12:29 +04:00
struct device_node * dp = pbm - > op - > dev . of_node ;
2010-07-23 02:04:30 +04:00
struct platform_device * op ;
2005-04-17 02:20:36 +04:00
unsigned long base = pbm - > controller_regs ;
u64 tmp ;
2007-05-09 04:23:31 +04:00
int err ;
2005-04-17 02:20:36 +04:00
2006-06-30 02:07:37 +04:00
if ( pbm - > chip_type = = PBM_CHIP_TYPE_SABRE )
dp = dp - > parent ;
op = of_find_device_by_node ( dp ) ;
if ( ! op )
return ;
/* Sabre/Hummingbird IRQ property layout is:
* 0 : PCI ERR
* 1 : UE ERR
* 2 : CE ERR
* 3 : POWER FAIL
*/
2010-06-18 21:09:58 +04:00
if ( op - > archdata . num_irqs < 4 )
2006-06-30 02:07:37 +04:00
return ;
2005-04-17 02:20:36 +04:00
/* We clear the error bits in the appropriate AFSR before
* registering the handler so that we don ' t get spurious
* interrupts .
*/
2008-09-10 15:13:10 +04:00
upa_writeq ( ( SABRE_UEAFSR_PDRD | SABRE_UEAFSR_PDWR |
SABRE_UEAFSR_SDRD | SABRE_UEAFSR_SDWR |
SABRE_UEAFSR_SDTE | SABRE_UEAFSR_PDTE ) ,
base + SABRE_UE_AFSR ) ;
2006-06-30 02:07:37 +04:00
2010-06-18 21:09:58 +04:00
err = request_irq ( op - > archdata . irqs [ 1 ] , sabre_ue_intr , 0 , " SABRE_UE " , pbm ) ;
2007-05-09 04:23:31 +04:00
if ( err )
printk ( KERN_WARNING " %s: Couldn't register UE, err=%d. \n " ,
pbm - > name , err ) ;
2005-04-17 02:20:36 +04:00
2008-09-10 15:13:10 +04:00
upa_writeq ( ( SABRE_CEAFSR_PDRD | SABRE_CEAFSR_PDWR |
SABRE_CEAFSR_SDRD | SABRE_CEAFSR_SDWR ) ,
base + SABRE_CE_AFSR ) ;
2005-04-17 02:20:36 +04:00
2010-06-18 21:09:58 +04:00
err = request_irq ( op - > archdata . irqs [ 2 ] , sabre_ce_intr , 0 , " SABRE_CE " , pbm ) ;
2007-05-09 04:23:31 +04:00
if ( err )
printk ( KERN_WARNING " %s: Couldn't register CE, err=%d. \n " ,
pbm - > name , err ) ;
2010-06-18 21:09:58 +04:00
err = request_irq ( op - > archdata . irqs [ 0 ] , psycho_pcierr_intr , 0 ,
2007-05-09 04:23:31 +04:00
" SABRE_PCIERR " , pbm ) ;
if ( err )
printk ( KERN_WARNING " %s: Couldn't register PCIERR, err=%d. \n " ,
pbm - > name , err ) ;
2005-04-17 02:20:36 +04:00
2008-09-10 15:13:10 +04:00
tmp = upa_readq ( base + SABRE_PCICTRL ) ;
2005-04-17 02:20:36 +04:00
tmp | = SABRE_PCICTRL_ERREN ;
2008-09-10 15:13:10 +04:00
upa_writeq ( tmp , base + SABRE_PCICTRL ) ;
2005-04-17 02:20:36 +04:00
}
2007-05-08 10:06:27 +04:00
static void apb_init ( struct pci_bus * sabre_bus )
2005-04-17 02:20:36 +04:00
{
struct pci_dev * pdev ;
list_for_each_entry ( pdev , & sabre_bus - > devices , bus_list ) {
if ( pdev - > vendor = = PCI_VENDOR_ID_SUN & &
pdev - > device = = PCI_DEVICE_ID_SUN_SIMBA ) {
u16 word16 ;
2007-03-04 23:53:19 +03:00
pci_read_config_word ( pdev , PCI_COMMAND , & word16 ) ;
2005-04-17 02:20:36 +04:00
word16 | = PCI_COMMAND_SERR | PCI_COMMAND_PARITY |
PCI_COMMAND_MASTER | PCI_COMMAND_MEMORY |
PCI_COMMAND_IO ;
2007-03-04 23:53:19 +03:00
pci_write_config_word ( pdev , PCI_COMMAND , word16 ) ;
2005-04-17 02:20:36 +04:00
/* Status register bits are "write 1 to clear". */
2007-03-04 23:53:19 +03:00
pci_write_config_word ( pdev , PCI_STATUS , 0xffff ) ;
pci_write_config_word ( pdev , PCI_SEC_STATUS , 0xffff ) ;
2005-04-17 02:20:36 +04:00
/* Use a primary/seconday latency timer value
* of 64.
*/
2007-03-04 23:53:19 +03:00
pci_write_config_byte ( pdev , PCI_LATENCY_TIMER , 64 ) ;
pci_write_config_byte ( pdev , PCI_SEC_LATENCY_TIMER , 64 ) ;
2005-04-17 02:20:36 +04:00
/* Enable reporting/forwarding of master aborts,
* parity , and SERR .
*/
2007-03-04 23:53:19 +03:00
pci_write_config_byte ( pdev , PCI_BRIDGE_CONTROL ,
( PCI_BRIDGE_CTL_PARITY |
PCI_BRIDGE_CTL_SERR |
PCI_BRIDGE_CTL_MASTER_ABORT ) ) ;
2005-04-17 02:20:36 +04:00
}
}
}
2009-04-07 12:03:58 +04:00
static void __devinit sabre_scan_bus ( struct pci_pbm_info * pbm ,
struct device * parent )
2005-04-17 02:20:36 +04:00
{
static int once ;
/* The APB bridge speaks to the Sabre host PCI bridge
* at 66 Mhz , but the front side of APB runs at 33 Mhz
* for both segments .
2007-06-07 01:03:08 +04:00
*
* Hummingbird systems do not use APB , so they run
* at 66 MHZ .
2005-04-17 02:20:36 +04:00
*/
2007-06-07 01:03:08 +04:00
if ( hummingbird_p )
pbm - > is_66mhz_capable = 1 ;
else
pbm - > is_66mhz_capable = 0 ;
2005-04-17 02:20:36 +04:00
/* This driver has not been verified to handle
* multiple SABREs yet , so trap this .
*
* Also note that the SABRE host bridge is hardwired
* to live at bus 0.
*/
if ( once ! = 0 ) {
2008-08-30 14:14:01 +04:00
printk ( KERN_ERR PFX " Multiple controllers unsupported. \n " ) ;
return ;
2005-04-17 02:20:36 +04:00
}
once + + ;
2008-09-02 05:32:22 +04:00
pbm - > pci_bus = pci_scan_one_pbm ( pbm , parent ) ;
2007-06-07 01:03:08 +04:00
if ( ! pbm - > pci_bus )
2007-03-01 10:35:04 +03:00
return ;
2005-04-17 02:20:36 +04:00
2007-06-07 01:03:08 +04:00
sabre_root_bus = pbm - > pci_bus ;
2005-04-17 02:20:36 +04:00
2007-06-07 01:03:08 +04:00
apb_init ( pbm - > pci_bus ) ;
2005-04-17 02:20:36 +04:00
2007-05-08 10:06:27 +04:00
sabre_register_error_handlers ( pbm ) ;
2005-04-17 02:20:36 +04:00
}
2009-04-07 12:03:58 +04:00
static void __devinit sabre_pbm_init ( struct pci_pbm_info * pbm ,
2010-07-23 02:04:30 +04:00
struct platform_device * op )
2005-04-17 02:20:36 +04:00
{
2008-09-10 14:15:56 +04:00
psycho_pbm_init_common ( pbm , op , " SABRE " , PBM_CHIP_TYPE_SABRE ) ;
2008-09-10 14:52:51 +04:00
pbm - > pci_afsr = pbm - > controller_regs + SABRE_PIOAFSR ;
pbm - > pci_afar = pbm - > controller_regs + SABRE_PIOAFAR ;
pbm - > pci_csr = pbm - > controller_regs + SABRE_PCICTRL ;
2008-09-02 05:32:22 +04:00
sabre_scan_bus ( pbm , & op - > dev ) ;
2005-04-17 02:20:36 +04:00
}
2011-02-23 06:01:33 +03:00
static int __devinit sabre_probe ( struct platform_device * op )
2005-04-17 02:20:36 +04:00
{
2007-03-29 12:50:16 +04:00
const struct linux_prom64_registers * pr_regs ;
2010-04-14 03:12:29 +04:00
struct device_node * dp = op - > dev . of_node ;
2007-05-08 11:19:02 +04:00
struct pci_pbm_info * pbm ;
2008-08-30 14:14:01 +04:00
u32 upa_portid , dma_mask ;
2007-04-27 08:08:21 +04:00
struct iommu * iommu ;
2008-08-30 14:14:01 +04:00
int tsbsize , err ;
2007-03-29 12:50:16 +04:00
const u32 * vdma ;
2005-04-17 02:20:36 +04:00
u64 clear_irq ;
2011-02-23 06:01:33 +03:00
hummingbird_p = op - > dev . of_match & & ( op - > dev . of_match - > data ! = NULL ) ;
2008-08-30 14:14:01 +04:00
if ( ! hummingbird_p ) {
struct device_node * cpu_dp ;
/* Of course, Sun has to encode things a thousand
* different ways , inconsistently .
*/
for_each_node_by_type ( cpu_dp , " cpu " ) {
if ( ! strcmp ( cpu_dp - > name , " SUNW,UltraSPARC-IIe " ) )
hummingbird_p = 1 ;
2005-04-17 02:20:36 +04:00
}
}
2008-08-30 14:14:01 +04:00
err = - ENOMEM ;
2008-09-10 10:54:02 +04:00
pbm = kzalloc ( sizeof ( * pbm ) , GFP_KERNEL ) ;
if ( ! pbm ) {
printk ( KERN_ERR PFX " Cannot allocate pci_pbm_info. \n " ) ;
2008-08-31 12:33:52 +04:00
goto out_err ;
2008-08-30 14:14:01 +04:00
}
2005-04-17 02:20:36 +04:00
2008-09-10 10:54:02 +04:00
iommu = kzalloc ( sizeof ( * iommu ) , GFP_KERNEL ) ;
2008-08-30 14:14:01 +04:00
if ( ! iommu ) {
printk ( KERN_ERR PFX " Cannot allocate PBM iommu. \n " ) ;
2008-08-31 12:33:52 +04:00
goto out_free_controller ;
2008-08-30 14:14:01 +04:00
}
2007-05-08 11:19:02 +04:00
pbm - > iommu = iommu ;
2005-04-17 02:20:36 +04:00
2007-03-04 23:53:19 +03:00
upa_portid = of_getintprop_default ( dp , " upa-portid " , 0xff ) ;
2005-04-17 02:20:36 +04:00
2007-05-08 11:19:02 +04:00
pbm - > portid = upa_portid ;
2005-04-17 02:20:36 +04:00
/*
* Map in SABRE register set and report the presence of this SABRE .
*/
2006-06-22 05:18:47 +04:00
2007-03-04 23:53:19 +03:00
pr_regs = of_get_property ( dp , " reg " , NULL ) ;
2008-08-30 14:14:01 +04:00
err = - ENODEV ;
if ( ! pr_regs ) {
printk ( KERN_ERR PFX " No reg property \n " ) ;
2008-08-31 12:33:52 +04:00
goto out_free_iommu ;
2008-08-30 14:14:01 +04:00
}
2005-04-17 02:20:36 +04:00
/*
* First REG in property is base of entire SABRE register space .
*/
2007-05-08 11:19:02 +04:00
pbm - > controller_regs = pr_regs [ 0 ] . phys_addr ;
2005-04-17 02:20:36 +04:00
/* Clear interrupts */
/* PCI first */
for ( clear_irq = SABRE_ICLR_A_SLOT0 ; clear_irq < SABRE_ICLR_B_SLOT0 + 0x80 ; clear_irq + = 8 )
2008-09-10 15:13:10 +04:00
upa_writeq ( 0x0UL , pbm - > controller_regs + clear_irq ) ;
2005-04-17 02:20:36 +04:00
/* Then OBIO */
for ( clear_irq = SABRE_ICLR_SCSI ; clear_irq < SABRE_ICLR_SCSI + 0x80 ; clear_irq + = 8 )
2008-09-10 15:13:10 +04:00
upa_writeq ( 0x0UL , pbm - > controller_regs + clear_irq ) ;
2005-04-17 02:20:36 +04:00
/* Error interrupts are enabled later after the bus scan. */
2008-09-10 15:13:10 +04:00
upa_writeq ( ( SABRE_PCICTRL_MRLEN | SABRE_PCICTRL_SERR |
SABRE_PCICTRL_ARBPARK | SABRE_PCICTRL_AEN ) ,
pbm - > controller_regs + SABRE_PCICTRL ) ;
2005-04-17 02:20:36 +04:00
/* Now map in PCI config space for entire SABRE. */
2008-09-10 10:54:02 +04:00
pbm - > config_space = pbm - > controller_regs + SABRE_CONFIGSPACE ;
2006-06-22 05:18:47 +04:00
2007-03-04 23:53:19 +03:00
vdma = of_get_property ( dp , " virtual-dma " , NULL ) ;
2008-08-30 14:14:01 +04:00
if ( ! vdma ) {
printk ( KERN_ERR PFX " No virtual-dma property \n " ) ;
2008-08-31 12:33:52 +04:00
goto out_free_iommu ;
2008-08-30 14:14:01 +04:00
}
2005-04-17 02:20:36 +04:00
dma_mask = vdma [ 0 ] ;
switch ( vdma [ 1 ] ) {
case 0x20000000 :
dma_mask | = 0x1fffffff ;
tsbsize = 64 ;
break ;
case 0x40000000 :
dma_mask | = 0x3fffffff ;
tsbsize = 128 ;
break ;
case 0x80000000 :
dma_mask | = 0x7fffffff ;
tsbsize = 128 ;
break ;
default :
2008-08-30 14:14:01 +04:00
printk ( KERN_ERR PFX " Strange virtual-dma size. \n " ) ;
2008-08-31 12:33:52 +04:00
goto out_free_iommu ;
2005-04-17 02:20:36 +04:00
}
2008-09-10 14:07:03 +04:00
err = psycho_iommu_init ( pbm , tsbsize , vdma [ 0 ] , dma_mask , SABRE_WRSYNC ) ;
2008-08-30 14:14:01 +04:00
if ( err )
2008-08-31 12:33:52 +04:00
goto out_free_iommu ;
2005-04-17 02:20:36 +04:00
/*
* Look for APB underneath .
*/
2008-09-10 10:54:02 +04:00
sabre_pbm_init ( pbm , op ) ;
pbm - > next = pci_pbm_root ;
pci_pbm_root = pbm ;
dev_set_drvdata ( & op - > dev , pbm ) ;
2008-08-30 14:14:01 +04:00
return 0 ;
2007-07-28 09:39:14 +04:00
2008-08-31 12:33:52 +04:00
out_free_iommu :
2008-09-10 10:54:02 +04:00
kfree ( pbm - > iommu ) ;
2008-08-31 12:33:52 +04:00
out_free_controller :
2008-09-10 10:54:02 +04:00
kfree ( pbm ) ;
2008-08-31 12:33:52 +04:00
out_err :
2008-08-30 14:14:01 +04:00
return err ;
2005-04-17 02:20:36 +04:00
}
2008-08-30 14:14:01 +04:00
2008-08-31 12:23:17 +04:00
static struct of_device_id __initdata sabre_match [ ] = {
2008-08-30 14:14:01 +04:00
{
. name = " pci " ,
. compatible = " pci108e,a001 " ,
. data = ( void * ) 1 ,
} ,
{
. name = " pci " ,
. compatible = " pci108e,a000 " ,
} ,
{ } ,
} ;
2011-02-23 06:01:33 +03:00
static struct platform_driver sabre_driver = {
2010-04-14 03:13:02 +04:00
. driver = {
. name = DRIVER_NAME ,
. owner = THIS_MODULE ,
. of_match_table = sabre_match ,
} ,
2008-08-30 14:14:01 +04:00
. probe = sabre_probe ,
} ;
static int __init sabre_init ( void )
{
2011-02-23 06:01:33 +03:00
return platform_driver_register ( & sabre_driver ) ;
2008-08-30 14:14:01 +04:00
}
subsys_initcall ( sabre_init ) ;