xref: /openbmc/qemu/hw/rx/rx62n.c (revision db1015e92e04835c9eb50c29625fe566d1202dbd)
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