Andrew Jeffery | 1c8a238 | 2016-06-27 15:37:33 +0100 | [diff] [blame] | 1 | /* |
| 2 | * ASPEED System Control Unit |
| 3 | * |
| 4 | * Andrew Jeffery <andrew@aj.id.au> |
| 5 | * |
| 6 | * Copyright 2016 IBM Corp. |
| 7 | * |
| 8 | * This code is licensed under the GPL version 2 or later. See |
| 9 | * the COPYING file in the top-level directory. |
| 10 | */ |
| 11 | |
| 12 | #include "qemu/osdep.h" |
| 13 | #include "hw/misc/aspeed_scu.h" |
| 14 | #include "hw/qdev-properties.h" |
| 15 | #include "qapi/error.h" |
| 16 | #include "qapi/visitor.h" |
| 17 | #include "qemu/bitops.h" |
Pranith Kumar | aa4b04a | 2016-06-27 17:53:04 -0400 | [diff] [blame] | 18 | #include "qemu/log.h" |
Andrew Jeffery | 1c8a238 | 2016-06-27 15:37:33 +0100 | [diff] [blame] | 19 | #include "trace.h" |
| 20 | |
| 21 | #define TO_REG(offset) ((offset) >> 2) |
| 22 | |
| 23 | #define PROT_KEY TO_REG(0x00) |
| 24 | #define SYS_RST_CTRL TO_REG(0x04) |
| 25 | #define CLK_SEL TO_REG(0x08) |
| 26 | #define CLK_STOP_CTRL TO_REG(0x0C) |
| 27 | #define FREQ_CNTR_CTRL TO_REG(0x10) |
| 28 | #define FREQ_CNTR_EVAL TO_REG(0x14) |
| 29 | #define IRQ_CTRL TO_REG(0x18) |
| 30 | #define D2PLL_PARAM TO_REG(0x1C) |
| 31 | #define MPLL_PARAM TO_REG(0x20) |
| 32 | #define HPLL_PARAM TO_REG(0x24) |
| 33 | #define FREQ_CNTR_RANGE TO_REG(0x28) |
| 34 | #define MISC_CTRL1 TO_REG(0x2C) |
| 35 | #define PCI_CTRL1 TO_REG(0x30) |
| 36 | #define PCI_CTRL2 TO_REG(0x34) |
| 37 | #define PCI_CTRL3 TO_REG(0x38) |
| 38 | #define SYS_RST_STATUS TO_REG(0x3C) |
| 39 | #define SOC_SCRATCH1 TO_REG(0x40) |
| 40 | #define SOC_SCRATCH2 TO_REG(0x44) |
| 41 | #define MAC_CLK_DELAY TO_REG(0x48) |
| 42 | #define MISC_CTRL2 TO_REG(0x4C) |
| 43 | #define VGA_SCRATCH1 TO_REG(0x50) |
| 44 | #define VGA_SCRATCH2 TO_REG(0x54) |
| 45 | #define VGA_SCRATCH3 TO_REG(0x58) |
| 46 | #define VGA_SCRATCH4 TO_REG(0x5C) |
| 47 | #define VGA_SCRATCH5 TO_REG(0x60) |
| 48 | #define VGA_SCRATCH6 TO_REG(0x64) |
| 49 | #define VGA_SCRATCH7 TO_REG(0x68) |
| 50 | #define VGA_SCRATCH8 TO_REG(0x6C) |
| 51 | #define HW_STRAP1 TO_REG(0x70) |
| 52 | #define RNG_CTRL TO_REG(0x74) |
| 53 | #define RNG_DATA TO_REG(0x78) |
| 54 | #define SILICON_REV TO_REG(0x7C) |
| 55 | #define PINMUX_CTRL1 TO_REG(0x80) |
| 56 | #define PINMUX_CTRL2 TO_REG(0x84) |
| 57 | #define PINMUX_CTRL3 TO_REG(0x88) |
| 58 | #define PINMUX_CTRL4 TO_REG(0x8C) |
| 59 | #define PINMUX_CTRL5 TO_REG(0x90) |
| 60 | #define PINMUX_CTRL6 TO_REG(0x94) |
| 61 | #define WDT_RST_CTRL TO_REG(0x9C) |
| 62 | #define PINMUX_CTRL7 TO_REG(0xA0) |
| 63 | #define PINMUX_CTRL8 TO_REG(0xA4) |
| 64 | #define PINMUX_CTRL9 TO_REG(0xA8) |
| 65 | #define WAKEUP_EN TO_REG(0xC0) |
| 66 | #define WAKEUP_CTRL TO_REG(0xC4) |
| 67 | #define HW_STRAP2 TO_REG(0xD0) |
| 68 | #define FREE_CNTR4 TO_REG(0xE0) |
| 69 | #define FREE_CNTR4_EXT TO_REG(0xE4) |
| 70 | #define CPU2_CTRL TO_REG(0x100) |
| 71 | #define CPU2_BASE_SEG1 TO_REG(0x104) |
| 72 | #define CPU2_BASE_SEG2 TO_REG(0x108) |
| 73 | #define CPU2_BASE_SEG3 TO_REG(0x10C) |
| 74 | #define CPU2_BASE_SEG4 TO_REG(0x110) |
| 75 | #define CPU2_BASE_SEG5 TO_REG(0x114) |
| 76 | #define CPU2_CACHE_CTRL TO_REG(0x118) |
| 77 | #define UART_HPLL_CLK TO_REG(0x160) |
| 78 | #define PCIE_CTRL TO_REG(0x180) |
| 79 | #define BMC_MMIO_CTRL TO_REG(0x184) |
| 80 | #define RELOC_DECODE_BASE1 TO_REG(0x188) |
| 81 | #define RELOC_DECODE_BASE2 TO_REG(0x18C) |
| 82 | #define MAILBOX_DECODE_BASE TO_REG(0x190) |
| 83 | #define SRAM_DECODE_BASE1 TO_REG(0x194) |
| 84 | #define SRAM_DECODE_BASE2 TO_REG(0x198) |
| 85 | #define BMC_REV TO_REG(0x19C) |
| 86 | #define BMC_DEV_ID TO_REG(0x1A4) |
| 87 | |
Cédric Le Goater | c491e15 | 2016-12-27 14:59:28 +0000 | [diff] [blame] | 88 | #define SCU_IO_REGION_SIZE 0x1000 |
Andrew Jeffery | 1c8a238 | 2016-06-27 15:37:33 +0100 | [diff] [blame] | 89 | |
Andrew Jeffery | 1c8a238 | 2016-06-27 15:37:33 +0100 | [diff] [blame] | 90 | static const uint32_t ast2400_a0_resets[ASPEED_SCU_NR_REGS] = { |
| 91 | [SYS_RST_CTRL] = 0xFFCFFEDCU, |
| 92 | [CLK_SEL] = 0xF3F40000U, |
| 93 | [CLK_STOP_CTRL] = 0x19FC3E8BU, |
| 94 | [D2PLL_PARAM] = 0x00026108U, |
| 95 | [MPLL_PARAM] = 0x00030291U, |
| 96 | [HPLL_PARAM] = 0x00000291U, |
| 97 | [MISC_CTRL1] = 0x00000010U, |
| 98 | [PCI_CTRL1] = 0x20001A03U, |
| 99 | [PCI_CTRL2] = 0x20001A03U, |
| 100 | [PCI_CTRL3] = 0x04000030U, |
| 101 | [SYS_RST_STATUS] = 0x00000001U, |
| 102 | [SOC_SCRATCH1] = 0x000000C0U, /* SoC completed DRAM init */ |
| 103 | [MISC_CTRL2] = 0x00000023U, |
| 104 | [RNG_CTRL] = 0x0000000EU, |
| 105 | [PINMUX_CTRL2] = 0x0000F000U, |
| 106 | [PINMUX_CTRL3] = 0x01000000U, |
| 107 | [PINMUX_CTRL4] = 0x000000FFU, |
| 108 | [PINMUX_CTRL5] = 0x0000A000U, |
| 109 | [WDT_RST_CTRL] = 0x003FFFF3U, |
| 110 | [PINMUX_CTRL8] = 0xFFFF0000U, |
| 111 | [PINMUX_CTRL9] = 0x000FFFFFU, |
| 112 | [FREE_CNTR4] = 0x000000FFU, |
| 113 | [FREE_CNTR4_EXT] = 0x000000FFU, |
| 114 | [CPU2_BASE_SEG1] = 0x80000000U, |
| 115 | [CPU2_BASE_SEG4] = 0x1E600000U, |
| 116 | [CPU2_BASE_SEG5] = 0xC0000000U, |
| 117 | [UART_HPLL_CLK] = 0x00001903U, |
| 118 | [PCIE_CTRL] = 0x0000007BU, |
| 119 | [BMC_DEV_ID] = 0x00002402U |
| 120 | }; |
| 121 | |
Cédric Le Goater | 365aff1 | 2016-09-22 18:13:05 +0100 | [diff] [blame] | 122 | /* SCU70 bit 23: 0 24Mhz. bit 11:9: 0b001 AXI:ABH ratio 2:1 */ |
| 123 | /* AST2500 revision A1 */ |
| 124 | |
| 125 | static const uint32_t ast2500_a1_resets[ASPEED_SCU_NR_REGS] = { |
| 126 | [SYS_RST_CTRL] = 0xFFCFFEDCU, |
| 127 | [CLK_SEL] = 0xF3F40000U, |
| 128 | [CLK_STOP_CTRL] = 0x19FC3E8BU, |
| 129 | [D2PLL_PARAM] = 0x00026108U, |
| 130 | [MPLL_PARAM] = 0x00030291U, |
| 131 | [HPLL_PARAM] = 0x93000400U, |
| 132 | [MISC_CTRL1] = 0x00000010U, |
| 133 | [PCI_CTRL1] = 0x20001A03U, |
| 134 | [PCI_CTRL2] = 0x20001A03U, |
| 135 | [PCI_CTRL3] = 0x04000030U, |
| 136 | [SYS_RST_STATUS] = 0x00000001U, |
| 137 | [SOC_SCRATCH1] = 0x000000C0U, /* SoC completed DRAM init */ |
| 138 | [MISC_CTRL2] = 0x00000023U, |
| 139 | [RNG_CTRL] = 0x0000000EU, |
| 140 | [PINMUX_CTRL2] = 0x0000F000U, |
| 141 | [PINMUX_CTRL3] = 0x03000000U, |
| 142 | [PINMUX_CTRL4] = 0x00000000U, |
| 143 | [PINMUX_CTRL5] = 0x0000A000U, |
| 144 | [WDT_RST_CTRL] = 0x023FFFF3U, |
| 145 | [PINMUX_CTRL8] = 0xFFFF0000U, |
| 146 | [PINMUX_CTRL9] = 0x000FFFFFU, |
| 147 | [FREE_CNTR4] = 0x000000FFU, |
| 148 | [FREE_CNTR4_EXT] = 0x000000FFU, |
| 149 | [CPU2_BASE_SEG1] = 0x80000000U, |
| 150 | [CPU2_BASE_SEG4] = 0x1E600000U, |
| 151 | [CPU2_BASE_SEG5] = 0xC0000000U, |
| 152 | [UART_HPLL_CLK] = 0x00001903U, |
| 153 | [PCIE_CTRL] = 0x0000007BU, |
| 154 | [BMC_DEV_ID] = 0x00002402U |
| 155 | }; |
| 156 | |
Andrew Jeffery | 1c8a238 | 2016-06-27 15:37:33 +0100 | [diff] [blame] | 157 | static uint64_t aspeed_scu_read(void *opaque, hwaddr offset, unsigned size) |
| 158 | { |
| 159 | AspeedSCUState *s = ASPEED_SCU(opaque); |
| 160 | int reg = TO_REG(offset); |
| 161 | |
| 162 | if (reg >= ARRAY_SIZE(s->regs)) { |
| 163 | qemu_log_mask(LOG_GUEST_ERROR, |
| 164 | "%s: Out-of-bounds read at offset 0x%" HWADDR_PRIx "\n", |
| 165 | __func__, offset); |
| 166 | return 0; |
| 167 | } |
| 168 | |
| 169 | switch (reg) { |
| 170 | case WAKEUP_EN: |
| 171 | qemu_log_mask(LOG_GUEST_ERROR, |
| 172 | "%s: Read of write-only offset 0x%" HWADDR_PRIx "\n", |
| 173 | __func__, offset); |
| 174 | break; |
| 175 | } |
| 176 | |
| 177 | return s->regs[reg]; |
| 178 | } |
| 179 | |
| 180 | static void aspeed_scu_write(void *opaque, hwaddr offset, uint64_t data, |
| 181 | unsigned size) |
| 182 | { |
| 183 | AspeedSCUState *s = ASPEED_SCU(opaque); |
| 184 | int reg = TO_REG(offset); |
| 185 | |
| 186 | if (reg >= ARRAY_SIZE(s->regs)) { |
| 187 | qemu_log_mask(LOG_GUEST_ERROR, |
| 188 | "%s: Out-of-bounds write at offset 0x%" HWADDR_PRIx "\n", |
| 189 | __func__, offset); |
| 190 | return; |
| 191 | } |
| 192 | |
| 193 | if (reg > PROT_KEY && reg < CPU2_BASE_SEG1 && |
Joel Stanley | b6e70d1 | 2017-11-14 22:50:18 +1030 | [diff] [blame] | 194 | s->regs[PROT_KEY] != ASPEED_SCU_PROT_KEY) { |
Andrew Jeffery | 1c8a238 | 2016-06-27 15:37:33 +0100 | [diff] [blame] | 195 | qemu_log_mask(LOG_GUEST_ERROR, "%s: SCU is locked!\n", __func__); |
| 196 | return; |
| 197 | } |
| 198 | |
| 199 | trace_aspeed_scu_write(offset, size, data); |
| 200 | |
| 201 | switch (reg) { |
| 202 | case FREQ_CNTR_EVAL: |
| 203 | case VGA_SCRATCH1 ... VGA_SCRATCH8: |
| 204 | case RNG_DATA: |
| 205 | case SILICON_REV: |
| 206 | case FREE_CNTR4: |
| 207 | case FREE_CNTR4_EXT: |
| 208 | qemu_log_mask(LOG_GUEST_ERROR, |
| 209 | "%s: Write to read-only offset 0x%" HWADDR_PRIx "\n", |
| 210 | __func__, offset); |
| 211 | return; |
| 212 | } |
| 213 | |
| 214 | s->regs[reg] = data; |
| 215 | } |
| 216 | |
| 217 | static const MemoryRegionOps aspeed_scu_ops = { |
| 218 | .read = aspeed_scu_read, |
| 219 | .write = aspeed_scu_write, |
| 220 | .endianness = DEVICE_LITTLE_ENDIAN, |
| 221 | .valid.min_access_size = 4, |
| 222 | .valid.max_access_size = 4, |
| 223 | .valid.unaligned = false, |
| 224 | }; |
| 225 | |
| 226 | static void aspeed_scu_reset(DeviceState *dev) |
| 227 | { |
| 228 | AspeedSCUState *s = ASPEED_SCU(dev); |
| 229 | const uint32_t *reset; |
| 230 | |
| 231 | switch (s->silicon_rev) { |
| 232 | case AST2400_A0_SILICON_REV: |
Cédric Le Goater | 6efbac9 | 2016-12-27 14:59:28 +0000 | [diff] [blame] | 233 | case AST2400_A1_SILICON_REV: |
Andrew Jeffery | 1c8a238 | 2016-06-27 15:37:33 +0100 | [diff] [blame] | 234 | reset = ast2400_a0_resets; |
| 235 | break; |
Cédric Le Goater | 365aff1 | 2016-09-22 18:13:05 +0100 | [diff] [blame] | 236 | case AST2500_A0_SILICON_REV: |
| 237 | case AST2500_A1_SILICON_REV: |
| 238 | reset = ast2500_a1_resets; |
| 239 | break; |
Andrew Jeffery | 1c8a238 | 2016-06-27 15:37:33 +0100 | [diff] [blame] | 240 | default: |
| 241 | g_assert_not_reached(); |
| 242 | } |
| 243 | |
| 244 | memcpy(s->regs, reset, sizeof(s->regs)); |
| 245 | s->regs[SILICON_REV] = s->silicon_rev; |
| 246 | s->regs[HW_STRAP1] = s->hw_strap1; |
| 247 | s->regs[HW_STRAP2] = s->hw_strap2; |
Joel Stanley | b6e70d1 | 2017-11-14 22:50:18 +1030 | [diff] [blame] | 248 | s->regs[PROT_KEY] = s->hw_prot_key; |
Andrew Jeffery | 1c8a238 | 2016-06-27 15:37:33 +0100 | [diff] [blame] | 249 | } |
| 250 | |
Cédric Le Goater | 365aff1 | 2016-09-22 18:13:05 +0100 | [diff] [blame] | 251 | static uint32_t aspeed_silicon_revs[] = { |
| 252 | AST2400_A0_SILICON_REV, |
Cédric Le Goater | 6efbac9 | 2016-12-27 14:59:28 +0000 | [diff] [blame] | 253 | AST2400_A1_SILICON_REV, |
Cédric Le Goater | 365aff1 | 2016-09-22 18:13:05 +0100 | [diff] [blame] | 254 | AST2500_A0_SILICON_REV, |
| 255 | AST2500_A1_SILICON_REV, |
| 256 | }; |
Andrew Jeffery | 1c8a238 | 2016-06-27 15:37:33 +0100 | [diff] [blame] | 257 | |
Cédric Le Goater | 79a9f32 | 2016-07-14 16:51:39 +0100 | [diff] [blame] | 258 | bool is_supported_silicon_rev(uint32_t silicon_rev) |
Andrew Jeffery | 1c8a238 | 2016-06-27 15:37:33 +0100 | [diff] [blame] | 259 | { |
| 260 | int i; |
| 261 | |
| 262 | for (i = 0; i < ARRAY_SIZE(aspeed_silicon_revs); i++) { |
| 263 | if (silicon_rev == aspeed_silicon_revs[i]) { |
| 264 | return true; |
| 265 | } |
| 266 | } |
| 267 | |
| 268 | return false; |
| 269 | } |
| 270 | |
| 271 | static void aspeed_scu_realize(DeviceState *dev, Error **errp) |
| 272 | { |
| 273 | SysBusDevice *sbd = SYS_BUS_DEVICE(dev); |
| 274 | AspeedSCUState *s = ASPEED_SCU(dev); |
| 275 | |
| 276 | if (!is_supported_silicon_rev(s->silicon_rev)) { |
| 277 | error_setg(errp, "Unknown silicon revision: 0x%" PRIx32, |
| 278 | s->silicon_rev); |
| 279 | return; |
| 280 | } |
| 281 | |
| 282 | memory_region_init_io(&s->iomem, OBJECT(s), &aspeed_scu_ops, s, |
| 283 | TYPE_ASPEED_SCU, SCU_IO_REGION_SIZE); |
| 284 | |
| 285 | sysbus_init_mmio(sbd, &s->iomem); |
| 286 | } |
| 287 | |
| 288 | static const VMStateDescription vmstate_aspeed_scu = { |
| 289 | .name = "aspeed.scu", |
| 290 | .version_id = 1, |
| 291 | .minimum_version_id = 1, |
| 292 | .fields = (VMStateField[]) { |
| 293 | VMSTATE_UINT32_ARRAY(regs, AspeedSCUState, ASPEED_SCU_NR_REGS), |
| 294 | VMSTATE_END_OF_LIST() |
| 295 | } |
| 296 | }; |
| 297 | |
| 298 | static Property aspeed_scu_properties[] = { |
| 299 | DEFINE_PROP_UINT32("silicon-rev", AspeedSCUState, silicon_rev, 0), |
| 300 | DEFINE_PROP_UINT32("hw-strap1", AspeedSCUState, hw_strap1, 0), |
Cédric Le Goater | 2ddfa28 | 2016-07-14 16:51:38 +0100 | [diff] [blame] | 301 | DEFINE_PROP_UINT32("hw-strap2", AspeedSCUState, hw_strap2, 0), |
Joel Stanley | b6e70d1 | 2017-11-14 22:50:18 +1030 | [diff] [blame] | 302 | DEFINE_PROP_UINT32("hw-prot-key", AspeedSCUState, hw_prot_key, 0), |
Andrew Jeffery | 1c8a238 | 2016-06-27 15:37:33 +0100 | [diff] [blame] | 303 | DEFINE_PROP_END_OF_LIST(), |
| 304 | }; |
| 305 | |
| 306 | static void aspeed_scu_class_init(ObjectClass *klass, void *data) |
| 307 | { |
| 308 | DeviceClass *dc = DEVICE_CLASS(klass); |
| 309 | dc->realize = aspeed_scu_realize; |
| 310 | dc->reset = aspeed_scu_reset; |
| 311 | dc->desc = "ASPEED System Control Unit"; |
| 312 | dc->vmsd = &vmstate_aspeed_scu; |
| 313 | dc->props = aspeed_scu_properties; |
| 314 | } |
| 315 | |
| 316 | static const TypeInfo aspeed_scu_info = { |
| 317 | .name = TYPE_ASPEED_SCU, |
| 318 | .parent = TYPE_SYS_BUS_DEVICE, |
| 319 | .instance_size = sizeof(AspeedSCUState), |
| 320 | .class_init = aspeed_scu_class_init, |
| 321 | }; |
| 322 | |
| 323 | static void aspeed_scu_register_types(void) |
| 324 | { |
| 325 | type_register_static(&aspeed_scu_info); |
| 326 | } |
| 327 | |
| 328 | type_init(aspeed_scu_register_types); |