Alexander Graf | 7fb6577 | 2011-02-01 15:51:28 +0100 | [diff] [blame] | 1 | /* |
| 2 | * QEMU ICH Emulation |
| 3 | * |
| 4 | * Copyright (c) 2010 Sebastian Herbszt <herbszt@gmx.de> |
| 5 | * Copyright (c) 2010 Alexander Graf <agraf@suse.de> |
| 6 | * |
| 7 | * This library is free software; you can redistribute it and/or |
| 8 | * modify it under the terms of the GNU Lesser General Public |
| 9 | * License as published by the Free Software Foundation; either |
| 10 | * version 2 of the License, or (at your option) any later version. |
| 11 | * |
| 12 | * This library is distributed in the hope that it will be useful, |
| 13 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| 15 | * Lesser General Public License for more details. |
| 16 | * |
| 17 | * You should have received a copy of the GNU Lesser General Public |
| 18 | * License along with this library; if not, see <http://www.gnu.org/licenses/>. |
| 19 | * |
| 20 | * |
| 21 | * lspci dump of a ICH-9 real device |
| 22 | * |
| 23 | * 00:1f.2 SATA controller [0106]: Intel Corporation 82801IR/IO/IH (ICH9R/DO/DH) 6 port SATA AHCI Controller [8086:2922] (rev 02) (prog-if 01 [AHCI 1.0]) |
| 24 | * Subsystem: Intel Corporation 82801IR/IO/IH (ICH9R/DO/DH) 6 port SATA AHCI Controller [8086:2922] |
| 25 | * Control: I/O+ Mem+ BusMaster+ SpecCycle- MemWINV- VGASnoop- ParErr- Stepping- SERR- FastB2B- DisINTx+ |
| 26 | * Status: Cap+ 66MHz+ UDF- FastB2B+ ParErr- DEVSEL=medium >TAbort- <TAbort- <MAbort- >SERR- <PERR- INTx- |
| 27 | * Latency: 0 |
| 28 | * Interrupt: pin B routed to IRQ 222 |
| 29 | * Region 0: I/O ports at d000 [size=8] |
| 30 | * Region 1: I/O ports at cc00 [size=4] |
| 31 | * Region 2: I/O ports at c880 [size=8] |
| 32 | * Region 3: I/O ports at c800 [size=4] |
| 33 | * Region 4: I/O ports at c480 [size=32] |
| 34 | * Region 5: Memory at febf9000 (32-bit, non-prefetchable) [size=2K] |
| 35 | * Capabilities: [80] Message Signalled Interrupts: Mask- 64bit- Count=1/16 Enable+ |
| 36 | * Address: fee0f00c Data: 41d9 |
| 37 | * Capabilities: [70] Power Management version 3 |
| 38 | * Flags: PMEClk- DSI- D1- D2- AuxCurrent=0mA PME(D0-,D1-,D2-,D3hot+,D3cold-) |
| 39 | * Status: D0 PME-Enable- DSel=0 DScale=0 PME- |
| 40 | * Capabilities: [a8] SATA HBA <?> |
| 41 | * Capabilities: [b0] Vendor Specific Information <?> |
| 42 | * Kernel driver in use: ahci |
| 43 | * Kernel modules: ahci |
| 44 | * 00: 86 80 22 29 07 04 b0 02 02 01 06 01 00 00 00 00 |
| 45 | * 10: 01 d0 00 00 01 cc 00 00 81 c8 00 00 01 c8 00 00 |
| 46 | * 20: 81 c4 00 00 00 90 bf fe 00 00 00 00 86 80 22 29 |
| 47 | * 30: 00 00 00 00 80 00 00 00 00 00 00 00 0f 02 00 00 |
| 48 | * 40: 00 80 00 80 00 00 00 00 00 00 00 00 00 00 00 00 |
| 49 | * 50: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 |
| 50 | * 60: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 |
| 51 | * 70: 01 a8 03 40 08 00 00 00 00 00 00 00 00 00 00 00 |
| 52 | * 80: 05 70 09 00 0c f0 e0 fe d9 41 00 00 00 00 00 00 |
| 53 | * 90: 40 00 0f 82 93 01 00 00 00 00 00 00 00 00 00 00 |
| 54 | * a0: ac 00 00 00 0a 00 12 00 12 b0 10 00 48 00 00 00 |
| 55 | * b0: 09 00 06 20 00 00 00 00 00 00 00 00 00 00 00 00 |
| 56 | * c0: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 |
| 57 | * d0: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 |
| 58 | * e0: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 |
| 59 | * f0: 00 00 00 00 00 00 00 00 86 0f 02 00 00 00 00 00 |
| 60 | * |
| 61 | */ |
| 62 | |
Sebastian Herbszt | 03c7a6a | 2011-02-01 15:51:26 +0100 | [diff] [blame] | 63 | #include <hw/hw.h> |
Michael S. Tsirkin | a2cb15b | 2012-12-12 14:24:50 +0200 | [diff] [blame] | 64 | #include <hw/pci/msi.h> |
Paolo Bonzini | 0d09e41 | 2013-02-05 17:06:20 +0100 | [diff] [blame] | 65 | #include <hw/i386/pc.h> |
Michael S. Tsirkin | a2cb15b | 2012-12-12 14:24:50 +0200 | [diff] [blame] | 66 | #include <hw/pci/pci.h> |
Paolo Bonzini | 0d09e41 | 2013-02-05 17:06:20 +0100 | [diff] [blame] | 67 | #include <hw/isa/isa.h> |
Markus Armbruster | 4be7463 | 2014-10-07 13:59:18 +0200 | [diff] [blame] | 68 | #include "sysemu/block-backend.h" |
Paolo Bonzini | 9c17d61 | 2012-12-17 18:20:04 +0100 | [diff] [blame] | 69 | #include "sysemu/dma.h" |
Sebastian Herbszt | 03c7a6a | 2011-02-01 15:51:26 +0100 | [diff] [blame] | 70 | |
| 71 | #include <hw/ide/pci.h> |
| 72 | #include <hw/ide/ahci.h> |
| 73 | |
John Snow | c8b5b20 | 2014-08-21 13:44:33 -0400 | [diff] [blame] | 74 | #define ICH9_MSI_CAP_OFFSET 0x80 |
Daniel Verkamp | 465f1ab | 2011-08-27 02:12:28 -0700 | [diff] [blame] | 75 | #define ICH9_SATA_CAP_OFFSET 0xA8 |
| 76 | |
| 77 | #define ICH9_IDP_BAR 4 |
| 78 | #define ICH9_MEM_BAR 5 |
| 79 | |
| 80 | #define ICH9_IDP_INDEX 0x10 |
| 81 | #define ICH9_IDP_INDEX_LOG2 0x04 |
| 82 | |
Jason Baron | a262302 | 2013-01-04 14:44:42 -0500 | [diff] [blame] | 83 | static const VMStateDescription vmstate_ich9_ahci = { |
| 84 | .name = "ich9_ahci", |
Jason Baron | a262302 | 2013-01-04 14:44:42 -0500 | [diff] [blame] | 85 | .version_id = 1, |
Juan Quintela | d49805a | 2014-04-16 15:32:32 +0200 | [diff] [blame] | 86 | .fields = (VMStateField[]) { |
Andreas Färber | 0d3aea5 | 2013-06-30 14:19:24 +0200 | [diff] [blame] | 87 | VMSTATE_PCI_DEVICE(parent_obj, AHCIPCIState), |
Jason Baron | a262302 | 2013-01-04 14:44:42 -0500 | [diff] [blame] | 88 | VMSTATE_AHCI(ahci, AHCIPCIState), |
| 89 | VMSTATE_END_OF_LIST() |
| 90 | }, |
Gerd Hoffmann | b7ce1b2 | 2011-07-08 10:48:37 +0200 | [diff] [blame] | 91 | }; |
| 92 | |
Jan Kiszka | 8ab60a0 | 2012-05-11 11:42:36 -0300 | [diff] [blame] | 93 | static void pci_ich9_reset(DeviceState *dev) |
Jan Kiszka | 868a1a5 | 2012-05-11 11:42:34 -0300 | [diff] [blame] | 94 | { |
Peter Crosthwaite | fd58922 | 2013-06-24 16:55:45 +1000 | [diff] [blame] | 95 | AHCIPCIState *d = ICH_AHCI(dev); |
Jan Kiszka | 868a1a5 | 2012-05-11 11:42:34 -0300 | [diff] [blame] | 96 | |
Jan Kiszka | 8ab60a0 | 2012-05-11 11:42:36 -0300 | [diff] [blame] | 97 | ahci_reset(&d->ahci); |
Jan Kiszka | 868a1a5 | 2012-05-11 11:42:34 -0300 | [diff] [blame] | 98 | } |
| 99 | |
Peter Crosthwaite | 0487eea | 2015-11-06 14:09:00 -0500 | [diff] [blame] | 100 | static void pci_ich9_ahci_init(Object *obj) |
| 101 | { |
| 102 | struct AHCIPCIState *d = ICH_AHCI(obj); |
| 103 | |
| 104 | ahci_init(&d->ahci, DEVICE(obj)); |
| 105 | } |
| 106 | |
Markus Armbruster | b8a2dac | 2015-01-19 15:52:34 +0100 | [diff] [blame] | 107 | static void pci_ich9_ahci_realize(PCIDevice *dev, Error **errp) |
Sebastian Herbszt | 03c7a6a | 2011-02-01 15:51:26 +0100 | [diff] [blame] | 108 | { |
| 109 | struct AHCIPCIState *d; |
Daniel Verkamp | 465f1ab | 2011-08-27 02:12:28 -0700 | [diff] [blame] | 110 | int sata_cap_offset; |
| 111 | uint8_t *sata_cap; |
Peter Crosthwaite | fd58922 | 2013-06-24 16:55:45 +1000 | [diff] [blame] | 112 | d = ICH_AHCI(dev); |
Sebastian Herbszt | 03c7a6a | 2011-02-01 15:51:26 +0100 | [diff] [blame] | 113 | |
Peter Crosthwaite | 0487eea | 2015-11-06 14:09:00 -0500 | [diff] [blame] | 114 | ahci_realize(&d->ahci, DEVICE(dev), pci_get_address_space(dev), 6); |
Michael S. Tsirkin | 69c8944 | 2011-05-15 19:27:34 +0300 | [diff] [blame] | 115 | |
Andreas Färber | 0d3aea5 | 2013-06-30 14:19:24 +0200 | [diff] [blame] | 116 | pci_config_set_prog_interface(dev->config, AHCI_PROGMODE_MAJOR_REV_1); |
Sebastian Herbszt | 03c7a6a | 2011-02-01 15:51:26 +0100 | [diff] [blame] | 117 | |
Andreas Färber | 0d3aea5 | 2013-06-30 14:19:24 +0200 | [diff] [blame] | 118 | dev->config[PCI_CACHE_LINE_SIZE] = 0x08; /* Cache line size */ |
| 119 | dev->config[PCI_LATENCY_TIMER] = 0x00; /* Latency timer */ |
| 120 | pci_config_set_interrupt_pin(dev->config, 1); |
Sebastian Herbszt | 03c7a6a | 2011-02-01 15:51:26 +0100 | [diff] [blame] | 121 | |
| 122 | /* XXX Software should program this register */ |
Andreas Färber | 0d3aea5 | 2013-06-30 14:19:24 +0200 | [diff] [blame] | 123 | dev->config[0x90] = 1 << 6; /* Address Map Register - AHCI mode */ |
Sebastian Herbszt | 03c7a6a | 2011-02-01 15:51:26 +0100 | [diff] [blame] | 124 | |
Marcel Apfelbaum | 9e64f8a | 2013-10-07 10:36:39 +0300 | [diff] [blame] | 125 | d->ahci.irq = pci_allocate_irq(dev); |
Sebastian Herbszt | 03c7a6a | 2011-02-01 15:51:26 +0100 | [diff] [blame] | 126 | |
Andreas Färber | 0d3aea5 | 2013-06-30 14:19:24 +0200 | [diff] [blame] | 127 | pci_register_bar(dev, ICH9_IDP_BAR, PCI_BASE_ADDRESS_SPACE_IO, |
Daniel Verkamp | 465f1ab | 2011-08-27 02:12:28 -0700 | [diff] [blame] | 128 | &d->ahci.idp); |
Andreas Färber | 0d3aea5 | 2013-06-30 14:19:24 +0200 | [diff] [blame] | 129 | pci_register_bar(dev, ICH9_MEM_BAR, PCI_BASE_ADDRESS_SPACE_MEMORY, |
Daniel Verkamp | 465f1ab | 2011-08-27 02:12:28 -0700 | [diff] [blame] | 130 | &d->ahci.mem); |
| 131 | |
Markus Armbruster | b8a2dac | 2015-01-19 15:52:34 +0100 | [diff] [blame] | 132 | sata_cap_offset = pci_add_capability2(dev, PCI_CAP_ID_SATA, |
| 133 | ICH9_SATA_CAP_OFFSET, SATA_CAP_SIZE, |
| 134 | errp); |
Daniel Verkamp | 465f1ab | 2011-08-27 02:12:28 -0700 | [diff] [blame] | 135 | if (sata_cap_offset < 0) { |
Markus Armbruster | b8a2dac | 2015-01-19 15:52:34 +0100 | [diff] [blame] | 136 | return; |
Daniel Verkamp | 465f1ab | 2011-08-27 02:12:28 -0700 | [diff] [blame] | 137 | } |
| 138 | |
Andreas Färber | 0d3aea5 | 2013-06-30 14:19:24 +0200 | [diff] [blame] | 139 | sata_cap = dev->config + sata_cap_offset; |
Daniel Verkamp | 465f1ab | 2011-08-27 02:12:28 -0700 | [diff] [blame] | 140 | pci_set_word(sata_cap + SATA_CAP_REV, 0x10); |
| 141 | pci_set_long(sata_cap + SATA_CAP_BAR, |
| 142 | (ICH9_IDP_BAR + 0x4) | (ICH9_IDP_INDEX_LOG2 << 4)); |
| 143 | d->ahci.idp_offset = ICH9_IDP_INDEX; |
Jan Kiszka | 96d19bc | 2011-05-08 19:54:52 +0200 | [diff] [blame] | 144 | |
John Snow | c8b5b20 | 2014-08-21 13:44:33 -0400 | [diff] [blame] | 145 | /* Although the AHCI 1.3 specification states that the first capability |
| 146 | * should be PMCAP, the Intel ICH9 data sheet specifies that the ICH9 |
| 147 | * AHCI device puts the MSI capability first, pointing to 0x80. */ |
| 148 | msi_init(dev, ICH9_MSI_CAP_OFFSET, 1, true, false); |
Sebastian Herbszt | 03c7a6a | 2011-02-01 15:51:26 +0100 | [diff] [blame] | 149 | } |
| 150 | |
Alex Williamson | f90c2bc | 2012-07-03 22:39:27 -0600 | [diff] [blame] | 151 | static void pci_ich9_uninit(PCIDevice *dev) |
Alexander Graf | 7fb6577 | 2011-02-01 15:51:28 +0100 | [diff] [blame] | 152 | { |
| 153 | struct AHCIPCIState *d; |
Peter Crosthwaite | fd58922 | 2013-06-24 16:55:45 +1000 | [diff] [blame] | 154 | d = ICH_AHCI(dev); |
Alexander Graf | 7fb6577 | 2011-02-01 15:51:28 +0100 | [diff] [blame] | 155 | |
Jan Kiszka | 45fe15c | 2011-05-02 20:00:47 +0200 | [diff] [blame] | 156 | msi_uninit(dev); |
Alexander Graf | 2c4b9d0 | 2011-02-01 15:51:31 +0100 | [diff] [blame] | 157 | ahci_uninit(&d->ahci); |
Marcel Apfelbaum | 9e64f8a | 2013-10-07 10:36:39 +0300 | [diff] [blame] | 158 | qemu_free_irq(d->ahci.irq); |
Alexander Graf | 7fb6577 | 2011-02-01 15:51:28 +0100 | [diff] [blame] | 159 | } |
| 160 | |
Anthony Liguori | 40021f0 | 2011-12-04 12:22:06 -0600 | [diff] [blame] | 161 | static void ich_ahci_class_init(ObjectClass *klass, void *data) |
| 162 | { |
Anthony Liguori | 39bffca | 2011-12-07 21:34:16 -0600 | [diff] [blame] | 163 | DeviceClass *dc = DEVICE_CLASS(klass); |
Anthony Liguori | 40021f0 | 2011-12-04 12:22:06 -0600 | [diff] [blame] | 164 | PCIDeviceClass *k = PCI_DEVICE_CLASS(klass); |
| 165 | |
Markus Armbruster | b8a2dac | 2015-01-19 15:52:34 +0100 | [diff] [blame] | 166 | k->realize = pci_ich9_ahci_realize; |
Anthony Liguori | 40021f0 | 2011-12-04 12:22:06 -0600 | [diff] [blame] | 167 | k->exit = pci_ich9_uninit; |
Anthony Liguori | 40021f0 | 2011-12-04 12:22:06 -0600 | [diff] [blame] | 168 | k->vendor_id = PCI_VENDOR_ID_INTEL; |
| 169 | k->device_id = PCI_DEVICE_ID_INTEL_82801IR; |
| 170 | k->revision = 0x02; |
| 171 | k->class_id = PCI_CLASS_STORAGE_SATA; |
Jason Baron | a262302 | 2013-01-04 14:44:42 -0500 | [diff] [blame] | 172 | dc->vmsd = &vmstate_ich9_ahci; |
Jan Kiszka | 8ab60a0 | 2012-05-11 11:42:36 -0300 | [diff] [blame] | 173 | dc->reset = pci_ich9_reset; |
Marcel Apfelbaum | 125ee0e | 2013-07-29 17:17:45 +0300 | [diff] [blame] | 174 | set_bit(DEVICE_CATEGORY_STORAGE, dc->categories); |
Anthony Liguori | 40021f0 | 2011-12-04 12:22:06 -0600 | [diff] [blame] | 175 | } |
| 176 | |
Andreas Färber | 8c43a6f | 2013-01-10 16:19:07 +0100 | [diff] [blame] | 177 | static const TypeInfo ich_ahci_info = { |
Peter Crosthwaite | fd58922 | 2013-06-24 16:55:45 +1000 | [diff] [blame] | 178 | .name = TYPE_ICH9_AHCI, |
Anthony Liguori | 39bffca | 2011-12-07 21:34:16 -0600 | [diff] [blame] | 179 | .parent = TYPE_PCI_DEVICE, |
| 180 | .instance_size = sizeof(AHCIPCIState), |
Peter Crosthwaite | 0487eea | 2015-11-06 14:09:00 -0500 | [diff] [blame] | 181 | .instance_init = pci_ich9_ahci_init, |
Anthony Liguori | 39bffca | 2011-12-07 21:34:16 -0600 | [diff] [blame] | 182 | .class_init = ich_ahci_class_init, |
Sebastian Herbszt | 03c7a6a | 2011-02-01 15:51:26 +0100 | [diff] [blame] | 183 | }; |
| 184 | |
Andreas Färber | 83f7d43 | 2012-02-09 15:20:55 +0100 | [diff] [blame] | 185 | static void ich_ahci_register_types(void) |
Sebastian Herbszt | 03c7a6a | 2011-02-01 15:51:26 +0100 | [diff] [blame] | 186 | { |
Anthony Liguori | 39bffca | 2011-12-07 21:34:16 -0600 | [diff] [blame] | 187 | type_register_static(&ich_ahci_info); |
Sebastian Herbszt | 03c7a6a | 2011-02-01 15:51:26 +0100 | [diff] [blame] | 188 | } |
Andreas Färber | 83f7d43 | 2012-02-09 15:20:55 +0100 | [diff] [blame] | 189 | |
| 190 | type_init(ich_ahci_register_types) |