17c4166a9SMark Cave-Ayland /* 27c4166a9SMark Cave-Ayland * PowerMac NewWorld MacIO GPIO emulation 37c4166a9SMark Cave-Ayland * 47c4166a9SMark Cave-Ayland * Copyright (c) 2016 Benjamin Herrenschmidt 57c4166a9SMark Cave-Ayland * Copyright (c) 2018 Mark Cave-Ayland 67c4166a9SMark Cave-Ayland * 77c4166a9SMark Cave-Ayland * Permission is hereby granted, free of charge, to any person obtaining a copy 87c4166a9SMark Cave-Ayland * of this software and associated documentation files (the "Software"), to deal 97c4166a9SMark Cave-Ayland * in the Software without restriction, including without limitation the rights 107c4166a9SMark Cave-Ayland * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 117c4166a9SMark Cave-Ayland * copies of the Software, and to permit persons to whom the Software is 127c4166a9SMark Cave-Ayland * furnished to do so, subject to the following conditions: 137c4166a9SMark Cave-Ayland * 147c4166a9SMark Cave-Ayland * The above copyright notice and this permission notice shall be included in 157c4166a9SMark Cave-Ayland * all copies or substantial portions of the Software. 167c4166a9SMark Cave-Ayland * 177c4166a9SMark Cave-Ayland * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 187c4166a9SMark Cave-Ayland * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 197c4166a9SMark Cave-Ayland * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 207c4166a9SMark Cave-Ayland * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 217c4166a9SMark Cave-Ayland * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 227c4166a9SMark Cave-Ayland * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN 237c4166a9SMark Cave-Ayland * THE SOFTWARE. 247c4166a9SMark Cave-Ayland */ 257c4166a9SMark Cave-Ayland 267c4166a9SMark Cave-Ayland #include "qemu/osdep.h" 277c4166a9SMark Cave-Ayland #include "hw/ppc/mac.h" 28a27bd6c7SMarkus Armbruster #include "hw/qdev-properties.h" 29d6454270SMarkus Armbruster #include "migration/vmstate.h" 307c4166a9SMark Cave-Ayland #include "hw/misc/macio/macio.h" 317c4166a9SMark Cave-Ayland #include "hw/misc/macio/gpio.h" 328f55ac13SMark Cave-Ayland #include "hw/nmi.h" 337c4166a9SMark Cave-Ayland #include "qemu/log.h" 340b8fa32fSMarkus Armbruster #include "qemu/module.h" 357c4166a9SMark Cave-Ayland #include "trace.h" 367c4166a9SMark Cave-Ayland 377c4166a9SMark Cave-Ayland 387c4166a9SMark Cave-Ayland void macio_set_gpio(MacIOGPIOState *s, uint32_t gpio, bool state) 397c4166a9SMark Cave-Ayland { 407c4166a9SMark Cave-Ayland uint8_t new_reg; 417c4166a9SMark Cave-Ayland 427c4166a9SMark Cave-Ayland trace_macio_set_gpio(gpio, state); 437c4166a9SMark Cave-Ayland 447c4166a9SMark Cave-Ayland if (s->gpio_regs[gpio] & 4) { 457c4166a9SMark Cave-Ayland qemu_log_mask(LOG_GUEST_ERROR, 467c4166a9SMark Cave-Ayland "GPIO: Setting GPIO %d while it's an output\n", gpio); 477c4166a9SMark Cave-Ayland } 487c4166a9SMark Cave-Ayland 497c4166a9SMark Cave-Ayland new_reg = s->gpio_regs[gpio] & ~2; 507c4166a9SMark Cave-Ayland if (state) { 517c4166a9SMark Cave-Ayland new_reg |= 2; 527c4166a9SMark Cave-Ayland } 537c4166a9SMark Cave-Ayland 547c4166a9SMark Cave-Ayland if (new_reg == s->gpio_regs[gpio]) { 557c4166a9SMark Cave-Ayland return; 567c4166a9SMark Cave-Ayland } 577c4166a9SMark Cave-Ayland 587c4166a9SMark Cave-Ayland s->gpio_regs[gpio] = new_reg; 597c4166a9SMark Cave-Ayland 60*b73eb727SMark Cave-Ayland /* 617c4166a9SMark Cave-Ayland * Note that we probably need to get access to the MPIC config to 627c4166a9SMark Cave-Ayland * decode polarity since qemu always use "raise" regardless. 637c4166a9SMark Cave-Ayland * 647c4166a9SMark Cave-Ayland * For now, we hard wire known GPIOs 657c4166a9SMark Cave-Ayland */ 667c4166a9SMark Cave-Ayland 677c4166a9SMark Cave-Ayland switch (gpio) { 687c4166a9SMark Cave-Ayland case 1: 697c4166a9SMark Cave-Ayland /* Level low */ 707c4166a9SMark Cave-Ayland if (!state) { 717c4166a9SMark Cave-Ayland trace_macio_gpio_irq_assert(gpio); 727c4166a9SMark Cave-Ayland qemu_irq_raise(s->gpio_extirqs[gpio]); 737c4166a9SMark Cave-Ayland } else { 747c4166a9SMark Cave-Ayland trace_macio_gpio_irq_deassert(gpio); 757c4166a9SMark Cave-Ayland qemu_irq_lower(s->gpio_extirqs[gpio]); 767c4166a9SMark Cave-Ayland } 777c4166a9SMark Cave-Ayland break; 787c4166a9SMark Cave-Ayland 797c4166a9SMark Cave-Ayland case 9: 807c4166a9SMark Cave-Ayland /* Edge, triggered by NMI below */ 817c4166a9SMark Cave-Ayland if (state) { 827c4166a9SMark Cave-Ayland trace_macio_gpio_irq_assert(gpio); 837c4166a9SMark Cave-Ayland qemu_irq_raise(s->gpio_extirqs[gpio]); 847c4166a9SMark Cave-Ayland } else { 857c4166a9SMark Cave-Ayland trace_macio_gpio_irq_deassert(gpio); 867c4166a9SMark Cave-Ayland qemu_irq_lower(s->gpio_extirqs[gpio]); 877c4166a9SMark Cave-Ayland } 887c4166a9SMark Cave-Ayland break; 897c4166a9SMark Cave-Ayland 907c4166a9SMark Cave-Ayland default: 917c4166a9SMark Cave-Ayland qemu_log_mask(LOG_UNIMP, "GPIO: setting unimplemented GPIO %d", gpio); 927c4166a9SMark Cave-Ayland } 937c4166a9SMark Cave-Ayland } 947c4166a9SMark Cave-Ayland 957c4166a9SMark Cave-Ayland static void macio_gpio_write(void *opaque, hwaddr addr, uint64_t value, 967c4166a9SMark Cave-Ayland unsigned size) 977c4166a9SMark Cave-Ayland { 987c4166a9SMark Cave-Ayland MacIOGPIOState *s = opaque; 997c4166a9SMark Cave-Ayland uint8_t ibit; 1007c4166a9SMark Cave-Ayland 1017c4166a9SMark Cave-Ayland trace_macio_gpio_write(addr, value); 1027c4166a9SMark Cave-Ayland 1037c4166a9SMark Cave-Ayland /* Levels regs are read-only */ 1047c4166a9SMark Cave-Ayland if (addr < 8) { 1057c4166a9SMark Cave-Ayland return; 1067c4166a9SMark Cave-Ayland } 1077c4166a9SMark Cave-Ayland 1087c4166a9SMark Cave-Ayland addr -= 8; 1097c4166a9SMark Cave-Ayland if (addr < 36) { 1107c4166a9SMark Cave-Ayland value &= ~2; 1117c4166a9SMark Cave-Ayland 1127c4166a9SMark Cave-Ayland if (value & 4) { 1137c4166a9SMark Cave-Ayland ibit = (value & 1) << 1; 1147c4166a9SMark Cave-Ayland } else { 1157c4166a9SMark Cave-Ayland ibit = s->gpio_regs[addr] & 2; 1167c4166a9SMark Cave-Ayland } 1177c4166a9SMark Cave-Ayland 1187c4166a9SMark Cave-Ayland s->gpio_regs[addr] = value | ibit; 1197c4166a9SMark Cave-Ayland } 1207c4166a9SMark Cave-Ayland } 1217c4166a9SMark Cave-Ayland 1227c4166a9SMark Cave-Ayland static uint64_t macio_gpio_read(void *opaque, hwaddr addr, unsigned size) 1237c4166a9SMark Cave-Ayland { 1247c4166a9SMark Cave-Ayland MacIOGPIOState *s = opaque; 1257c4166a9SMark Cave-Ayland uint64_t val = 0; 1267c4166a9SMark Cave-Ayland 1277c4166a9SMark Cave-Ayland /* Levels regs */ 1287c4166a9SMark Cave-Ayland if (addr < 8) { 1297c4166a9SMark Cave-Ayland val = s->gpio_levels[addr]; 1307c4166a9SMark Cave-Ayland } else { 1317c4166a9SMark Cave-Ayland addr -= 8; 1327c4166a9SMark Cave-Ayland 1337c4166a9SMark Cave-Ayland if (addr < 36) { 1347c4166a9SMark Cave-Ayland val = s->gpio_regs[addr]; 1357c4166a9SMark Cave-Ayland } 1367c4166a9SMark Cave-Ayland } 1377c4166a9SMark Cave-Ayland 1387c4166a9SMark Cave-Ayland trace_macio_gpio_write(addr, val); 1397c4166a9SMark Cave-Ayland return val; 1407c4166a9SMark Cave-Ayland } 1417c4166a9SMark Cave-Ayland 1427c4166a9SMark Cave-Ayland static const MemoryRegionOps macio_gpio_ops = { 1437c4166a9SMark Cave-Ayland .read = macio_gpio_read, 1447c4166a9SMark Cave-Ayland .write = macio_gpio_write, 1457c4166a9SMark Cave-Ayland .endianness = DEVICE_LITTLE_ENDIAN, 1467c4166a9SMark Cave-Ayland .impl = { 1477c4166a9SMark Cave-Ayland .min_access_size = 1, 1487c4166a9SMark Cave-Ayland .max_access_size = 1, 1497c4166a9SMark Cave-Ayland }, 1507c4166a9SMark Cave-Ayland }; 1517c4166a9SMark Cave-Ayland 1527c4166a9SMark Cave-Ayland static void macio_gpio_init(Object *obj) 1537c4166a9SMark Cave-Ayland { 1547c4166a9SMark Cave-Ayland SysBusDevice *sbd = SYS_BUS_DEVICE(obj); 1557c4166a9SMark Cave-Ayland MacIOGPIOState *s = MACIO_GPIO(obj); 156*b73eb727SMark Cave-Ayland int i; 1577c4166a9SMark Cave-Ayland 158*b73eb727SMark Cave-Ayland for (i = 0; i < 10; i++) { 159*b73eb727SMark Cave-Ayland sysbus_init_irq(sbd, &s->gpio_extirqs[i]); 160*b73eb727SMark Cave-Ayland } 1617c4166a9SMark Cave-Ayland 1627c4166a9SMark Cave-Ayland memory_region_init_io(&s->gpiomem, OBJECT(s), &macio_gpio_ops, obj, 1637c4166a9SMark Cave-Ayland "gpio", 0x30); 1647c4166a9SMark Cave-Ayland sysbus_init_mmio(sbd, &s->gpiomem); 1657c4166a9SMark Cave-Ayland } 1667c4166a9SMark Cave-Ayland 1677c4166a9SMark Cave-Ayland static const VMStateDescription vmstate_macio_gpio = { 1687c4166a9SMark Cave-Ayland .name = "macio_gpio", 1697c4166a9SMark Cave-Ayland .version_id = 0, 1707c4166a9SMark Cave-Ayland .minimum_version_id = 0, 1717c4166a9SMark Cave-Ayland .fields = (VMStateField[]) { 1727c4166a9SMark Cave-Ayland VMSTATE_UINT8_ARRAY(gpio_levels, MacIOGPIOState, 8), 1737c4166a9SMark Cave-Ayland VMSTATE_UINT8_ARRAY(gpio_regs, MacIOGPIOState, 36), 1747c4166a9SMark Cave-Ayland VMSTATE_END_OF_LIST() 1757c4166a9SMark Cave-Ayland } 1767c4166a9SMark Cave-Ayland }; 1777c4166a9SMark Cave-Ayland 1787c4166a9SMark Cave-Ayland static void macio_gpio_reset(DeviceState *dev) 1797c4166a9SMark Cave-Ayland { 1807c4166a9SMark Cave-Ayland MacIOGPIOState *s = MACIO_GPIO(dev); 1817c4166a9SMark Cave-Ayland 1827c4166a9SMark Cave-Ayland /* GPIO 1 is up by default */ 1837c4166a9SMark Cave-Ayland macio_set_gpio(s, 1, true); 1847c4166a9SMark Cave-Ayland } 1857c4166a9SMark Cave-Ayland 1868f55ac13SMark Cave-Ayland static void macio_gpio_nmi(NMIState *n, int cpu_index, Error **errp) 1878f55ac13SMark Cave-Ayland { 1888f55ac13SMark Cave-Ayland macio_set_gpio(MACIO_GPIO(n), 9, true); 1898f55ac13SMark Cave-Ayland macio_set_gpio(MACIO_GPIO(n), 9, false); 1908f55ac13SMark Cave-Ayland } 1918f55ac13SMark Cave-Ayland 1927c4166a9SMark Cave-Ayland static void macio_gpio_class_init(ObjectClass *oc, void *data) 1937c4166a9SMark Cave-Ayland { 1947c4166a9SMark Cave-Ayland DeviceClass *dc = DEVICE_CLASS(oc); 1958f55ac13SMark Cave-Ayland NMIClass *nc = NMI_CLASS(oc); 1967c4166a9SMark Cave-Ayland 1977c4166a9SMark Cave-Ayland dc->reset = macio_gpio_reset; 1987c4166a9SMark Cave-Ayland dc->vmsd = &vmstate_macio_gpio; 1998f55ac13SMark Cave-Ayland nc->nmi_monitor_handler = macio_gpio_nmi; 2007c4166a9SMark Cave-Ayland } 2017c4166a9SMark Cave-Ayland 2027c4166a9SMark Cave-Ayland static const TypeInfo macio_gpio_init_info = { 2037c4166a9SMark Cave-Ayland .name = TYPE_MACIO_GPIO, 2047c4166a9SMark Cave-Ayland .parent = TYPE_SYS_BUS_DEVICE, 2057c4166a9SMark Cave-Ayland .instance_size = sizeof(MacIOGPIOState), 2067c4166a9SMark Cave-Ayland .instance_init = macio_gpio_init, 2077c4166a9SMark Cave-Ayland .class_init = macio_gpio_class_init, 2088f55ac13SMark Cave-Ayland .interfaces = (InterfaceInfo[]) { 2098f55ac13SMark Cave-Ayland { TYPE_NMI }, 2108f55ac13SMark Cave-Ayland { } 2118f55ac13SMark Cave-Ayland }, 2127c4166a9SMark Cave-Ayland }; 2137c4166a9SMark Cave-Ayland 2147c4166a9SMark Cave-Ayland static void macio_gpio_register_types(void) 2157c4166a9SMark Cave-Ayland { 2167c4166a9SMark Cave-Ayland type_register_static(&macio_gpio_init_info); 2177c4166a9SMark Cave-Ayland } 2187c4166a9SMark Cave-Ayland 2197c4166a9SMark Cave-Ayland type_init(macio_gpio_register_types) 220