162306a36Sopenharmony_ci/* 262306a36Sopenharmony_ci * Basic EISA bus support for the SGI Indigo-2. 362306a36Sopenharmony_ci * 462306a36Sopenharmony_ci * (C) 2002 Pascal Dameme <netinet@freesurf.fr> 562306a36Sopenharmony_ci * and Marc Zyngier <mzyngier@freesurf.fr> 662306a36Sopenharmony_ci * 762306a36Sopenharmony_ci * This code is released under both the GPL version 2 and BSD 862306a36Sopenharmony_ci * licenses. Either license may be used. 962306a36Sopenharmony_ci * 1062306a36Sopenharmony_ci * This code offers a very basic support for this EISA bus present in 1162306a36Sopenharmony_ci * the SGI Indigo-2. It currently only supports PIO (forget about DMA 1262306a36Sopenharmony_ci * for the time being). This is enough for a low-end ethernet card, 1362306a36Sopenharmony_ci * but forget about your favorite SCSI card... 1462306a36Sopenharmony_ci * 1562306a36Sopenharmony_ci * TODO : 1662306a36Sopenharmony_ci * - Fix bugs... 1762306a36Sopenharmony_ci * - Add ISA support 1862306a36Sopenharmony_ci * - Add DMA (yeah, right...). 1962306a36Sopenharmony_ci * - Fix more bugs. 2062306a36Sopenharmony_ci */ 2162306a36Sopenharmony_ci 2262306a36Sopenharmony_ci#include <linux/eisa.h> 2362306a36Sopenharmony_ci#include <linux/types.h> 2462306a36Sopenharmony_ci#include <linux/init.h> 2562306a36Sopenharmony_ci#include <linux/irq.h> 2662306a36Sopenharmony_ci#include <linux/kernel_stat.h> 2762306a36Sopenharmony_ci#include <linux/signal.h> 2862306a36Sopenharmony_ci#include <linux/sched.h> 2962306a36Sopenharmony_ci#include <linux/interrupt.h> 3062306a36Sopenharmony_ci#include <linux/delay.h> 3162306a36Sopenharmony_ci#include <asm/io.h> 3262306a36Sopenharmony_ci#include <asm/irq.h> 3362306a36Sopenharmony_ci#include <asm/mipsregs.h> 3462306a36Sopenharmony_ci#include <asm/addrspace.h> 3562306a36Sopenharmony_ci#include <asm/processor.h> 3662306a36Sopenharmony_ci#include <asm/sgi/ioc.h> 3762306a36Sopenharmony_ci#include <asm/sgi/mc.h> 3862306a36Sopenharmony_ci#include <asm/sgi/ip22.h> 3962306a36Sopenharmony_ci#include <asm/i8259.h> 4062306a36Sopenharmony_ci 4162306a36Sopenharmony_ci/* I2 has four EISA slots. */ 4262306a36Sopenharmony_ci#define IP22_EISA_MAX_SLOTS 4 4362306a36Sopenharmony_ci#define EISA_MAX_IRQ 16 4462306a36Sopenharmony_ci 4562306a36Sopenharmony_ci#define EIU_MODE_REG 0x0001ffc0 4662306a36Sopenharmony_ci#define EIU_STAT_REG 0x0001ffc4 4762306a36Sopenharmony_ci#define EIU_PREMPT_REG 0x0001ffc8 4862306a36Sopenharmony_ci#define EIU_QUIET_REG 0x0001ffcc 4962306a36Sopenharmony_ci#define EIU_INTRPT_ACK 0x00010004 5062306a36Sopenharmony_ci 5162306a36Sopenharmony_cistatic char __init *decode_eisa_sig(unsigned long addr) 5262306a36Sopenharmony_ci{ 5362306a36Sopenharmony_ci static char sig_str[EISA_SIG_LEN] __initdata; 5462306a36Sopenharmony_ci u8 sig[4]; 5562306a36Sopenharmony_ci u16 rev; 5662306a36Sopenharmony_ci int i; 5762306a36Sopenharmony_ci 5862306a36Sopenharmony_ci for (i = 0; i < 4; i++) { 5962306a36Sopenharmony_ci sig[i] = inb(addr + i); 6062306a36Sopenharmony_ci 6162306a36Sopenharmony_ci if (!i && (sig[0] & 0x80)) 6262306a36Sopenharmony_ci return NULL; 6362306a36Sopenharmony_ci } 6462306a36Sopenharmony_ci 6562306a36Sopenharmony_ci sig_str[0] = ((sig[0] >> 2) & 0x1f) + ('A' - 1); 6662306a36Sopenharmony_ci sig_str[1] = (((sig[0] & 3) << 3) | (sig[1] >> 5)) + ('A' - 1); 6762306a36Sopenharmony_ci sig_str[2] = (sig[1] & 0x1f) + ('A' - 1); 6862306a36Sopenharmony_ci rev = (sig[2] << 8) | sig[3]; 6962306a36Sopenharmony_ci sprintf(sig_str + 3, "%04X", rev); 7062306a36Sopenharmony_ci 7162306a36Sopenharmony_ci return sig_str; 7262306a36Sopenharmony_ci} 7362306a36Sopenharmony_ci 7462306a36Sopenharmony_cistatic irqreturn_t ip22_eisa_intr(int irq, void *dev_id) 7562306a36Sopenharmony_ci{ 7662306a36Sopenharmony_ci u8 eisa_irq = inb(EIU_INTRPT_ACK); 7762306a36Sopenharmony_ci 7862306a36Sopenharmony_ci inb(EISA_DMA1_STATUS); 7962306a36Sopenharmony_ci inb(EISA_DMA2_STATUS); 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_ci if (eisa_irq < EISA_MAX_IRQ) { 8262306a36Sopenharmony_ci do_IRQ(eisa_irq); 8362306a36Sopenharmony_ci return IRQ_HANDLED; 8462306a36Sopenharmony_ci } 8562306a36Sopenharmony_ci 8662306a36Sopenharmony_ci /* Oops, Bad Stuff Happened... */ 8762306a36Sopenharmony_ci printk(KERN_ERR "eisa_irq %d out of bound\n", eisa_irq); 8862306a36Sopenharmony_ci 8962306a36Sopenharmony_ci outb(0x20, EISA_INT2_CTRL); 9062306a36Sopenharmony_ci outb(0x20, EISA_INT1_CTRL); 9162306a36Sopenharmony_ci 9262306a36Sopenharmony_ci return IRQ_NONE; 9362306a36Sopenharmony_ci} 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_ciint __init ip22_eisa_init(void) 9662306a36Sopenharmony_ci{ 9762306a36Sopenharmony_ci int i, c; 9862306a36Sopenharmony_ci char *str; 9962306a36Sopenharmony_ci 10062306a36Sopenharmony_ci if (!(sgimc->systemid & SGIMC_SYSID_EPRESENT)) { 10162306a36Sopenharmony_ci printk(KERN_INFO "EISA: bus not present.\n"); 10262306a36Sopenharmony_ci return 1; 10362306a36Sopenharmony_ci } 10462306a36Sopenharmony_ci 10562306a36Sopenharmony_ci printk(KERN_INFO "EISA: Probing bus...\n"); 10662306a36Sopenharmony_ci for (c = 0, i = 1; i <= IP22_EISA_MAX_SLOTS; i++) { 10762306a36Sopenharmony_ci if ((str = decode_eisa_sig(0x1000 * i + EISA_VENDOR_ID_OFFSET))) { 10862306a36Sopenharmony_ci printk(KERN_INFO "EISA: slot %d : %s detected.\n", 10962306a36Sopenharmony_ci i, str); 11062306a36Sopenharmony_ci c++; 11162306a36Sopenharmony_ci } 11262306a36Sopenharmony_ci } 11362306a36Sopenharmony_ci printk(KERN_INFO "EISA: Detected %d card%s.\n", c, c < 2 ? "" : "s"); 11462306a36Sopenharmony_ci#ifdef CONFIG_ISA 11562306a36Sopenharmony_ci printk(KERN_INFO "ISA support compiled in.\n"); 11662306a36Sopenharmony_ci#endif 11762306a36Sopenharmony_ci 11862306a36Sopenharmony_ci /* Warning : BlackMagicAhead(tm). 11962306a36Sopenharmony_ci Please wave your favorite dead chicken over the busses */ 12062306a36Sopenharmony_ci 12162306a36Sopenharmony_ci /* First say hello to the EIU */ 12262306a36Sopenharmony_ci outl(0x0000FFFF, EIU_PREMPT_REG); 12362306a36Sopenharmony_ci outl(1, EIU_QUIET_REG); 12462306a36Sopenharmony_ci outl(0x40f3c07F, EIU_MODE_REG); 12562306a36Sopenharmony_ci 12662306a36Sopenharmony_ci /* Now be nice to the EISA chipset */ 12762306a36Sopenharmony_ci outb(1, EISA_EXT_NMI_RESET_CTRL); 12862306a36Sopenharmony_ci udelay(50); /* Wait long enough for the dust to settle */ 12962306a36Sopenharmony_ci outb(0, EISA_EXT_NMI_RESET_CTRL); 13062306a36Sopenharmony_ci outb(0, EISA_DMA2_WRITE_SINGLE); 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_ci init_i8259_irqs(); 13362306a36Sopenharmony_ci 13462306a36Sopenharmony_ci if (request_irq(SGI_EISA_IRQ, ip22_eisa_intr, 0, "EISA", NULL)) 13562306a36Sopenharmony_ci pr_err("Failed to request irq %d (EISA)\n", SGI_EISA_IRQ); 13662306a36Sopenharmony_ci 13762306a36Sopenharmony_ci EISA_bus = 1; 13862306a36Sopenharmony_ci return 0; 13962306a36Sopenharmony_ci} 140