Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 1 | /* |
| 2 | * QEMU OpenRISC CPU |
| 3 | * |
| 4 | * Copyright (c) 2012 Jia Liu <proljc@gmail.com> |
| 5 | * |
| 6 | * This library is free software; you can redistribute it and/or |
| 7 | * modify it under the terms of the GNU Lesser General Public |
| 8 | * License as published by the Free Software Foundation; either |
Thomas Huth | 779fc6a | 2019-01-23 15:08:54 +0100 | [diff] [blame] | 9 | * version 2.1 of the License, or (at your option) any later version. |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 10 | * |
| 11 | * This library is distributed in the hope that it will be useful, |
| 12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| 14 | * Lesser General Public License for more details. |
| 15 | * |
| 16 | * You should have received a copy of the GNU Lesser General Public |
| 17 | * License along with this library; if not, see <http://www.gnu.org/licenses/>. |
| 18 | */ |
| 19 | |
Peter Maydell | ed2decc | 2016-01-26 18:17:22 +0000 | [diff] [blame] | 20 | #include "qemu/osdep.h" |
Markus Armbruster | da34e65 | 2016-03-14 09:01:28 +0100 | [diff] [blame] | 21 | #include "qapi/error.h" |
Markus Armbruster | 0442428 | 2019-04-17 21:17:57 +0200 | [diff] [blame] | 22 | #include "qemu/qemu-print.h" |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 23 | #include "cpu.h" |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 24 | |
Andreas Färber | f45748f | 2013-06-21 19:09:18 +0200 | [diff] [blame] | 25 | static void openrisc_cpu_set_pc(CPUState *cs, vaddr value) |
| 26 | { |
| 27 | OpenRISCCPU *cpu = OPENRISC_CPU(cs); |
| 28 | |
| 29 | cpu->env.pc = value; |
Richard Henderson | e8f2904 | 2018-05-27 14:02:17 -0500 | [diff] [blame] | 30 | cpu->env.dflag = 0; |
Andreas Färber | f45748f | 2013-06-21 19:09:18 +0200 | [diff] [blame] | 31 | } |
| 32 | |
Andreas Färber | 8c2e1b0 | 2013-08-25 18:53:55 +0200 | [diff] [blame] | 33 | static bool openrisc_cpu_has_work(CPUState *cs) |
| 34 | { |
| 35 | return cs->interrupt_request & (CPU_INTERRUPT_HARD | |
| 36 | CPU_INTERRUPT_TIMER); |
| 37 | } |
| 38 | |
Richard Henderson | d5cabcc | 2018-05-23 08:14:46 -0700 | [diff] [blame] | 39 | static void openrisc_disas_set_info(CPUState *cpu, disassemble_info *info) |
| 40 | { |
| 41 | info->print_insn = print_insn_or1k; |
| 42 | } |
| 43 | |
Peter Maydell | 781c67c | 2020-03-03 10:05:11 +0000 | [diff] [blame] | 44 | static void openrisc_cpu_reset(DeviceState *dev) |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 45 | { |
Peter Maydell | 781c67c | 2020-03-03 10:05:11 +0000 | [diff] [blame] | 46 | CPUState *s = CPU(dev); |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 47 | OpenRISCCPU *cpu = OPENRISC_CPU(s); |
| 48 | OpenRISCCPUClass *occ = OPENRISC_CPU_GET_CLASS(cpu); |
| 49 | |
Peter Maydell | 781c67c | 2020-03-03 10:05:11 +0000 | [diff] [blame] | 50 | occ->parent_reset(dev); |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 51 | |
Alex Bennée | 1f5c00c | 2016-11-14 14:19:17 +0000 | [diff] [blame] | 52 | memset(&cpu->env, 0, offsetof(CPUOpenRISCState, end_reset_fields)); |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 53 | |
| 54 | cpu->env.pc = 0x100; |
| 55 | cpu->env.sr = SR_FO | SR_SM; |
Richard Henderson | 930c3d0 | 2015-02-18 22:19:18 -0800 | [diff] [blame] | 56 | cpu->env.lock_addr = -1; |
Andreas Färber | 2710342 | 2013-08-26 08:31:06 +0200 | [diff] [blame] | 57 | s->exception_index = -1; |
Richard Henderson | a465772 | 2019-08-26 15:10:10 -0700 | [diff] [blame] | 58 | cpu_set_fpcsr(&cpu->env, 0); |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 59 | |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 60 | #ifndef CONFIG_USER_ONLY |
| 61 | cpu->env.picmr = 0x00000000; |
| 62 | cpu->env.picsr = 0x00000000; |
| 63 | |
| 64 | cpu->env.ttmr = 0x00000000; |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 65 | #endif |
| 66 | } |
| 67 | |
Peter Maydell | 71b3254 | 2020-11-27 22:51:27 +0000 | [diff] [blame] | 68 | #ifndef CONFIG_USER_ONLY |
| 69 | static void openrisc_cpu_set_irq(void *opaque, int irq, int level) |
| 70 | { |
| 71 | OpenRISCCPU *cpu = (OpenRISCCPU *)opaque; |
| 72 | CPUState *cs = CPU(cpu); |
| 73 | uint32_t irq_bit; |
| 74 | |
| 75 | if (irq > 31 || irq < 0) { |
| 76 | return; |
| 77 | } |
| 78 | |
| 79 | irq_bit = 1U << irq; |
| 80 | |
| 81 | if (level) { |
| 82 | cpu->env.picsr |= irq_bit; |
| 83 | } else { |
| 84 | cpu->env.picsr &= ~irq_bit; |
| 85 | } |
| 86 | |
| 87 | if (cpu->env.picsr & cpu->env.picmr) { |
| 88 | cpu_interrupt(cs, CPU_INTERRUPT_HARD); |
| 89 | } else { |
| 90 | cpu_reset_interrupt(cs, CPU_INTERRUPT_HARD); |
| 91 | cpu->env.picsr = 0; |
| 92 | } |
| 93 | } |
| 94 | #endif |
| 95 | |
Andreas Färber | c296262 | 2013-01-05 14:11:07 +0100 | [diff] [blame] | 96 | static void openrisc_cpu_realizefn(DeviceState *dev, Error **errp) |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 97 | { |
Andreas Färber | 14a10fc | 2013-07-27 02:53:25 +0200 | [diff] [blame] | 98 | CPUState *cs = CPU(dev); |
Andreas Färber | c296262 | 2013-01-05 14:11:07 +0100 | [diff] [blame] | 99 | OpenRISCCPUClass *occ = OPENRISC_CPU_GET_CLASS(dev); |
Laurent Vivier | ce5b1bb | 2016-10-20 13:26:03 +0200 | [diff] [blame] | 100 | Error *local_err = NULL; |
| 101 | |
| 102 | cpu_exec_realizefn(cs, &local_err); |
| 103 | if (local_err != NULL) { |
| 104 | error_propagate(errp, local_err); |
| 105 | return; |
| 106 | } |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 107 | |
Andreas Färber | 14a10fc | 2013-07-27 02:53:25 +0200 | [diff] [blame] | 108 | qemu_init_vcpu(cs); |
| 109 | cpu_reset(cs); |
Andreas Färber | c296262 | 2013-01-05 14:11:07 +0100 | [diff] [blame] | 110 | |
| 111 | occ->parent_realize(dev, errp); |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 112 | } |
| 113 | |
| 114 | static void openrisc_cpu_initfn(Object *obj) |
| 115 | { |
| 116 | OpenRISCCPU *cpu = OPENRISC_CPU(obj); |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 117 | |
Richard Henderson | 7506ed9 | 2019-03-28 11:26:22 -1000 | [diff] [blame] | 118 | cpu_set_cpustate_pointers(cpu); |
Peter Maydell | 71b3254 | 2020-11-27 22:51:27 +0000 | [diff] [blame] | 119 | |
| 120 | #ifndef CONFIG_USER_ONLY |
| 121 | qdev_init_gpio_in_named(DEVICE(cpu), openrisc_cpu_set_irq, "IRQ", NR_IRQS); |
| 122 | #endif |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 123 | } |
| 124 | |
| 125 | /* CPU models */ |
Andreas Färber | bd039ce | 2013-01-23 11:17:14 +0100 | [diff] [blame] | 126 | |
| 127 | static ObjectClass *openrisc_cpu_class_by_name(const char *cpu_model) |
| 128 | { |
| 129 | ObjectClass *oc; |
Dongxue Zhang | 071b336 | 2013-07-02 17:11:55 +0800 | [diff] [blame] | 130 | char *typename; |
Andreas Färber | bd039ce | 2013-01-23 11:17:14 +0100 | [diff] [blame] | 131 | |
Igor Mammedov | a677273 | 2017-10-05 15:50:51 +0200 | [diff] [blame] | 132 | typename = g_strdup_printf(OPENRISC_CPU_TYPE_NAME("%s"), cpu_model); |
Dongxue Zhang | 071b336 | 2013-07-02 17:11:55 +0800 | [diff] [blame] | 133 | oc = object_class_by_name(typename); |
Jia Liu | 9b146e9 | 2013-07-23 18:32:30 +0800 | [diff] [blame] | 134 | g_free(typename); |
Andreas Färber | c432b78 | 2013-01-23 12:39:38 +0100 | [diff] [blame] | 135 | if (oc != NULL && (!object_class_dynamic_cast(oc, TYPE_OPENRISC_CPU) || |
| 136 | object_class_is_abstract(oc))) { |
Andreas Färber | bd039ce | 2013-01-23 11:17:14 +0100 | [diff] [blame] | 137 | return NULL; |
| 138 | } |
| 139 | return oc; |
| 140 | } |
| 141 | |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 142 | static void or1200_initfn(Object *obj) |
| 143 | { |
| 144 | OpenRISCCPU *cpu = OPENRISC_CPU(obj); |
| 145 | |
Richard Henderson | c7efab4 | 2019-08-25 15:02:54 -0700 | [diff] [blame] | 146 | cpu->env.vr = 0x13000008; |
| 147 | cpu->env.upr = UPR_UP | UPR_DMP | UPR_IMP | UPR_PICP | UPR_TTP | UPR_PMP; |
Stafford Horne | 48a1b62 | 2017-04-22 00:28:55 +0900 | [diff] [blame] | 148 | cpu->env.cpucfgr = CPUCFGR_NSGF | CPUCFGR_OB32S | CPUCFGR_OF32S | |
| 149 | CPUCFGR_EVBARP; |
Richard Henderson | c7efab4 | 2019-08-25 15:02:54 -0700 | [diff] [blame] | 150 | |
| 151 | /* 1Way, TLB_SIZE entries. */ |
| 152 | cpu->env.dmmucfgr = (DMMUCFGR_NTW & (0 << 2)) |
| 153 | | (DMMUCFGR_NTS & (ctz32(TLB_SIZE) << 2)); |
| 154 | cpu->env.immucfgr = (IMMUCFGR_NTW & (0 << 2)) |
| 155 | | (IMMUCFGR_NTS & (ctz32(TLB_SIZE) << 2)); |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 156 | } |
| 157 | |
| 158 | static void openrisc_any_initfn(Object *obj) |
| 159 | { |
| 160 | OpenRISCCPU *cpu = OPENRISC_CPU(obj); |
| 161 | |
Richard Henderson | 8bebf7d | 2019-08-25 15:23:42 -0700 | [diff] [blame] | 162 | cpu->env.vr = 0x13000040; /* Obsolete VER + UVRP for new SPRs */ |
| 163 | cpu->env.vr2 = 0; /* No version specific id */ |
Richard Henderson | 9e3bab0 | 2019-08-25 17:33:53 -0700 | [diff] [blame] | 164 | cpu->env.avr = 0x01030000; /* Architecture v1.3 */ |
Richard Henderson | 8bebf7d | 2019-08-25 15:23:42 -0700 | [diff] [blame] | 165 | |
Richard Henderson | c7efab4 | 2019-08-25 15:02:54 -0700 | [diff] [blame] | 166 | cpu->env.upr = UPR_UP | UPR_DMP | UPR_IMP | UPR_PICP | UPR_TTP | UPR_PMP; |
Richard Henderson | fe636d3 | 2019-08-25 15:44:11 -0700 | [diff] [blame] | 167 | cpu->env.cpucfgr = CPUCFGR_NSGF | CPUCFGR_OB32S | CPUCFGR_OF32S | |
Richard Henderson | 62f2b03 | 2019-05-06 14:49:25 -0700 | [diff] [blame] | 168 | CPUCFGR_AVRP | CPUCFGR_EVBARP | CPUCFGR_OF64A32S; |
Richard Henderson | c7efab4 | 2019-08-25 15:02:54 -0700 | [diff] [blame] | 169 | |
| 170 | /* 1Way, TLB_SIZE entries. */ |
| 171 | cpu->env.dmmucfgr = (DMMUCFGR_NTW & (0 << 2)) |
| 172 | | (DMMUCFGR_NTS & (ctz32(TLB_SIZE) << 2)); |
| 173 | cpu->env.immucfgr = (IMMUCFGR_NTW & (0 << 2)) |
| 174 | | (IMMUCFGR_NTS & (ctz32(TLB_SIZE) << 2)); |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 175 | } |
| 176 | |
Claudio Fontana | 7827168 | 2021-02-04 17:39:23 +0100 | [diff] [blame^] | 177 | #include "hw/core/tcg-cpu-ops.h" |
| 178 | |
| 179 | static struct TCGCPUOps openrisc_tcg_ops = { |
| 180 | .initialize = openrisc_translate_init, |
| 181 | .cpu_exec_interrupt = openrisc_cpu_exec_interrupt, |
| 182 | .tlb_fill = openrisc_cpu_tlb_fill, |
| 183 | |
| 184 | #ifndef CONFIG_USER_ONLY |
| 185 | .do_interrupt = openrisc_cpu_do_interrupt, |
| 186 | #endif /* !CONFIG_USER_ONLY */ |
| 187 | }; |
| 188 | |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 189 | static void openrisc_cpu_class_init(ObjectClass *oc, void *data) |
| 190 | { |
| 191 | OpenRISCCPUClass *occ = OPENRISC_CPU_CLASS(oc); |
| 192 | CPUClass *cc = CPU_CLASS(occ); |
Andreas Färber | c296262 | 2013-01-05 14:11:07 +0100 | [diff] [blame] | 193 | DeviceClass *dc = DEVICE_CLASS(oc); |
| 194 | |
Philippe Mathieu-Daudé | bf85388 | 2018-01-13 23:04:12 -0300 | [diff] [blame] | 195 | device_class_set_parent_realize(dc, openrisc_cpu_realizefn, |
| 196 | &occ->parent_realize); |
Peter Maydell | 781c67c | 2020-03-03 10:05:11 +0000 | [diff] [blame] | 197 | device_class_set_parent_reset(dc, openrisc_cpu_reset, &occ->parent_reset); |
Andreas Färber | bd039ce | 2013-01-23 11:17:14 +0100 | [diff] [blame] | 198 | |
| 199 | cc->class_by_name = openrisc_cpu_class_by_name; |
Andreas Färber | 8c2e1b0 | 2013-08-25 18:53:55 +0200 | [diff] [blame] | 200 | cc->has_work = openrisc_cpu_has_work; |
Andreas Färber | 878096e | 2013-05-27 01:33:50 +0200 | [diff] [blame] | 201 | cc->dump_state = openrisc_cpu_dump_state; |
Andreas Färber | f45748f | 2013-06-21 19:09:18 +0200 | [diff] [blame] | 202 | cc->set_pc = openrisc_cpu_set_pc; |
Andreas Färber | 5b50e79 | 2013-06-29 04:18:45 +0200 | [diff] [blame] | 203 | cc->gdb_read_register = openrisc_cpu_gdb_read_register; |
| 204 | cc->gdb_write_register = openrisc_cpu_gdb_write_register; |
Richard Henderson | 35e911a | 2019-04-02 16:55:37 +0700 | [diff] [blame] | 205 | #ifndef CONFIG_USER_ONLY |
Andreas Färber | 00b941e | 2013-06-29 18:55:54 +0200 | [diff] [blame] | 206 | cc->get_phys_page_debug = openrisc_cpu_get_phys_page_debug; |
| 207 | dc->vmsd = &vmstate_openrisc_cpu; |
| 208 | #endif |
Andreas Färber | a0e372f | 2013-06-28 23:18:47 +0200 | [diff] [blame] | 209 | cc->gdb_num_core_regs = 32 + 3; |
Richard Henderson | d5cabcc | 2018-05-23 08:14:46 -0700 | [diff] [blame] | 210 | cc->disas_set_info = openrisc_disas_set_info; |
Claudio Fontana | 7827168 | 2021-02-04 17:39:23 +0100 | [diff] [blame^] | 211 | cc->tcg_ops = &openrisc_tcg_ops; |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 212 | } |
| 213 | |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 214 | /* Sort alphabetically by type name, except for "any". */ |
| 215 | static gint openrisc_cpu_list_compare(gconstpointer a, gconstpointer b) |
| 216 | { |
| 217 | ObjectClass *class_a = (ObjectClass *)a; |
| 218 | ObjectClass *class_b = (ObjectClass *)b; |
| 219 | const char *name_a, *name_b; |
| 220 | |
| 221 | name_a = object_class_get_name(class_a); |
| 222 | name_b = object_class_get_name(class_b); |
Andreas Färber | 478032a | 2013-01-27 22:50:35 +0100 | [diff] [blame] | 223 | if (strcmp(name_a, "any-" TYPE_OPENRISC_CPU) == 0) { |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 224 | return 1; |
Andreas Färber | 478032a | 2013-01-27 22:50:35 +0100 | [diff] [blame] | 225 | } else if (strcmp(name_b, "any-" TYPE_OPENRISC_CPU) == 0) { |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 226 | return -1; |
| 227 | } else { |
| 228 | return strcmp(name_a, name_b); |
| 229 | } |
| 230 | } |
| 231 | |
| 232 | static void openrisc_cpu_list_entry(gpointer data, gpointer user_data) |
| 233 | { |
| 234 | ObjectClass *oc = data; |
Andreas Färber | 478032a | 2013-01-27 22:50:35 +0100 | [diff] [blame] | 235 | const char *typename; |
| 236 | char *name; |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 237 | |
Andreas Färber | 478032a | 2013-01-27 22:50:35 +0100 | [diff] [blame] | 238 | typename = object_class_get_name(oc); |
| 239 | name = g_strndup(typename, |
| 240 | strlen(typename) - strlen("-" TYPE_OPENRISC_CPU)); |
Markus Armbruster | 0442428 | 2019-04-17 21:17:57 +0200 | [diff] [blame] | 241 | qemu_printf(" %s\n", name); |
Andreas Färber | 478032a | 2013-01-27 22:50:35 +0100 | [diff] [blame] | 242 | g_free(name); |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 243 | } |
| 244 | |
Markus Armbruster | 0442428 | 2019-04-17 21:17:57 +0200 | [diff] [blame] | 245 | void cpu_openrisc_list(void) |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 246 | { |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 247 | GSList *list; |
| 248 | |
| 249 | list = object_class_get_list(TYPE_OPENRISC_CPU, false); |
| 250 | list = g_slist_sort(list, openrisc_cpu_list_compare); |
Markus Armbruster | 0442428 | 2019-04-17 21:17:57 +0200 | [diff] [blame] | 251 | qemu_printf("Available CPUs:\n"); |
| 252 | g_slist_foreach(list, openrisc_cpu_list_entry, NULL); |
Jia Liu | e67db06 | 2012-07-20 15:50:39 +0800 | [diff] [blame] | 253 | g_slist_free(list); |
| 254 | } |
| 255 | |
Igor Mammedov | a677273 | 2017-10-05 15:50:51 +0200 | [diff] [blame] | 256 | #define DEFINE_OPENRISC_CPU_TYPE(cpu_model, initfn) \ |
| 257 | { \ |
| 258 | .parent = TYPE_OPENRISC_CPU, \ |
| 259 | .instance_init = initfn, \ |
| 260 | .name = OPENRISC_CPU_TYPE_NAME(cpu_model), \ |
| 261 | } |
| 262 | |
| 263 | static const TypeInfo openrisc_cpus_type_infos[] = { |
| 264 | { /* base class should be registered first */ |
| 265 | .name = TYPE_OPENRISC_CPU, |
| 266 | .parent = TYPE_CPU, |
| 267 | .instance_size = sizeof(OpenRISCCPU), |
| 268 | .instance_init = openrisc_cpu_initfn, |
| 269 | .abstract = true, |
| 270 | .class_size = sizeof(OpenRISCCPUClass), |
| 271 | .class_init = openrisc_cpu_class_init, |
| 272 | }, |
| 273 | DEFINE_OPENRISC_CPU_TYPE("or1200", or1200_initfn), |
| 274 | DEFINE_OPENRISC_CPU_TYPE("any", openrisc_any_initfn), |
| 275 | }; |
| 276 | |
| 277 | DEFINE_TYPES(openrisc_cpus_type_infos) |