1 /* 2 * RX62N Microcontroller 3 * 4 * Datasheet: RX62N Group, RX621 Group User's Manual: Hardware 5 * (Rev.1.40 R01UH0033EJ0140) 6 * 7 * Copyright (c) 2019 Yoshinori Sato 8 * Copyright (c) 2020 Philippe Mathieu-Daudé 9 * 10 * This program is free software; you can redistribute it and/or modify it 11 * under the terms and conditions of the GNU General Public License, 12 * version 2 or later, as published by the Free Software Foundation. 13 * 14 * This program is distributed in the hope it will be useful, but WITHOUT 15 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 16 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for 17 * more details. 18 * 19 * You should have received a copy of the GNU General Public License along with 20 * this program. If not, see <http://www.gnu.org/licenses/>. 21 */ 22 23 #include "qemu/osdep.h" 24 #include "qapi/error.h" 25 #include "qemu/error-report.h" 26 #include "hw/hw.h" 27 #include "hw/rx/rx62n.h" 28 #include "hw/loader.h" 29 #include "hw/sysbus.h" 30 #include "hw/qdev-properties.h" 31 #include "sysemu/sysemu.h" 32 #include "sysemu/qtest.h" 33 #include "cpu.h" 34 #include "qom/object.h" 35 36 /* 37 * RX62N Internal Memory 38 */ 39 #define RX62N_IRAM_BASE 0x00000000 40 #define RX62N_DFLASH_BASE 0x00100000 41 #define RX62N_CFLASH_BASE 0xfff80000 42 43 /* 44 * RX62N Peripheral Address 45 * See users manual section 5 46 */ 47 #define RX62N_ICU_BASE 0x00087000 48 #define RX62N_TMR_BASE 0x00088200 49 #define RX62N_CMT_BASE 0x00088000 50 #define RX62N_SCI_BASE 0x00088240 51 52 /* 53 * RX62N Peripheral IRQ 54 * See users manual section 11 55 */ 56 #define RX62N_TMR_IRQ 174 57 #define RX62N_CMT_IRQ 28 58 #define RX62N_SCI_IRQ 214 59 60 #define RX62N_XTAL_MIN_HZ (8 * 1000 * 1000) 61 #define RX62N_XTAL_MAX_HZ (14 * 1000 * 1000) 62 #define RX62N_PCLK_MAX_HZ (50 * 1000 * 1000) 63 64 struct RX62NClass { 65 /*< private >*/ 66 DeviceClass parent_class; 67 /*< public >*/ 68 const char *name; 69 uint64_t ram_size; 70 uint64_t rom_flash_size; 71 uint64_t data_flash_size; 72 }; 73 typedef struct RX62NClass RX62NClass; 74 75 #define RX62N_MCU_CLASS(klass) \ 76 OBJECT_CLASS_CHECK(RX62NClass, (klass), TYPE_RX62N_MCU) 77 #define RX62N_MCU_GET_CLASS(obj) \ 78 OBJECT_GET_CLASS(RX62NClass, (obj), TYPE_RX62N_MCU) 79 80 /* 81 * IRQ -> IPR mapping table 82 * 0x00 - 0x91: IPR no (IPR00 to IPR91) 83 * 0xff: IPR not assigned 84 * See "11.3.1 Interrupt Vector Table" in hardware manual. 85 */ 86 static const uint8_t ipr_table[NR_IRQS] = { 87 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 88 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, /* 15 */ 89 0x00, 0xff, 0xff, 0xff, 0xff, 0x01, 0xff, 0x02, 90 0xff, 0xff, 0xff, 0x03, 0x04, 0x05, 0x06, 0x07, /* 31 */ 91 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f, 92 0x10, 0x11, 0x12, 0x13, 0x14, 0x14, 0x14, 0x14, /* 47 */ 93 0x15, 0x15, 0x15, 0x15, 0xff, 0xff, 0xff, 0xff, 94 0x18, 0x18, 0x18, 0x18, 0x18, 0x1d, 0x1e, 0x1f, /* 63 */ 95 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 96 0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f, /* 79 */ 97 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 98 0xff, 0xff, 0x3a, 0x3b, 0x3c, 0xff, 0xff, 0xff, /* 95 */ 99 0x40, 0xff, 0x44, 0x45, 0xff, 0xff, 0x48, 0xff, 100 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, /* 111 */ 101 0xff, 0xff, 0x51, 0x51, 0x51, 0x51, 0x52, 0x52, 102 0x52, 0x53, 0x53, 0x54, 0x54, 0x55, 0x55, 0x56, /* 127 */ 103 0x56, 0x57, 0x57, 0x57, 0x57, 0x58, 0x59, 0x59, 104 0x59, 0x59, 0x5a, 0x5b, 0x5b, 0x5b, 0x5c, 0x5c, /* 143 */ 105 0x5c, 0x5c, 0x5d, 0x5d, 0x5d, 0x5e, 0x5e, 0x5f, 106 0x5f, 0x60, 0x60, 0x61, 0x61, 0x62, 0x62, 0x62, /* 159 */ 107 0x62, 0x63, 0x64, 0x64, 0x64, 0x64, 0x65, 0x66, 108 0x66, 0x66, 0x67, 0x67, 0x67, 0x67, 0x68, 0x68, /* 175 */ 109 0x68, 0x69, 0x69, 0x69, 0x6a, 0x6a, 0x6a, 0x6b, 110 0x6b, 0x6b, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, /* 191 */ 111 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x70, 0x71, 112 0x72, 0x73, 0x74, 0x75, 0xff, 0xff, 0xff, 0xff, /* 207 */ 113 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x80, 0x80, 114 0x80, 0x80, 0x81, 0x81, 0x81, 0x81, 0x82, 0x82, /* 223 */ 115 0x82, 0x82, 0x83, 0x83, 0x83, 0x83, 0xff, 0xff, 116 0xff, 0xff, 0x85, 0x85, 0x85, 0x85, 0x86, 0x86, /* 239 */ 117 0x86, 0x86, 0xff, 0xff, 0xff, 0xff, 0x88, 0x89, 118 0x8a, 0x8b, 0x8c, 0x8d, 0x8e, 0x8f, 0x90, 0x91, /* 255 */ 119 }; 120 121 /* 122 * Level triggerd IRQ list 123 * Not listed IRQ is Edge trigger. 124 * See "11.3.1 Interrupt Vector Table" in hardware manual. 125 */ 126 static const uint8_t levelirq[] = { 127 16, 21, 32, 44, 47, 48, 51, 64, 65, 66, 128 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 129 77, 78, 79, 90, 91, 170, 171, 172, 173, 214, 130 217, 218, 221, 222, 225, 226, 229, 234, 237, 238, 131 241, 246, 249, 250, 253, 132 }; 133 134 static void register_icu(RX62NState *s) 135 { 136 int i; 137 SysBusDevice *icu; 138 139 object_initialize_child(OBJECT(s), "icu", &s->icu, TYPE_RX_ICU); 140 icu = SYS_BUS_DEVICE(&s->icu); 141 qdev_prop_set_uint32(DEVICE(icu), "len-ipr-map", NR_IRQS); 142 for (i = 0; i < NR_IRQS; i++) { 143 char propname[32]; 144 snprintf(propname, sizeof(propname), "ipr-map[%d]", i); 145 qdev_prop_set_uint32(DEVICE(icu), propname, ipr_table[i]); 146 } 147 qdev_prop_set_uint32(DEVICE(icu), "len-trigger-level", 148 ARRAY_SIZE(levelirq)); 149 for (i = 0; i < ARRAY_SIZE(levelirq); i++) { 150 char propname[32]; 151 snprintf(propname, sizeof(propname), "trigger-level[%d]", i); 152 qdev_prop_set_uint32(DEVICE(icu), propname, levelirq[i]); 153 } 154 155 for (i = 0; i < NR_IRQS; i++) { 156 s->irq[i] = qdev_get_gpio_in(DEVICE(icu), i); 157 } 158 sysbus_realize(icu, &error_abort); 159 sysbus_connect_irq(icu, 0, qdev_get_gpio_in(DEVICE(&s->cpu), RX_CPU_IRQ)); 160 sysbus_connect_irq(icu, 1, qdev_get_gpio_in(DEVICE(&s->cpu), RX_CPU_FIR)); 161 sysbus_connect_irq(icu, 2, s->irq[SWI]); 162 sysbus_mmio_map(SYS_BUS_DEVICE(icu), 0, RX62N_ICU_BASE); 163 } 164 165 static void register_tmr(RX62NState *s, int unit) 166 { 167 SysBusDevice *tmr; 168 int i, irqbase; 169 170 object_initialize_child(OBJECT(s), "tmr[*]", 171 &s->tmr[unit], TYPE_RENESAS_TMR); 172 tmr = SYS_BUS_DEVICE(&s->tmr[unit]); 173 qdev_prop_set_uint64(DEVICE(tmr), "input-freq", s->pclk_freq_hz); 174 sysbus_realize(tmr, &error_abort); 175 176 irqbase = RX62N_TMR_IRQ + TMR_NR_IRQ * unit; 177 for (i = 0; i < TMR_NR_IRQ; i++) { 178 sysbus_connect_irq(tmr, i, s->irq[irqbase + i]); 179 } 180 sysbus_mmio_map(tmr, 0, RX62N_TMR_BASE + unit * 0x10); 181 } 182 183 static void register_cmt(RX62NState *s, int unit) 184 { 185 SysBusDevice *cmt; 186 int i, irqbase; 187 188 object_initialize_child(OBJECT(s), "cmt[*]", 189 &s->cmt[unit], TYPE_RENESAS_CMT); 190 cmt = SYS_BUS_DEVICE(&s->cmt[unit]); 191 qdev_prop_set_uint64(DEVICE(cmt), "input-freq", s->pclk_freq_hz); 192 sysbus_realize(cmt, &error_abort); 193 194 irqbase = RX62N_CMT_IRQ + CMT_NR_IRQ * unit; 195 for (i = 0; i < CMT_NR_IRQ; i++) { 196 sysbus_connect_irq(cmt, i, s->irq[irqbase + i]); 197 } 198 sysbus_mmio_map(cmt, 0, RX62N_CMT_BASE + unit * 0x10); 199 } 200 201 static void register_sci(RX62NState *s, int unit) 202 { 203 SysBusDevice *sci; 204 int i, irqbase; 205 206 object_initialize_child(OBJECT(s), "sci[*]", 207 &s->sci[unit], TYPE_RENESAS_SCI); 208 sci = SYS_BUS_DEVICE(&s->sci[unit]); 209 qdev_prop_set_chr(DEVICE(sci), "chardev", serial_hd(unit)); 210 qdev_prop_set_uint64(DEVICE(sci), "input-freq", s->pclk_freq_hz); 211 sysbus_realize(sci, &error_abort); 212 213 irqbase = RX62N_SCI_IRQ + SCI_NR_IRQ * unit; 214 for (i = 0; i < SCI_NR_IRQ; i++) { 215 sysbus_connect_irq(sci, i, s->irq[irqbase + i]); 216 } 217 sysbus_mmio_map(sci, 0, RX62N_SCI_BASE + unit * 0x08); 218 } 219 220 static void rx62n_realize(DeviceState *dev, Error **errp) 221 { 222 RX62NState *s = RX62N_MCU(dev); 223 RX62NClass *rxc = RX62N_MCU_GET_CLASS(dev); 224 225 if (s->xtal_freq_hz == 0) { 226 error_setg(errp, "\"xtal-frequency-hz\" property must be provided."); 227 return; 228 } 229 /* XTAL range: 8-14 MHz */ 230 if (s->xtal_freq_hz < RX62N_XTAL_MIN_HZ 231 || s->xtal_freq_hz > RX62N_XTAL_MAX_HZ) { 232 error_setg(errp, "\"xtal-frequency-hz\" property in incorrect range."); 233 return; 234 } 235 /* Use a 4x fixed multiplier */ 236 s->pclk_freq_hz = 4 * s->xtal_freq_hz; 237 /* PCLK range: 8-50 MHz */ 238 assert(s->pclk_freq_hz <= RX62N_PCLK_MAX_HZ); 239 240 memory_region_init_ram(&s->iram, OBJECT(dev), "iram", 241 rxc->ram_size, &error_abort); 242 memory_region_add_subregion(s->sysmem, RX62N_IRAM_BASE, &s->iram); 243 memory_region_init_rom(&s->d_flash, OBJECT(dev), "flash-data", 244 rxc->data_flash_size, &error_abort); 245 memory_region_add_subregion(s->sysmem, RX62N_DFLASH_BASE, &s->d_flash); 246 memory_region_init_rom(&s->c_flash, OBJECT(dev), "flash-code", 247 rxc->rom_flash_size, &error_abort); 248 memory_region_add_subregion(s->sysmem, RX62N_CFLASH_BASE, &s->c_flash); 249 250 if (!s->kernel) { 251 if (bios_name) { 252 rom_add_file_fixed(bios_name, RX62N_CFLASH_BASE, 0); 253 } else if (!qtest_enabled()) { 254 error_report("No bios or kernel specified"); 255 exit(1); 256 } 257 } 258 259 /* Initialize CPU */ 260 object_initialize_child(OBJECT(s), "cpu", &s->cpu, TYPE_RX62N_CPU); 261 qdev_realize(DEVICE(&s->cpu), NULL, &error_abort); 262 263 register_icu(s); 264 s->cpu.env.ack = qdev_get_gpio_in_named(DEVICE(&s->icu), "ack", 0); 265 register_tmr(s, 0); 266 register_tmr(s, 1); 267 register_cmt(s, 0); 268 register_cmt(s, 1); 269 register_sci(s, 0); 270 } 271 272 static Property rx62n_properties[] = { 273 DEFINE_PROP_LINK("main-bus", RX62NState, sysmem, TYPE_MEMORY_REGION, 274 MemoryRegion *), 275 DEFINE_PROP_BOOL("load-kernel", RX62NState, kernel, false), 276 DEFINE_PROP_UINT32("xtal-frequency-hz", RX62NState, xtal_freq_hz, 0), 277 DEFINE_PROP_END_OF_LIST(), 278 }; 279 280 static void rx62n_class_init(ObjectClass *klass, void *data) 281 { 282 DeviceClass *dc = DEVICE_CLASS(klass); 283 284 dc->realize = rx62n_realize; 285 device_class_set_props(dc, rx62n_properties); 286 } 287 288 static void r5f562n7_class_init(ObjectClass *oc, void *data) 289 { 290 RX62NClass *rxc = RX62N_MCU_CLASS(oc); 291 292 rxc->ram_size = 64 * KiB; 293 rxc->rom_flash_size = 384 * KiB; 294 rxc->data_flash_size = 32 * KiB; 295 }; 296 297 static void r5f562n8_class_init(ObjectClass *oc, void *data) 298 { 299 RX62NClass *rxc = RX62N_MCU_CLASS(oc); 300 301 rxc->ram_size = 96 * KiB; 302 rxc->rom_flash_size = 512 * KiB; 303 rxc->data_flash_size = 32 * KiB; 304 }; 305 306 static const TypeInfo rx62n_types[] = { 307 { 308 .name = TYPE_R5F562N7_MCU, 309 .parent = TYPE_RX62N_MCU, 310 .class_init = r5f562n7_class_init, 311 }, { 312 .name = TYPE_R5F562N8_MCU, 313 .parent = TYPE_RX62N_MCU, 314 .class_init = r5f562n8_class_init, 315 }, { 316 .name = TYPE_RX62N_MCU, 317 .parent = TYPE_DEVICE, 318 .instance_size = sizeof(RX62NState), 319 .class_size = sizeof(RX62NClass), 320 .class_init = rx62n_class_init, 321 .abstract = true, 322 } 323 }; 324 325 DEFINE_TYPES(rx62n_types) 326