18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * LASI Device Driver 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * (c) Copyright 1999 Red Hat Software 68c2ecf20Sopenharmony_ci * Portions (c) Copyright 1999 The Puffin Group Inc. 78c2ecf20Sopenharmony_ci * Portions (c) Copyright 1999 Hewlett-Packard 88c2ecf20Sopenharmony_ci * 98c2ecf20Sopenharmony_ci * by Alan Cox <alan@redhat.com> and 108c2ecf20Sopenharmony_ci * Alex deVries <alex@onefishtwo.ca> 118c2ecf20Sopenharmony_ci */ 128c2ecf20Sopenharmony_ci 138c2ecf20Sopenharmony_ci#include <linux/errno.h> 148c2ecf20Sopenharmony_ci#include <linux/init.h> 158c2ecf20Sopenharmony_ci#include <linux/interrupt.h> 168c2ecf20Sopenharmony_ci#include <linux/slab.h> 178c2ecf20Sopenharmony_ci#include <linux/module.h> 188c2ecf20Sopenharmony_ci#include <linux/pm.h> 198c2ecf20Sopenharmony_ci#include <linux/types.h> 208c2ecf20Sopenharmony_ci 218c2ecf20Sopenharmony_ci#include <asm/io.h> 228c2ecf20Sopenharmony_ci#include <asm/hardware.h> 238c2ecf20Sopenharmony_ci#include <asm/led.h> 248c2ecf20Sopenharmony_ci 258c2ecf20Sopenharmony_ci#include "gsc.h" 268c2ecf20Sopenharmony_ci 278c2ecf20Sopenharmony_ci 288c2ecf20Sopenharmony_ci#define LASI_VER 0xC008 /* LASI Version */ 298c2ecf20Sopenharmony_ci 308c2ecf20Sopenharmony_ci#define LASI_IO_CONF 0x7FFFE /* LASI primary configuration register */ 318c2ecf20Sopenharmony_ci#define LASI_IO_CONF2 0x7FFFF /* LASI secondary configuration register */ 328c2ecf20Sopenharmony_ci 338c2ecf20Sopenharmony_cistatic void lasi_choose_irq(struct parisc_device *dev, void *ctrl) 348c2ecf20Sopenharmony_ci{ 358c2ecf20Sopenharmony_ci int irq; 368c2ecf20Sopenharmony_ci 378c2ecf20Sopenharmony_ci switch (dev->id.sversion) { 388c2ecf20Sopenharmony_ci case 0x74: irq = 7; break; /* Centronics */ 398c2ecf20Sopenharmony_ci case 0x7B: irq = 13; break; /* Audio */ 408c2ecf20Sopenharmony_ci case 0x81: irq = 14; break; /* Lasi itself */ 418c2ecf20Sopenharmony_ci case 0x82: irq = 9; break; /* SCSI */ 428c2ecf20Sopenharmony_ci case 0x83: irq = 20; break; /* Floppy */ 438c2ecf20Sopenharmony_ci case 0x84: irq = 26; break; /* PS/2 Keyboard */ 448c2ecf20Sopenharmony_ci case 0x87: irq = 18; break; /* ISDN */ 458c2ecf20Sopenharmony_ci case 0x8A: irq = 8; break; /* LAN */ 468c2ecf20Sopenharmony_ci case 0x8C: irq = 5; break; /* RS232 */ 478c2ecf20Sopenharmony_ci case 0x8D: irq = (dev->hw_path == 13) ? 16 : 17; break; 488c2ecf20Sopenharmony_ci /* Telephone */ 498c2ecf20Sopenharmony_ci default: return; /* unknown */ 508c2ecf20Sopenharmony_ci } 518c2ecf20Sopenharmony_ci 528c2ecf20Sopenharmony_ci gsc_asic_assign_irq(ctrl, irq, &dev->irq); 538c2ecf20Sopenharmony_ci} 548c2ecf20Sopenharmony_ci 558c2ecf20Sopenharmony_cistatic void __init 568c2ecf20Sopenharmony_cilasi_init_irq(struct gsc_asic *this_lasi) 578c2ecf20Sopenharmony_ci{ 588c2ecf20Sopenharmony_ci unsigned long lasi_base = this_lasi->hpa; 598c2ecf20Sopenharmony_ci 608c2ecf20Sopenharmony_ci /* Stop LASI barking for a bit */ 618c2ecf20Sopenharmony_ci gsc_writel(0x00000000, lasi_base+OFFSET_IMR); 628c2ecf20Sopenharmony_ci 638c2ecf20Sopenharmony_ci /* clear pending interrupts */ 648c2ecf20Sopenharmony_ci gsc_readl(lasi_base+OFFSET_IRR); 658c2ecf20Sopenharmony_ci 668c2ecf20Sopenharmony_ci /* We're not really convinced we want to reset the onboard 678c2ecf20Sopenharmony_ci * devices. Firmware does it for us... 688c2ecf20Sopenharmony_ci */ 698c2ecf20Sopenharmony_ci 708c2ecf20Sopenharmony_ci /* Resets */ 718c2ecf20Sopenharmony_ci /* gsc_writel(0xFFFFFFFF, lasi_base+0x2000);*/ /* Parallel */ 728c2ecf20Sopenharmony_ci if(pdc_add_valid(lasi_base+0x4004) == PDC_OK) 738c2ecf20Sopenharmony_ci gsc_writel(0xFFFFFFFF, lasi_base+0x4004); /* Audio */ 748c2ecf20Sopenharmony_ci /* gsc_writel(0xFFFFFFFF, lasi_base+0x5000);*/ /* Serial */ 758c2ecf20Sopenharmony_ci /* gsc_writel(0xFFFFFFFF, lasi_base+0x6000);*/ /* SCSI */ 768c2ecf20Sopenharmony_ci gsc_writel(0xFFFFFFFF, lasi_base+0x7000); /* LAN */ 778c2ecf20Sopenharmony_ci gsc_writel(0xFFFFFFFF, lasi_base+0x8000); /* Keyboard */ 788c2ecf20Sopenharmony_ci gsc_writel(0xFFFFFFFF, lasi_base+0xA000); /* FDC */ 798c2ecf20Sopenharmony_ci 808c2ecf20Sopenharmony_ci /* Ok we hit it on the head with a hammer, our Dog is now 818c2ecf20Sopenharmony_ci ** comatose and muzzled. Devices will now unmask LASI 828c2ecf20Sopenharmony_ci ** interrupts as they are registered as irq's in the LASI range. 838c2ecf20Sopenharmony_ci */ 848c2ecf20Sopenharmony_ci /* XXX: I thought it was `awks that got `it on the `ead with an 858c2ecf20Sopenharmony_ci * `ammer. -- willy 868c2ecf20Sopenharmony_ci */ 878c2ecf20Sopenharmony_ci} 888c2ecf20Sopenharmony_ci 898c2ecf20Sopenharmony_ci 908c2ecf20Sopenharmony_ci/* 918c2ecf20Sopenharmony_ci ** lasi_led_init() 928c2ecf20Sopenharmony_ci ** 938c2ecf20Sopenharmony_ci ** lasi_led_init() initializes the LED controller on the LASI. 948c2ecf20Sopenharmony_ci ** 958c2ecf20Sopenharmony_ci ** Since Mirage and Electra machines use a different LED 968c2ecf20Sopenharmony_ci ** address register, we need to check for these machines 978c2ecf20Sopenharmony_ci ** explicitly. 988c2ecf20Sopenharmony_ci */ 998c2ecf20Sopenharmony_ci 1008c2ecf20Sopenharmony_ci#ifndef CONFIG_CHASSIS_LCD_LED 1018c2ecf20Sopenharmony_ci 1028c2ecf20Sopenharmony_ci#define lasi_led_init(x) /* nothing */ 1038c2ecf20Sopenharmony_ci 1048c2ecf20Sopenharmony_ci#else 1058c2ecf20Sopenharmony_ci 1068c2ecf20Sopenharmony_cistatic void __init lasi_led_init(unsigned long lasi_hpa) 1078c2ecf20Sopenharmony_ci{ 1088c2ecf20Sopenharmony_ci unsigned long datareg; 1098c2ecf20Sopenharmony_ci 1108c2ecf20Sopenharmony_ci switch (CPU_HVERSION) { 1118c2ecf20Sopenharmony_ci /* Gecko machines have only one single LED, which can be permanently 1128c2ecf20Sopenharmony_ci turned on by writing a zero into the power control register. */ 1138c2ecf20Sopenharmony_ci case 0x600: /* Gecko (712/60) */ 1148c2ecf20Sopenharmony_ci case 0x601: /* Gecko (712/80) */ 1158c2ecf20Sopenharmony_ci case 0x602: /* Gecko (712/100) */ 1168c2ecf20Sopenharmony_ci case 0x603: /* Anole 64 (743/64) */ 1178c2ecf20Sopenharmony_ci case 0x604: /* Anole 100 (743/100) */ 1188c2ecf20Sopenharmony_ci case 0x605: /* Gecko (712/120) */ 1198c2ecf20Sopenharmony_ci datareg = lasi_hpa + 0x0000C000; 1208c2ecf20Sopenharmony_ci gsc_writeb(0, datareg); 1218c2ecf20Sopenharmony_ci return; /* no need to register the LED interrupt-function */ 1228c2ecf20Sopenharmony_ci 1238c2ecf20Sopenharmony_ci /* Mirage and Electra machines need special offsets */ 1248c2ecf20Sopenharmony_ci case 0x60A: /* Mirage Jr (715/64) */ 1258c2ecf20Sopenharmony_ci case 0x60B: /* Mirage 100 */ 1268c2ecf20Sopenharmony_ci case 0x60C: /* Mirage 100+ */ 1278c2ecf20Sopenharmony_ci case 0x60D: /* Electra 100 */ 1288c2ecf20Sopenharmony_ci case 0x60E: /* Electra 120 */ 1298c2ecf20Sopenharmony_ci datareg = lasi_hpa - 0x00020000; 1308c2ecf20Sopenharmony_ci break; 1318c2ecf20Sopenharmony_ci 1328c2ecf20Sopenharmony_ci default: 1338c2ecf20Sopenharmony_ci datareg = lasi_hpa + 0x0000C000; 1348c2ecf20Sopenharmony_ci break; 1358c2ecf20Sopenharmony_ci } 1368c2ecf20Sopenharmony_ci 1378c2ecf20Sopenharmony_ci register_led_driver(DISPLAY_MODEL_LASI, LED_CMD_REG_NONE, datareg); 1388c2ecf20Sopenharmony_ci} 1398c2ecf20Sopenharmony_ci#endif 1408c2ecf20Sopenharmony_ci 1418c2ecf20Sopenharmony_ci/* 1428c2ecf20Sopenharmony_ci * lasi_power_off 1438c2ecf20Sopenharmony_ci * 1448c2ecf20Sopenharmony_ci * Function for lasi to turn off the power. This is accomplished by setting a 1458c2ecf20Sopenharmony_ci * 1 to PWR_ON_L in the Power Control Register 1468c2ecf20Sopenharmony_ci * 1478c2ecf20Sopenharmony_ci */ 1488c2ecf20Sopenharmony_ci 1498c2ecf20Sopenharmony_cistatic unsigned long lasi_power_off_hpa __read_mostly; 1508c2ecf20Sopenharmony_ci 1518c2ecf20Sopenharmony_cistatic void lasi_power_off(void) 1528c2ecf20Sopenharmony_ci{ 1538c2ecf20Sopenharmony_ci unsigned long datareg; 1548c2ecf20Sopenharmony_ci 1558c2ecf20Sopenharmony_ci /* calculate addr of the Power Control Register */ 1568c2ecf20Sopenharmony_ci datareg = lasi_power_off_hpa + 0x0000C000; 1578c2ecf20Sopenharmony_ci 1588c2ecf20Sopenharmony_ci /* Power down the machine */ 1598c2ecf20Sopenharmony_ci gsc_writel(0x02, datareg); 1608c2ecf20Sopenharmony_ci} 1618c2ecf20Sopenharmony_ci 1628c2ecf20Sopenharmony_cistatic int __init lasi_init_chip(struct parisc_device *dev) 1638c2ecf20Sopenharmony_ci{ 1648c2ecf20Sopenharmony_ci extern void (*chassis_power_off)(void); 1658c2ecf20Sopenharmony_ci struct gsc_asic *lasi; 1668c2ecf20Sopenharmony_ci int ret; 1678c2ecf20Sopenharmony_ci 1688c2ecf20Sopenharmony_ci lasi = kzalloc(sizeof(*lasi), GFP_KERNEL); 1698c2ecf20Sopenharmony_ci if (!lasi) 1708c2ecf20Sopenharmony_ci return -ENOMEM; 1718c2ecf20Sopenharmony_ci 1728c2ecf20Sopenharmony_ci lasi->name = "Lasi"; 1738c2ecf20Sopenharmony_ci lasi->hpa = dev->hpa.start; 1748c2ecf20Sopenharmony_ci 1758c2ecf20Sopenharmony_ci /* Check the 4-bit (yes, only 4) version register */ 1768c2ecf20Sopenharmony_ci lasi->version = gsc_readl(lasi->hpa + LASI_VER) & 0xf; 1778c2ecf20Sopenharmony_ci printk(KERN_INFO "%s version %d at 0x%lx found.\n", 1788c2ecf20Sopenharmony_ci lasi->name, lasi->version, lasi->hpa); 1798c2ecf20Sopenharmony_ci 1808c2ecf20Sopenharmony_ci /* initialize the chassis LEDs really early */ 1818c2ecf20Sopenharmony_ci lasi_led_init(lasi->hpa); 1828c2ecf20Sopenharmony_ci 1838c2ecf20Sopenharmony_ci /* Stop LASI barking for a bit */ 1848c2ecf20Sopenharmony_ci lasi_init_irq(lasi); 1858c2ecf20Sopenharmony_ci 1868c2ecf20Sopenharmony_ci /* the IRQ lasi should use */ 1878c2ecf20Sopenharmony_ci dev->irq = gsc_alloc_irq(&lasi->gsc_irq); 1888c2ecf20Sopenharmony_ci if (dev->irq < 0) { 1898c2ecf20Sopenharmony_ci printk(KERN_ERR "%s(): cannot get GSC irq\n", 1908c2ecf20Sopenharmony_ci __func__); 1918c2ecf20Sopenharmony_ci kfree(lasi); 1928c2ecf20Sopenharmony_ci return -EBUSY; 1938c2ecf20Sopenharmony_ci } 1948c2ecf20Sopenharmony_ci 1958c2ecf20Sopenharmony_ci lasi->eim = ((u32) lasi->gsc_irq.txn_addr) | lasi->gsc_irq.txn_data; 1968c2ecf20Sopenharmony_ci 1978c2ecf20Sopenharmony_ci ret = request_irq(lasi->gsc_irq.irq, gsc_asic_intr, 0, "lasi", lasi); 1988c2ecf20Sopenharmony_ci if (ret < 0) { 1998c2ecf20Sopenharmony_ci kfree(lasi); 2008c2ecf20Sopenharmony_ci return ret; 2018c2ecf20Sopenharmony_ci } 2028c2ecf20Sopenharmony_ci 2038c2ecf20Sopenharmony_ci /* enable IRQ's for devices below LASI */ 2048c2ecf20Sopenharmony_ci gsc_writel(lasi->eim, lasi->hpa + OFFSET_IAR); 2058c2ecf20Sopenharmony_ci 2068c2ecf20Sopenharmony_ci /* Done init'ing, register this driver */ 2078c2ecf20Sopenharmony_ci ret = gsc_common_setup(dev, lasi); 2088c2ecf20Sopenharmony_ci if (ret) { 2098c2ecf20Sopenharmony_ci kfree(lasi); 2108c2ecf20Sopenharmony_ci return ret; 2118c2ecf20Sopenharmony_ci } 2128c2ecf20Sopenharmony_ci 2138c2ecf20Sopenharmony_ci gsc_fixup_irqs(dev, lasi, lasi_choose_irq); 2148c2ecf20Sopenharmony_ci 2158c2ecf20Sopenharmony_ci /* initialize the power off function */ 2168c2ecf20Sopenharmony_ci /* FIXME: Record the LASI HPA for the power off function. This should 2178c2ecf20Sopenharmony_ci * ensure that only the first LASI (the one controlling the power off) 2188c2ecf20Sopenharmony_ci * should set the HPA here */ 2198c2ecf20Sopenharmony_ci lasi_power_off_hpa = lasi->hpa; 2208c2ecf20Sopenharmony_ci chassis_power_off = lasi_power_off; 2218c2ecf20Sopenharmony_ci 2228c2ecf20Sopenharmony_ci return ret; 2238c2ecf20Sopenharmony_ci} 2248c2ecf20Sopenharmony_ci 2258c2ecf20Sopenharmony_cistatic struct parisc_device_id lasi_tbl[] __initdata = { 2268c2ecf20Sopenharmony_ci { HPHW_BA, HVERSION_REV_ANY_ID, HVERSION_ANY_ID, 0x00081 }, 2278c2ecf20Sopenharmony_ci { 0, } 2288c2ecf20Sopenharmony_ci}; 2298c2ecf20Sopenharmony_ci 2308c2ecf20Sopenharmony_cistruct parisc_driver lasi_driver __refdata = { 2318c2ecf20Sopenharmony_ci .name = "lasi", 2328c2ecf20Sopenharmony_ci .id_table = lasi_tbl, 2338c2ecf20Sopenharmony_ci .probe = lasi_init_chip, 2348c2ecf20Sopenharmony_ci}; 235