xref: /qemu/hw/avr/atmega.c (revision 06b40d250ecfa1633209c2e431a7a38acfd03a98)
1af55b781SPhilippe Mathieu-Daudé /*
2af55b781SPhilippe Mathieu-Daudé  * QEMU ATmega MCU
3af55b781SPhilippe Mathieu-Daudé  *
4af55b781SPhilippe Mathieu-Daudé  * Copyright (c) 2019-2020 Philippe Mathieu-Daudé
5af55b781SPhilippe Mathieu-Daudé  *
6af55b781SPhilippe Mathieu-Daudé  * This work is licensed under the terms of the GNU GPLv2 or later.
7af55b781SPhilippe Mathieu-Daudé  * See the COPYING file in the top-level directory.
8af55b781SPhilippe Mathieu-Daudé  * SPDX-License-Identifier: GPL-2.0-or-later
9af55b781SPhilippe Mathieu-Daudé  */
10af55b781SPhilippe Mathieu-Daudé 
11af55b781SPhilippe Mathieu-Daudé #include "qemu/osdep.h"
12af55b781SPhilippe Mathieu-Daudé #include "qemu/module.h"
13af55b781SPhilippe Mathieu-Daudé #include "qemu/units.h"
14af55b781SPhilippe Mathieu-Daudé #include "qapi/error.h"
159c2ff9cdSPierrick Bouvier #include "exec/target_page.h"
168be545baSRichard Henderson #include "system/memory.h"
17dfc56946SRichard Henderson #include "system/address-spaces.h"
1832cad1ffSPhilippe Mathieu-Daudé #include "system/system.h"
19af55b781SPhilippe Mathieu-Daudé #include "hw/qdev-properties.h"
20af55b781SPhilippe Mathieu-Daudé #include "hw/sysbus.h"
21db1015e9SEduardo Habkost #include "qom/object.h"
22af55b781SPhilippe Mathieu-Daudé #include "hw/misc/unimp.h"
239f99072fSRichard Henderson #include "migration/vmstate.h"
24af55b781SPhilippe Mathieu-Daudé #include "atmega.h"
25af55b781SPhilippe Mathieu-Daudé 
26af55b781SPhilippe Mathieu-Daudé enum AtmegaPeripheral {
27af55b781SPhilippe Mathieu-Daudé     POWER0, POWER1,
28af55b781SPhilippe Mathieu-Daudé     GPIOA, GPIOB, GPIOC, GPIOD, GPIOE, GPIOF,
29af55b781SPhilippe Mathieu-Daudé     GPIOG, GPIOH, GPIOI, GPIOJ, GPIOK, GPIOL,
30af55b781SPhilippe Mathieu-Daudé     USART0, USART1, USART2, USART3,
31af55b781SPhilippe Mathieu-Daudé     TIMER0, TIMER1, TIMER2, TIMER3, TIMER4, TIMER5,
32af55b781SPhilippe Mathieu-Daudé     PERIFMAX
33af55b781SPhilippe Mathieu-Daudé };
34af55b781SPhilippe Mathieu-Daudé 
35af55b781SPhilippe Mathieu-Daudé #define GPIO(n)     (n + GPIOA)
36af55b781SPhilippe Mathieu-Daudé #define USART(n)    (n + USART0)
37af55b781SPhilippe Mathieu-Daudé #define TIMER(n)    (n + TIMER0)
38af55b781SPhilippe Mathieu-Daudé #define POWER(n)    (n + POWER0)
39af55b781SPhilippe Mathieu-Daudé 
40af55b781SPhilippe Mathieu-Daudé typedef struct {
41af55b781SPhilippe Mathieu-Daudé     uint16_t addr;
42af55b781SPhilippe Mathieu-Daudé     enum AtmegaPeripheral power_index;
43af55b781SPhilippe Mathieu-Daudé     uint8_t power_bit;
44af55b781SPhilippe Mathieu-Daudé     /* timer specific */
45af55b781SPhilippe Mathieu-Daudé     uint16_t intmask_addr;
46af55b781SPhilippe Mathieu-Daudé     uint16_t intflag_addr;
47af55b781SPhilippe Mathieu-Daudé     bool is_timer16;
48af55b781SPhilippe Mathieu-Daudé } peripheral_cfg;
49af55b781SPhilippe Mathieu-Daudé 
50db1015e9SEduardo Habkost struct AtmegaMcuClass {
51af55b781SPhilippe Mathieu-Daudé     /*< private >*/
52af55b781SPhilippe Mathieu-Daudé     SysBusDeviceClass parent_class;
53af55b781SPhilippe Mathieu-Daudé     /*< public >*/
54af55b781SPhilippe Mathieu-Daudé     const char *uc_name;
55af55b781SPhilippe Mathieu-Daudé     const char *cpu_type;
56af55b781SPhilippe Mathieu-Daudé     size_t flash_size;
57af55b781SPhilippe Mathieu-Daudé     size_t eeprom_size;
58af55b781SPhilippe Mathieu-Daudé     size_t sram_size;
59af55b781SPhilippe Mathieu-Daudé     size_t io_size;
60af55b781SPhilippe Mathieu-Daudé     size_t gpio_count;
61af55b781SPhilippe Mathieu-Daudé     size_t adc_count;
62af55b781SPhilippe Mathieu-Daudé     const uint8_t *irq;
63af55b781SPhilippe Mathieu-Daudé     const peripheral_cfg *dev;
64db1015e9SEduardo Habkost };
65db1015e9SEduardo Habkost typedef struct AtmegaMcuClass AtmegaMcuClass;
66af55b781SPhilippe Mathieu-Daudé 
678110fa1dSEduardo Habkost DECLARE_CLASS_CHECKERS(AtmegaMcuClass, ATMEGA_MCU,
688110fa1dSEduardo Habkost                        TYPE_ATMEGA_MCU)
69af55b781SPhilippe Mathieu-Daudé 
70af55b781SPhilippe Mathieu-Daudé static const peripheral_cfg dev168_328[PERIFMAX] = {
71af55b781SPhilippe Mathieu-Daudé     [USART0]        = {  0xc0, POWER0, 1 },
72af55b781SPhilippe Mathieu-Daudé     [TIMER2]        = {  0xb0, POWER0, 6, 0x70, 0x37, false },
73af55b781SPhilippe Mathieu-Daudé     [TIMER1]        = {  0x80, POWER0, 3, 0x6f, 0x36, true },
74af55b781SPhilippe Mathieu-Daudé     [POWER0]        = {  0x64 },
75af55b781SPhilippe Mathieu-Daudé     [TIMER0]        = {  0x44, POWER0, 5, 0x6e, 0x35, false },
76af55b781SPhilippe Mathieu-Daudé     [GPIOD]         = {  0x29 },
77af55b781SPhilippe Mathieu-Daudé     [GPIOC]         = {  0x26 },
78af55b781SPhilippe Mathieu-Daudé     [GPIOB]         = {  0x23 },
79af55b781SPhilippe Mathieu-Daudé }, dev1280_2560[PERIFMAX] = {
80af55b781SPhilippe Mathieu-Daudé     [USART3]        = { 0x130, POWER1, 2 },
81af55b781SPhilippe Mathieu-Daudé     [TIMER5]        = { 0x120, POWER1, 5, 0x73, 0x3a, true },
82af55b781SPhilippe Mathieu-Daudé     [GPIOL]         = { 0x109 },
83af55b781SPhilippe Mathieu-Daudé     [GPIOK]         = { 0x106 },
84af55b781SPhilippe Mathieu-Daudé     [GPIOJ]         = { 0x103 },
85af55b781SPhilippe Mathieu-Daudé     [GPIOH]         = { 0x100 },
86af55b781SPhilippe Mathieu-Daudé     [USART2]        = {  0xd0, POWER1, 1 },
87af55b781SPhilippe Mathieu-Daudé     [USART1]        = {  0xc8, POWER1, 0 },
88af55b781SPhilippe Mathieu-Daudé     [USART0]        = {  0xc0, POWER0, 1 },
89af55b781SPhilippe Mathieu-Daudé     [TIMER2]        = {  0xb0, POWER0, 6, 0x70, 0x37, false }, /* TODO async */
90af55b781SPhilippe Mathieu-Daudé     [TIMER4]        = {  0xa0, POWER1, 4, 0x72, 0x39, true },
91af55b781SPhilippe Mathieu-Daudé     [TIMER3]        = {  0x90, POWER1, 3, 0x71, 0x38, true },
92af55b781SPhilippe Mathieu-Daudé     [TIMER1]        = {  0x80, POWER0, 3, 0x6f, 0x36, true },
93af55b781SPhilippe Mathieu-Daudé     [POWER1]        = {  0x65 },
94af55b781SPhilippe Mathieu-Daudé     [POWER0]        = {  0x64 },
95af55b781SPhilippe Mathieu-Daudé     [TIMER0]        = {  0x44, POWER0, 5, 0x6e, 0x35, false },
96af55b781SPhilippe Mathieu-Daudé     [GPIOG]         = {  0x32 },
97af55b781SPhilippe Mathieu-Daudé     [GPIOF]         = {  0x2f },
98af55b781SPhilippe Mathieu-Daudé     [GPIOE]         = {  0x2c },
99af55b781SPhilippe Mathieu-Daudé     [GPIOD]         = {  0x29 },
100af55b781SPhilippe Mathieu-Daudé     [GPIOC]         = {  0x26 },
101af55b781SPhilippe Mathieu-Daudé     [GPIOB]         = {  0x23 },
102af55b781SPhilippe Mathieu-Daudé     [GPIOA]         = {  0x20 },
103af55b781SPhilippe Mathieu-Daudé };
104af55b781SPhilippe Mathieu-Daudé 
105af55b781SPhilippe Mathieu-Daudé enum AtmegaIrq {
106af55b781SPhilippe Mathieu-Daudé     USART0_RXC_IRQ, USART0_DRE_IRQ, USART0_TXC_IRQ,
107af55b781SPhilippe Mathieu-Daudé     USART1_RXC_IRQ, USART1_DRE_IRQ, USART1_TXC_IRQ,
108af55b781SPhilippe Mathieu-Daudé     USART2_RXC_IRQ, USART2_DRE_IRQ, USART2_TXC_IRQ,
109af55b781SPhilippe Mathieu-Daudé     USART3_RXC_IRQ, USART3_DRE_IRQ, USART3_TXC_IRQ,
110af55b781SPhilippe Mathieu-Daudé     TIMER0_CAPT_IRQ, TIMER0_COMPA_IRQ, TIMER0_COMPB_IRQ,
111af55b781SPhilippe Mathieu-Daudé         TIMER0_COMPC_IRQ, TIMER0_OVF_IRQ,
112af55b781SPhilippe Mathieu-Daudé     TIMER1_CAPT_IRQ, TIMER1_COMPA_IRQ, TIMER1_COMPB_IRQ,
113af55b781SPhilippe Mathieu-Daudé         TIMER1_COMPC_IRQ, TIMER1_OVF_IRQ,
114af55b781SPhilippe Mathieu-Daudé     TIMER2_CAPT_IRQ, TIMER2_COMPA_IRQ, TIMER2_COMPB_IRQ,
115af55b781SPhilippe Mathieu-Daudé         TIMER2_COMPC_IRQ, TIMER2_OVF_IRQ,
116af55b781SPhilippe Mathieu-Daudé     TIMER3_CAPT_IRQ, TIMER3_COMPA_IRQ, TIMER3_COMPB_IRQ,
117af55b781SPhilippe Mathieu-Daudé         TIMER3_COMPC_IRQ, TIMER3_OVF_IRQ,
118af55b781SPhilippe Mathieu-Daudé     TIMER4_CAPT_IRQ, TIMER4_COMPA_IRQ, TIMER4_COMPB_IRQ,
119af55b781SPhilippe Mathieu-Daudé         TIMER4_COMPC_IRQ, TIMER4_OVF_IRQ,
120af55b781SPhilippe Mathieu-Daudé     TIMER5_CAPT_IRQ, TIMER5_COMPA_IRQ, TIMER5_COMPB_IRQ,
121af55b781SPhilippe Mathieu-Daudé         TIMER5_COMPC_IRQ, TIMER5_OVF_IRQ,
122af55b781SPhilippe Mathieu-Daudé     IRQ_COUNT
123af55b781SPhilippe Mathieu-Daudé };
124af55b781SPhilippe Mathieu-Daudé 
125af55b781SPhilippe Mathieu-Daudé #define USART_IRQ_COUNT     3
126af55b781SPhilippe Mathieu-Daudé #define USART_RXC_IRQ(n)    (n * USART_IRQ_COUNT + USART0_RXC_IRQ)
127af55b781SPhilippe Mathieu-Daudé #define USART_DRE_IRQ(n)    (n * USART_IRQ_COUNT + USART0_DRE_IRQ)
128af55b781SPhilippe Mathieu-Daudé #define USART_TXC_IRQ(n)    (n * USART_IRQ_COUNT + USART0_TXC_IRQ)
129af55b781SPhilippe Mathieu-Daudé #define TIMER_IRQ_COUNT     5
130af55b781SPhilippe Mathieu-Daudé #define TIMER_CAPT_IRQ(n)   (n * TIMER_IRQ_COUNT + TIMER0_CAPT_IRQ)
131af55b781SPhilippe Mathieu-Daudé #define TIMER_COMPA_IRQ(n)  (n * TIMER_IRQ_COUNT + TIMER0_COMPA_IRQ)
132af55b781SPhilippe Mathieu-Daudé #define TIMER_COMPB_IRQ(n)  (n * TIMER_IRQ_COUNT + TIMER0_COMPB_IRQ)
133af55b781SPhilippe Mathieu-Daudé #define TIMER_COMPC_IRQ(n)  (n * TIMER_IRQ_COUNT + TIMER0_COMPC_IRQ)
134af55b781SPhilippe Mathieu-Daudé #define TIMER_OVF_IRQ(n)    (n * TIMER_IRQ_COUNT + TIMER0_OVF_IRQ)
135af55b781SPhilippe Mathieu-Daudé 
136af55b781SPhilippe Mathieu-Daudé static const uint8_t irq168_328[IRQ_COUNT] = {
137af55b781SPhilippe Mathieu-Daudé     [TIMER2_COMPA_IRQ]      = 8,
138af55b781SPhilippe Mathieu-Daudé     [TIMER2_COMPB_IRQ]      = 9,
139af55b781SPhilippe Mathieu-Daudé     [TIMER2_OVF_IRQ]        = 10,
140af55b781SPhilippe Mathieu-Daudé     [TIMER1_CAPT_IRQ]       = 11,
141af55b781SPhilippe Mathieu-Daudé     [TIMER1_COMPA_IRQ]      = 12,
142af55b781SPhilippe Mathieu-Daudé     [TIMER1_COMPB_IRQ]      = 13,
143af55b781SPhilippe Mathieu-Daudé     [TIMER1_OVF_IRQ]        = 14,
144af55b781SPhilippe Mathieu-Daudé     [TIMER0_COMPA_IRQ]      = 15,
145af55b781SPhilippe Mathieu-Daudé     [TIMER0_COMPB_IRQ]      = 16,
146af55b781SPhilippe Mathieu-Daudé     [TIMER0_OVF_IRQ]        = 17,
147af55b781SPhilippe Mathieu-Daudé     [USART0_RXC_IRQ]        = 19,
148af55b781SPhilippe Mathieu-Daudé     [USART0_DRE_IRQ]        = 20,
149af55b781SPhilippe Mathieu-Daudé     [USART0_TXC_IRQ]        = 21,
150af55b781SPhilippe Mathieu-Daudé }, irq1280_2560[IRQ_COUNT] = {
151af55b781SPhilippe Mathieu-Daudé     [TIMER2_COMPA_IRQ]      = 14,
152af55b781SPhilippe Mathieu-Daudé     [TIMER2_COMPB_IRQ]      = 15,
153af55b781SPhilippe Mathieu-Daudé     [TIMER2_OVF_IRQ]        = 16,
154af55b781SPhilippe Mathieu-Daudé     [TIMER1_CAPT_IRQ]       = 17,
155af55b781SPhilippe Mathieu-Daudé     [TIMER1_COMPA_IRQ]      = 18,
156af55b781SPhilippe Mathieu-Daudé     [TIMER1_COMPB_IRQ]      = 19,
157af55b781SPhilippe Mathieu-Daudé     [TIMER1_COMPC_IRQ]      = 20,
158af55b781SPhilippe Mathieu-Daudé     [TIMER1_OVF_IRQ]        = 21,
159af55b781SPhilippe Mathieu-Daudé     [TIMER0_COMPA_IRQ]      = 22,
160af55b781SPhilippe Mathieu-Daudé     [TIMER0_COMPB_IRQ]      = 23,
161af55b781SPhilippe Mathieu-Daudé     [TIMER0_OVF_IRQ]        = 24,
162af55b781SPhilippe Mathieu-Daudé     [USART0_RXC_IRQ]        = 26,
163af55b781SPhilippe Mathieu-Daudé     [USART0_DRE_IRQ]        = 27,
164af55b781SPhilippe Mathieu-Daudé     [USART0_TXC_IRQ]        = 28,
165af55b781SPhilippe Mathieu-Daudé     [TIMER3_CAPT_IRQ]       = 32,
166af55b781SPhilippe Mathieu-Daudé     [TIMER3_COMPA_IRQ]      = 33,
167af55b781SPhilippe Mathieu-Daudé     [TIMER3_COMPB_IRQ]      = 34,
168af55b781SPhilippe Mathieu-Daudé     [TIMER3_COMPC_IRQ]      = 35,
169af55b781SPhilippe Mathieu-Daudé     [TIMER3_OVF_IRQ]        = 36,
170af55b781SPhilippe Mathieu-Daudé     [USART1_RXC_IRQ]        = 37,
171af55b781SPhilippe Mathieu-Daudé     [USART1_DRE_IRQ]        = 38,
172af55b781SPhilippe Mathieu-Daudé     [USART1_TXC_IRQ]        = 39,
173af55b781SPhilippe Mathieu-Daudé     [TIMER4_CAPT_IRQ]       = 42,
174af55b781SPhilippe Mathieu-Daudé     [TIMER4_COMPA_IRQ]      = 43,
175af55b781SPhilippe Mathieu-Daudé     [TIMER4_COMPB_IRQ]      = 44,
176af55b781SPhilippe Mathieu-Daudé     [TIMER4_COMPC_IRQ]      = 45,
177af55b781SPhilippe Mathieu-Daudé     [TIMER4_OVF_IRQ]        = 46,
178af55b781SPhilippe Mathieu-Daudé     [TIMER5_CAPT_IRQ]       = 47,
179af55b781SPhilippe Mathieu-Daudé     [TIMER5_COMPA_IRQ]      = 48,
180af55b781SPhilippe Mathieu-Daudé     [TIMER5_COMPB_IRQ]      = 49,
181af55b781SPhilippe Mathieu-Daudé     [TIMER5_COMPC_IRQ]      = 50,
182af55b781SPhilippe Mathieu-Daudé     [TIMER5_OVF_IRQ]        = 51,
183af55b781SPhilippe Mathieu-Daudé     [USART2_RXC_IRQ]        = 52,
184af55b781SPhilippe Mathieu-Daudé     [USART2_DRE_IRQ]        = 53,
185af55b781SPhilippe Mathieu-Daudé     [USART2_TXC_IRQ]        = 54,
186af55b781SPhilippe Mathieu-Daudé     [USART3_RXC_IRQ]        = 55,
187af55b781SPhilippe Mathieu-Daudé     [USART3_DRE_IRQ]        = 56,
188af55b781SPhilippe Mathieu-Daudé     [USART3_TXC_IRQ]        = 57,
189af55b781SPhilippe Mathieu-Daudé };
190af55b781SPhilippe Mathieu-Daudé 
connect_peripheral_irq(const AtmegaMcuClass * k,SysBusDevice * dev,int dev_irqn,DeviceState * cpu,unsigned peripheral_index)191af55b781SPhilippe Mathieu-Daudé static void connect_peripheral_irq(const AtmegaMcuClass *k,
192af55b781SPhilippe Mathieu-Daudé                                    SysBusDevice *dev, int dev_irqn,
193af55b781SPhilippe Mathieu-Daudé                                    DeviceState *cpu,
194af55b781SPhilippe Mathieu-Daudé                                    unsigned peripheral_index)
195af55b781SPhilippe Mathieu-Daudé {
196af55b781SPhilippe Mathieu-Daudé     int cpu_irq = k->irq[peripheral_index];
197af55b781SPhilippe Mathieu-Daudé 
198af55b781SPhilippe Mathieu-Daudé     if (!cpu_irq) {
199af55b781SPhilippe Mathieu-Daudé         return;
200af55b781SPhilippe Mathieu-Daudé     }
201af55b781SPhilippe Mathieu-Daudé     /* FIXME move that to avr_cpu_set_int() once 'sample' board is removed */
202af55b781SPhilippe Mathieu-Daudé     assert(cpu_irq >= 2);
203af55b781SPhilippe Mathieu-Daudé     cpu_irq -= 2;
204af55b781SPhilippe Mathieu-Daudé 
205af55b781SPhilippe Mathieu-Daudé     sysbus_connect_irq(dev, dev_irqn, qdev_get_gpio_in(cpu, cpu_irq));
206af55b781SPhilippe Mathieu-Daudé }
207af55b781SPhilippe Mathieu-Daudé 
connect_power_reduction_gpio(AtmegaMcuState * s,const AtmegaMcuClass * k,DeviceState * cpu,unsigned peripheral_index)208af55b781SPhilippe Mathieu-Daudé static void connect_power_reduction_gpio(AtmegaMcuState *s,
209af55b781SPhilippe Mathieu-Daudé                                          const AtmegaMcuClass *k,
210af55b781SPhilippe Mathieu-Daudé                                          DeviceState *cpu,
211af55b781SPhilippe Mathieu-Daudé                                          unsigned peripheral_index)
212af55b781SPhilippe Mathieu-Daudé {
213af55b781SPhilippe Mathieu-Daudé     unsigned power_index = k->dev[peripheral_index].power_index;
214af55b781SPhilippe Mathieu-Daudé     assert(k->dev[power_index].addr);
215af55b781SPhilippe Mathieu-Daudé     sysbus_connect_irq(SYS_BUS_DEVICE(&s->pwr[power_index - POWER0]),
216af55b781SPhilippe Mathieu-Daudé                        k->dev[peripheral_index].power_bit,
217af55b781SPhilippe Mathieu-Daudé                        qdev_get_gpio_in(cpu, 0));
218af55b781SPhilippe Mathieu-Daudé }
219af55b781SPhilippe Mathieu-Daudé 
atmega_realize(DeviceState * dev,Error ** errp)220af55b781SPhilippe Mathieu-Daudé static void atmega_realize(DeviceState *dev, Error **errp)
221af55b781SPhilippe Mathieu-Daudé {
222af55b781SPhilippe Mathieu-Daudé     AtmegaMcuState *s = ATMEGA_MCU(dev);
223af55b781SPhilippe Mathieu-Daudé     const AtmegaMcuClass *mc = ATMEGA_MCU_GET_CLASS(dev);
224af55b781SPhilippe Mathieu-Daudé     DeviceState *cpudev;
225af55b781SPhilippe Mathieu-Daudé     SysBusDevice *sbd;
226af55b781SPhilippe Mathieu-Daudé     char *devname;
227af55b781SPhilippe Mathieu-Daudé     size_t i;
228af55b781SPhilippe Mathieu-Daudé 
229af55b781SPhilippe Mathieu-Daudé     if (!s->xtal_freq_hz) {
230af55b781SPhilippe Mathieu-Daudé         error_setg(errp, "\"xtal-frequency-hz\" property must be provided.");
231af55b781SPhilippe Mathieu-Daudé         return;
232af55b781SPhilippe Mathieu-Daudé     }
233af55b781SPhilippe Mathieu-Daudé 
234af55b781SPhilippe Mathieu-Daudé     /* CPU */
235af55b781SPhilippe Mathieu-Daudé     object_initialize_child(OBJECT(dev), "cpu", &s->cpu, mc->cpu_type);
236235948bfSGihun Nam 
237235948bfSGihun Nam     object_property_set_uint(OBJECT(&s->cpu), "init-sp",
238235948bfSGihun Nam                              mc->io_size + mc->sram_size - 1, &error_abort);
239235948bfSGihun Nam 
24074fb2f4fSPhilippe Mathieu-Daudé     qdev_realize(DEVICE(&s->cpu), NULL, &error_abort);
241af55b781SPhilippe Mathieu-Daudé     cpudev = DEVICE(&s->cpu);
242af55b781SPhilippe Mathieu-Daudé 
2439f99072fSRichard Henderson     /*
2449f99072fSRichard Henderson      * SRAM
2459f99072fSRichard Henderson      *
2469f99072fSRichard Henderson      * Softmmu is not able mix i/o and ram on the same page.
2479f99072fSRichard Henderson      * Therefore in all cases, the first page exclusively contains i/o.
2489f99072fSRichard Henderson      *
2499f99072fSRichard Henderson      * If the MCU's i/o region matches the page size, then we can simply
2509f99072fSRichard Henderson      * allocate all ram starting at the second page.  Otherwise, we must
2519f99072fSRichard Henderson      * allocate some ram as i/o to complete the first page.
2529f99072fSRichard Henderson      */
2539f99072fSRichard Henderson     assert(mc->io_size == 0x100 || mc->io_size == 0x200);
2549f99072fSRichard Henderson     if (mc->io_size >= TARGET_PAGE_SIZE) {
255af55b781SPhilippe Mathieu-Daudé         memory_region_init_ram(&s->sram, OBJECT(dev), "sram", mc->sram_size,
256af55b781SPhilippe Mathieu-Daudé                                &error_abort);
257af55b781SPhilippe Mathieu-Daudé         memory_region_add_subregion(get_system_memory(),
258af55b781SPhilippe Mathieu-Daudé                                     OFFSET_DATA + mc->io_size, &s->sram);
2599f99072fSRichard Henderson     } else {
2609f99072fSRichard Henderson         int sram_io_size = TARGET_PAGE_SIZE - mc->io_size;
2619f99072fSRichard Henderson         void *sram_io_mem = g_malloc0(sram_io_size);
2629f99072fSRichard Henderson 
2639f99072fSRichard Henderson         memory_region_init_ram_device_ptr(&s->sram_io, OBJECT(dev), "sram-as-io",
2649f99072fSRichard Henderson                                           sram_io_size, sram_io_mem);
2659f99072fSRichard Henderson         memory_region_add_subregion(get_system_memory(),
2669f99072fSRichard Henderson                                     OFFSET_DATA + mc->io_size, &s->sram_io);
2679f99072fSRichard Henderson         vmstate_register_ram(&s->sram_io, dev);
2689f99072fSRichard Henderson 
2699f99072fSRichard Henderson         memory_region_init_ram(&s->sram, OBJECT(dev), "sram",
2709f99072fSRichard Henderson                                mc->sram_size - sram_io_size, &error_abort);
2719f99072fSRichard Henderson         memory_region_add_subregion(get_system_memory(),
2729f99072fSRichard Henderson                                     OFFSET_DATA + TARGET_PAGE_SIZE, &s->sram);
2739f99072fSRichard Henderson     }
274af55b781SPhilippe Mathieu-Daudé 
275af55b781SPhilippe Mathieu-Daudé     /* Flash */
276af55b781SPhilippe Mathieu-Daudé     memory_region_init_rom(&s->flash, OBJECT(dev),
277af55b781SPhilippe Mathieu-Daudé                            "flash", mc->flash_size, &error_fatal);
278af55b781SPhilippe Mathieu-Daudé     memory_region_add_subregion(get_system_memory(), OFFSET_CODE, &s->flash);
279af55b781SPhilippe Mathieu-Daudé 
280af55b781SPhilippe Mathieu-Daudé     /*
281af55b781SPhilippe Mathieu-Daudé      * I/O
282af55b781SPhilippe Mathieu-Daudé      *
283af55b781SPhilippe Mathieu-Daudé      * 0x00 - 0x1f: Registers
284af55b781SPhilippe Mathieu-Daudé      * 0x20 - 0x5f: I/O memory
285af55b781SPhilippe Mathieu-Daudé      * 0x60 - 0xff: Extended I/O
286af55b781SPhilippe Mathieu-Daudé      */
287af55b781SPhilippe Mathieu-Daudé     s->io = qdev_new(TYPE_UNIMPLEMENTED_DEVICE);
288af55b781SPhilippe Mathieu-Daudé     qdev_prop_set_string(s->io, "name", "I/O");
289af55b781SPhilippe Mathieu-Daudé     qdev_prop_set_uint64(s->io, "size", mc->io_size);
290af55b781SPhilippe Mathieu-Daudé     sysbus_realize_and_unref(SYS_BUS_DEVICE(s->io), &error_fatal);
291af55b781SPhilippe Mathieu-Daudé     sysbus_mmio_map_overlap(SYS_BUS_DEVICE(s->io), 0, OFFSET_DATA, -1234);
292af55b781SPhilippe Mathieu-Daudé 
293af55b781SPhilippe Mathieu-Daudé     /* Power Reduction */
294af55b781SPhilippe Mathieu-Daudé     for (i = 0; i < POWER_MAX; i++) {
295af55b781SPhilippe Mathieu-Daudé         int idx = POWER(i);
296af55b781SPhilippe Mathieu-Daudé         if (!mc->dev[idx].addr) {
297af55b781SPhilippe Mathieu-Daudé             continue;
298af55b781SPhilippe Mathieu-Daudé         }
299af55b781SPhilippe Mathieu-Daudé         devname = g_strdup_printf("power%zu", i);
300af55b781SPhilippe Mathieu-Daudé         object_initialize_child(OBJECT(dev), devname, &s->pwr[i],
301af55b781SPhilippe Mathieu-Daudé                                 TYPE_AVR_MASK);
302af55b781SPhilippe Mathieu-Daudé         sysbus_realize(SYS_BUS_DEVICE(&s->pwr[i]), &error_abort);
303af55b781SPhilippe Mathieu-Daudé         sysbus_mmio_map(SYS_BUS_DEVICE(&s->pwr[i]), 0,
304af55b781SPhilippe Mathieu-Daudé                         OFFSET_DATA + mc->dev[idx].addr);
305af55b781SPhilippe Mathieu-Daudé         g_free(devname);
306af55b781SPhilippe Mathieu-Daudé     }
307af55b781SPhilippe Mathieu-Daudé 
308af55b781SPhilippe Mathieu-Daudé     /* GPIO */
309af55b781SPhilippe Mathieu-Daudé     for (i = 0; i < GPIO_MAX; i++) {
310af55b781SPhilippe Mathieu-Daudé         int idx = GPIO(i);
311af55b781SPhilippe Mathieu-Daudé         if (!mc->dev[idx].addr) {
312af55b781SPhilippe Mathieu-Daudé             continue;
313af55b781SPhilippe Mathieu-Daudé         }
314af55b781SPhilippe Mathieu-Daudé         devname = g_strdup_printf("atmega-gpio-%c", 'a' + (char)i);
315af55b781SPhilippe Mathieu-Daudé         create_unimplemented_device(devname,
316af55b781SPhilippe Mathieu-Daudé                                     OFFSET_DATA + mc->dev[idx].addr, 3);
317af55b781SPhilippe Mathieu-Daudé         g_free(devname);
318af55b781SPhilippe Mathieu-Daudé     }
319af55b781SPhilippe Mathieu-Daudé 
320af55b781SPhilippe Mathieu-Daudé     /* USART */
321af55b781SPhilippe Mathieu-Daudé     for (i = 0; i < USART_MAX; i++) {
322af55b781SPhilippe Mathieu-Daudé         int idx = USART(i);
323af55b781SPhilippe Mathieu-Daudé         if (!mc->dev[idx].addr) {
324af55b781SPhilippe Mathieu-Daudé             continue;
325af55b781SPhilippe Mathieu-Daudé         }
326af55b781SPhilippe Mathieu-Daudé         devname = g_strdup_printf("usart%zu", i);
327af55b781SPhilippe Mathieu-Daudé         object_initialize_child(OBJECT(dev), devname, &s->usart[i],
328af55b781SPhilippe Mathieu-Daudé                                 TYPE_AVR_USART);
329af55b781SPhilippe Mathieu-Daudé         qdev_prop_set_chr(DEVICE(&s->usart[i]), "chardev", serial_hd(i));
330af55b781SPhilippe Mathieu-Daudé         sbd = SYS_BUS_DEVICE(&s->usart[i]);
331af55b781SPhilippe Mathieu-Daudé         sysbus_realize(sbd, &error_abort);
332af55b781SPhilippe Mathieu-Daudé         sysbus_mmio_map(sbd, 0, OFFSET_DATA + mc->dev[USART(i)].addr);
333af55b781SPhilippe Mathieu-Daudé         connect_peripheral_irq(mc, sbd, 0, cpudev, USART_RXC_IRQ(i));
334af55b781SPhilippe Mathieu-Daudé         connect_peripheral_irq(mc, sbd, 1, cpudev, USART_DRE_IRQ(i));
335af55b781SPhilippe Mathieu-Daudé         connect_peripheral_irq(mc, sbd, 2, cpudev, USART_TXC_IRQ(i));
336af55b781SPhilippe Mathieu-Daudé         connect_power_reduction_gpio(s, mc, DEVICE(&s->usart[i]), idx);
337af55b781SPhilippe Mathieu-Daudé         g_free(devname);
338af55b781SPhilippe Mathieu-Daudé     }
339af55b781SPhilippe Mathieu-Daudé 
340af55b781SPhilippe Mathieu-Daudé     /* Timer */
341af55b781SPhilippe Mathieu-Daudé     for (i = 0; i < TIMER_MAX; i++) {
342af55b781SPhilippe Mathieu-Daudé         int idx = TIMER(i);
343af55b781SPhilippe Mathieu-Daudé         if (!mc->dev[idx].addr) {
344af55b781SPhilippe Mathieu-Daudé             continue;
345af55b781SPhilippe Mathieu-Daudé         }
346af55b781SPhilippe Mathieu-Daudé         if (!mc->dev[idx].is_timer16) {
347af55b781SPhilippe Mathieu-Daudé             create_unimplemented_device("avr-timer8",
348af55b781SPhilippe Mathieu-Daudé                                         OFFSET_DATA + mc->dev[idx].addr, 5);
349af55b781SPhilippe Mathieu-Daudé             create_unimplemented_device("avr-timer8-intmask",
350af55b781SPhilippe Mathieu-Daudé                                         OFFSET_DATA
351af55b781SPhilippe Mathieu-Daudé                                         + mc->dev[idx].intmask_addr, 1);
352af55b781SPhilippe Mathieu-Daudé             create_unimplemented_device("avr-timer8-intflag",
353af55b781SPhilippe Mathieu-Daudé                                         OFFSET_DATA
354af55b781SPhilippe Mathieu-Daudé                                         + mc->dev[idx].intflag_addr, 1);
355af55b781SPhilippe Mathieu-Daudé             continue;
356af55b781SPhilippe Mathieu-Daudé         }
357af55b781SPhilippe Mathieu-Daudé         devname = g_strdup_printf("timer%zu", i);
358af55b781SPhilippe Mathieu-Daudé         object_initialize_child(OBJECT(dev), devname, &s->timer[i],
359af55b781SPhilippe Mathieu-Daudé                                 TYPE_AVR_TIMER16);
360af55b781SPhilippe Mathieu-Daudé         object_property_set_uint(OBJECT(&s->timer[i]), "cpu-frequency-hz",
361af55b781SPhilippe Mathieu-Daudé                                  s->xtal_freq_hz, &error_abort);
362af55b781SPhilippe Mathieu-Daudé         sbd = SYS_BUS_DEVICE(&s->timer[i]);
363af55b781SPhilippe Mathieu-Daudé         sysbus_realize(sbd, &error_abort);
364af55b781SPhilippe Mathieu-Daudé         sysbus_mmio_map(sbd, 0, OFFSET_DATA + mc->dev[idx].addr);
365af55b781SPhilippe Mathieu-Daudé         sysbus_mmio_map(sbd, 1, OFFSET_DATA + mc->dev[idx].intmask_addr);
366af55b781SPhilippe Mathieu-Daudé         sysbus_mmio_map(sbd, 2, OFFSET_DATA + mc->dev[idx].intflag_addr);
367af55b781SPhilippe Mathieu-Daudé         connect_peripheral_irq(mc, sbd, 0, cpudev, TIMER_CAPT_IRQ(i));
368af55b781SPhilippe Mathieu-Daudé         connect_peripheral_irq(mc, sbd, 1, cpudev, TIMER_COMPA_IRQ(i));
369af55b781SPhilippe Mathieu-Daudé         connect_peripheral_irq(mc, sbd, 2, cpudev, TIMER_COMPB_IRQ(i));
370af55b781SPhilippe Mathieu-Daudé         connect_peripheral_irq(mc, sbd, 3, cpudev, TIMER_COMPC_IRQ(i));
371af55b781SPhilippe Mathieu-Daudé         connect_peripheral_irq(mc, sbd, 4, cpudev, TIMER_OVF_IRQ(i));
372af55b781SPhilippe Mathieu-Daudé         connect_power_reduction_gpio(s, mc, DEVICE(&s->timer[i]), idx);
373af55b781SPhilippe Mathieu-Daudé         g_free(devname);
374af55b781SPhilippe Mathieu-Daudé     }
375af55b781SPhilippe Mathieu-Daudé 
376af55b781SPhilippe Mathieu-Daudé     create_unimplemented_device("avr-twi",          OFFSET_DATA + 0x0b8, 6);
377af55b781SPhilippe Mathieu-Daudé     create_unimplemented_device("avr-adc",          OFFSET_DATA + 0x078, 8);
378af55b781SPhilippe Mathieu-Daudé     create_unimplemented_device("avr-ext-mem-ctrl", OFFSET_DATA + 0x074, 2);
379af55b781SPhilippe Mathieu-Daudé     create_unimplemented_device("avr-watchdog",     OFFSET_DATA + 0x060, 1);
380af55b781SPhilippe Mathieu-Daudé     create_unimplemented_device("avr-spi",          OFFSET_DATA + 0x04c, 3);
381af55b781SPhilippe Mathieu-Daudé     create_unimplemented_device("avr-eeprom",       OFFSET_DATA + 0x03f, 3);
382af55b781SPhilippe Mathieu-Daudé }
383af55b781SPhilippe Mathieu-Daudé 
38447c7764bSRichard Henderson static const Property atmega_props[] = {
385af55b781SPhilippe Mathieu-Daudé     DEFINE_PROP_UINT64("xtal-frequency-hz", AtmegaMcuState,
386af55b781SPhilippe Mathieu-Daudé                        xtal_freq_hz, 0),
387af55b781SPhilippe Mathieu-Daudé };
388af55b781SPhilippe Mathieu-Daudé 
atmega_class_init(ObjectClass * oc,const void * data)389*12d1a768SPhilippe Mathieu-Daudé static void atmega_class_init(ObjectClass *oc, const void *data)
390af55b781SPhilippe Mathieu-Daudé {
391af55b781SPhilippe Mathieu-Daudé     DeviceClass *dc = DEVICE_CLASS(oc);
392af55b781SPhilippe Mathieu-Daudé 
393af55b781SPhilippe Mathieu-Daudé     dc->realize = atmega_realize;
394af55b781SPhilippe Mathieu-Daudé     device_class_set_props(dc, atmega_props);
395af55b781SPhilippe Mathieu-Daudé     /* Reason: Mapped at fixed location on the system bus */
396af55b781SPhilippe Mathieu-Daudé     dc->user_creatable = false;
397af55b781SPhilippe Mathieu-Daudé }
398af55b781SPhilippe Mathieu-Daudé 
atmega168_class_init(ObjectClass * oc,const void * data)399*12d1a768SPhilippe Mathieu-Daudé static void atmega168_class_init(ObjectClass *oc, const void *data)
400af55b781SPhilippe Mathieu-Daudé {
401af55b781SPhilippe Mathieu-Daudé     AtmegaMcuClass *amc = ATMEGA_MCU_CLASS(oc);
402af55b781SPhilippe Mathieu-Daudé 
403af55b781SPhilippe Mathieu-Daudé     amc->cpu_type = AVR_CPU_TYPE_NAME("avr5");
404af55b781SPhilippe Mathieu-Daudé     amc->flash_size = 16 * KiB;
405af55b781SPhilippe Mathieu-Daudé     amc->eeprom_size = 512;
406af55b781SPhilippe Mathieu-Daudé     amc->sram_size = 1 * KiB;
407af55b781SPhilippe Mathieu-Daudé     amc->io_size = 256;
408af55b781SPhilippe Mathieu-Daudé     amc->gpio_count = 23;
409af55b781SPhilippe Mathieu-Daudé     amc->adc_count = 6;
410af55b781SPhilippe Mathieu-Daudé     amc->irq = irq168_328;
411af55b781SPhilippe Mathieu-Daudé     amc->dev = dev168_328;
412af55b781SPhilippe Mathieu-Daudé };
413af55b781SPhilippe Mathieu-Daudé 
atmega328_class_init(ObjectClass * oc,const void * data)414*12d1a768SPhilippe Mathieu-Daudé static void atmega328_class_init(ObjectClass *oc, const void *data)
415af55b781SPhilippe Mathieu-Daudé {
416af55b781SPhilippe Mathieu-Daudé     AtmegaMcuClass *amc = ATMEGA_MCU_CLASS(oc);
417af55b781SPhilippe Mathieu-Daudé 
418af55b781SPhilippe Mathieu-Daudé     amc->cpu_type = AVR_CPU_TYPE_NAME("avr5");
419af55b781SPhilippe Mathieu-Daudé     amc->flash_size = 32 * KiB;
420af55b781SPhilippe Mathieu-Daudé     amc->eeprom_size = 1 * KiB;
421af55b781SPhilippe Mathieu-Daudé     amc->sram_size = 2 * KiB;
422af55b781SPhilippe Mathieu-Daudé     amc->io_size = 256;
423af55b781SPhilippe Mathieu-Daudé     amc->gpio_count = 23;
424af55b781SPhilippe Mathieu-Daudé     amc->adc_count = 6;
425af55b781SPhilippe Mathieu-Daudé     amc->irq = irq168_328;
426af55b781SPhilippe Mathieu-Daudé     amc->dev = dev168_328;
427af55b781SPhilippe Mathieu-Daudé };
428af55b781SPhilippe Mathieu-Daudé 
atmega1280_class_init(ObjectClass * oc,const void * data)429*12d1a768SPhilippe Mathieu-Daudé static void atmega1280_class_init(ObjectClass *oc, const void *data)
430af55b781SPhilippe Mathieu-Daudé {
431af55b781SPhilippe Mathieu-Daudé     AtmegaMcuClass *amc = ATMEGA_MCU_CLASS(oc);
432af55b781SPhilippe Mathieu-Daudé 
4336cd04a88SFrederic Konrad     amc->cpu_type = AVR_CPU_TYPE_NAME("avr51");
434af55b781SPhilippe Mathieu-Daudé     amc->flash_size = 128 * KiB;
435af55b781SPhilippe Mathieu-Daudé     amc->eeprom_size = 4 * KiB;
436af55b781SPhilippe Mathieu-Daudé     amc->sram_size = 8 * KiB;
437af55b781SPhilippe Mathieu-Daudé     amc->io_size = 512;
438af55b781SPhilippe Mathieu-Daudé     amc->gpio_count = 86;
439af55b781SPhilippe Mathieu-Daudé     amc->adc_count = 16;
440af55b781SPhilippe Mathieu-Daudé     amc->irq = irq1280_2560;
441af55b781SPhilippe Mathieu-Daudé     amc->dev = dev1280_2560;
442af55b781SPhilippe Mathieu-Daudé };
443af55b781SPhilippe Mathieu-Daudé 
atmega2560_class_init(ObjectClass * oc,const void * data)444*12d1a768SPhilippe Mathieu-Daudé static void atmega2560_class_init(ObjectClass *oc, const void *data)
445af55b781SPhilippe Mathieu-Daudé {
446af55b781SPhilippe Mathieu-Daudé     AtmegaMcuClass *amc = ATMEGA_MCU_CLASS(oc);
447af55b781SPhilippe Mathieu-Daudé 
448af55b781SPhilippe Mathieu-Daudé     amc->cpu_type = AVR_CPU_TYPE_NAME("avr6");
449af55b781SPhilippe Mathieu-Daudé     amc->flash_size = 256 * KiB;
450af55b781SPhilippe Mathieu-Daudé     amc->eeprom_size = 4 * KiB;
451af55b781SPhilippe Mathieu-Daudé     amc->sram_size = 8 * KiB;
452af55b781SPhilippe Mathieu-Daudé     amc->io_size = 512;
453af55b781SPhilippe Mathieu-Daudé     amc->gpio_count = 54;
454af55b781SPhilippe Mathieu-Daudé     amc->adc_count = 16;
455af55b781SPhilippe Mathieu-Daudé     amc->irq = irq1280_2560;
456af55b781SPhilippe Mathieu-Daudé     amc->dev = dev1280_2560;
457af55b781SPhilippe Mathieu-Daudé };
458af55b781SPhilippe Mathieu-Daudé 
459af55b781SPhilippe Mathieu-Daudé static const TypeInfo atmega_mcu_types[] = {
460af55b781SPhilippe Mathieu-Daudé     {
461af55b781SPhilippe Mathieu-Daudé         .name           = TYPE_ATMEGA168_MCU,
462af55b781SPhilippe Mathieu-Daudé         .parent         = TYPE_ATMEGA_MCU,
463af55b781SPhilippe Mathieu-Daudé         .class_init     = atmega168_class_init,
464af55b781SPhilippe Mathieu-Daudé     }, {
465af55b781SPhilippe Mathieu-Daudé         .name           = TYPE_ATMEGA328_MCU,
466af55b781SPhilippe Mathieu-Daudé         .parent         = TYPE_ATMEGA_MCU,
467af55b781SPhilippe Mathieu-Daudé         .class_init     = atmega328_class_init,
468af55b781SPhilippe Mathieu-Daudé     }, {
469af55b781SPhilippe Mathieu-Daudé         .name           = TYPE_ATMEGA1280_MCU,
470af55b781SPhilippe Mathieu-Daudé         .parent         = TYPE_ATMEGA_MCU,
471af55b781SPhilippe Mathieu-Daudé         .class_init     = atmega1280_class_init,
472af55b781SPhilippe Mathieu-Daudé     }, {
473af55b781SPhilippe Mathieu-Daudé         .name           = TYPE_ATMEGA2560_MCU,
474af55b781SPhilippe Mathieu-Daudé         .parent         = TYPE_ATMEGA_MCU,
475af55b781SPhilippe Mathieu-Daudé         .class_init     = atmega2560_class_init,
476af55b781SPhilippe Mathieu-Daudé     }, {
477af55b781SPhilippe Mathieu-Daudé         .name           = TYPE_ATMEGA_MCU,
478af55b781SPhilippe Mathieu-Daudé         .parent         = TYPE_SYS_BUS_DEVICE,
479af55b781SPhilippe Mathieu-Daudé         .instance_size  = sizeof(AtmegaMcuState),
480af55b781SPhilippe Mathieu-Daudé         .class_size     = sizeof(AtmegaMcuClass),
481af55b781SPhilippe Mathieu-Daudé         .class_init     = atmega_class_init,
482af55b781SPhilippe Mathieu-Daudé         .abstract       = true,
483af55b781SPhilippe Mathieu-Daudé     }
484af55b781SPhilippe Mathieu-Daudé };
485af55b781SPhilippe Mathieu-Daudé 
486af55b781SPhilippe Mathieu-Daudé DEFINE_TYPES(atmega_mcu_types)
487