atmega.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457
  1. /*
  2. * QEMU ATmega MCU
  3. *
  4. * Copyright (c) 2019-2020 Philippe Mathieu-Daudé
  5. *
  6. * This work is licensed under the terms of the GNU GPLv2 or later.
  7. * See the COPYING file in the top-level directory.
  8. * SPDX-License-Identifier: GPL-2.0-or-later
  9. */
  10. #include "qemu/osdep.h"
  11. #include "qemu/module.h"
  12. #include "qemu/units.h"
  13. #include "qapi/error.h"
  14. #include "exec/memory.h"
  15. #include "exec/address-spaces.h"
  16. #include "sysemu/sysemu.h"
  17. #include "hw/qdev-properties.h"
  18. #include "hw/sysbus.h"
  19. #include "qom/object.h"
  20. #include "hw/misc/unimp.h"
  21. #include "atmega.h"
  22. enum AtmegaPeripheral {
  23. POWER0, POWER1,
  24. GPIOA, GPIOB, GPIOC, GPIOD, GPIOE, GPIOF,
  25. GPIOG, GPIOH, GPIOI, GPIOJ, GPIOK, GPIOL,
  26. USART0, USART1, USART2, USART3,
  27. TIMER0, TIMER1, TIMER2, TIMER3, TIMER4, TIMER5,
  28. PERIFMAX
  29. };
  30. #define GPIO(n) (n + GPIOA)
  31. #define USART(n) (n + USART0)
  32. #define TIMER(n) (n + TIMER0)
  33. #define POWER(n) (n + POWER0)
  34. typedef struct {
  35. uint16_t addr;
  36. enum AtmegaPeripheral power_index;
  37. uint8_t power_bit;
  38. /* timer specific */
  39. uint16_t intmask_addr;
  40. uint16_t intflag_addr;
  41. bool is_timer16;
  42. } peripheral_cfg;
  43. struct AtmegaMcuClass {
  44. /*< private >*/
  45. SysBusDeviceClass parent_class;
  46. /*< public >*/
  47. const char *uc_name;
  48. const char *cpu_type;
  49. size_t flash_size;
  50. size_t eeprom_size;
  51. size_t sram_size;
  52. size_t io_size;
  53. size_t gpio_count;
  54. size_t adc_count;
  55. const uint8_t *irq;
  56. const peripheral_cfg *dev;
  57. };
  58. typedef struct AtmegaMcuClass AtmegaMcuClass;
  59. DECLARE_CLASS_CHECKERS(AtmegaMcuClass, ATMEGA_MCU,
  60. TYPE_ATMEGA_MCU)
  61. static const peripheral_cfg dev168_328[PERIFMAX] = {
  62. [USART0] = { 0xc0, POWER0, 1 },
  63. [TIMER2] = { 0xb0, POWER0, 6, 0x70, 0x37, false },
  64. [TIMER1] = { 0x80, POWER0, 3, 0x6f, 0x36, true },
  65. [POWER0] = { 0x64 },
  66. [TIMER0] = { 0x44, POWER0, 5, 0x6e, 0x35, false },
  67. [GPIOD] = { 0x29 },
  68. [GPIOC] = { 0x26 },
  69. [GPIOB] = { 0x23 },
  70. }, dev1280_2560[PERIFMAX] = {
  71. [USART3] = { 0x130, POWER1, 2 },
  72. [TIMER5] = { 0x120, POWER1, 5, 0x73, 0x3a, true },
  73. [GPIOL] = { 0x109 },
  74. [GPIOK] = { 0x106 },
  75. [GPIOJ] = { 0x103 },
  76. [GPIOH] = { 0x100 },
  77. [USART2] = { 0xd0, POWER1, 1 },
  78. [USART1] = { 0xc8, POWER1, 0 },
  79. [USART0] = { 0xc0, POWER0, 1 },
  80. [TIMER2] = { 0xb0, POWER0, 6, 0x70, 0x37, false }, /* TODO async */
  81. [TIMER4] = { 0xa0, POWER1, 4, 0x72, 0x39, true },
  82. [TIMER3] = { 0x90, POWER1, 3, 0x71, 0x38, true },
  83. [TIMER1] = { 0x80, POWER0, 3, 0x6f, 0x36, true },
  84. [POWER1] = { 0x65 },
  85. [POWER0] = { 0x64 },
  86. [TIMER0] = { 0x44, POWER0, 5, 0x6e, 0x35, false },
  87. [GPIOG] = { 0x32 },
  88. [GPIOF] = { 0x2f },
  89. [GPIOE] = { 0x2c },
  90. [GPIOD] = { 0x29 },
  91. [GPIOC] = { 0x26 },
  92. [GPIOB] = { 0x23 },
  93. [GPIOA] = { 0x20 },
  94. };
  95. enum AtmegaIrq {
  96. USART0_RXC_IRQ, USART0_DRE_IRQ, USART0_TXC_IRQ,
  97. USART1_RXC_IRQ, USART1_DRE_IRQ, USART1_TXC_IRQ,
  98. USART2_RXC_IRQ, USART2_DRE_IRQ, USART2_TXC_IRQ,
  99. USART3_RXC_IRQ, USART3_DRE_IRQ, USART3_TXC_IRQ,
  100. TIMER0_CAPT_IRQ, TIMER0_COMPA_IRQ, TIMER0_COMPB_IRQ,
  101. TIMER0_COMPC_IRQ, TIMER0_OVF_IRQ,
  102. TIMER1_CAPT_IRQ, TIMER1_COMPA_IRQ, TIMER1_COMPB_IRQ,
  103. TIMER1_COMPC_IRQ, TIMER1_OVF_IRQ,
  104. TIMER2_CAPT_IRQ, TIMER2_COMPA_IRQ, TIMER2_COMPB_IRQ,
  105. TIMER2_COMPC_IRQ, TIMER2_OVF_IRQ,
  106. TIMER3_CAPT_IRQ, TIMER3_COMPA_IRQ, TIMER3_COMPB_IRQ,
  107. TIMER3_COMPC_IRQ, TIMER3_OVF_IRQ,
  108. TIMER4_CAPT_IRQ, TIMER4_COMPA_IRQ, TIMER4_COMPB_IRQ,
  109. TIMER4_COMPC_IRQ, TIMER4_OVF_IRQ,
  110. TIMER5_CAPT_IRQ, TIMER5_COMPA_IRQ, TIMER5_COMPB_IRQ,
  111. TIMER5_COMPC_IRQ, TIMER5_OVF_IRQ,
  112. IRQ_COUNT
  113. };
  114. #define USART_IRQ_COUNT 3
  115. #define USART_RXC_IRQ(n) (n * USART_IRQ_COUNT + USART0_RXC_IRQ)
  116. #define USART_DRE_IRQ(n) (n * USART_IRQ_COUNT + USART0_DRE_IRQ)
  117. #define USART_TXC_IRQ(n) (n * USART_IRQ_COUNT + USART0_TXC_IRQ)
  118. #define TIMER_IRQ_COUNT 5
  119. #define TIMER_CAPT_IRQ(n) (n * TIMER_IRQ_COUNT + TIMER0_CAPT_IRQ)
  120. #define TIMER_COMPA_IRQ(n) (n * TIMER_IRQ_COUNT + TIMER0_COMPA_IRQ)
  121. #define TIMER_COMPB_IRQ(n) (n * TIMER_IRQ_COUNT + TIMER0_COMPB_IRQ)
  122. #define TIMER_COMPC_IRQ(n) (n * TIMER_IRQ_COUNT + TIMER0_COMPC_IRQ)
  123. #define TIMER_OVF_IRQ(n) (n * TIMER_IRQ_COUNT + TIMER0_OVF_IRQ)
  124. static const uint8_t irq168_328[IRQ_COUNT] = {
  125. [TIMER2_COMPA_IRQ] = 8,
  126. [TIMER2_COMPB_IRQ] = 9,
  127. [TIMER2_OVF_IRQ] = 10,
  128. [TIMER1_CAPT_IRQ] = 11,
  129. [TIMER1_COMPA_IRQ] = 12,
  130. [TIMER1_COMPB_IRQ] = 13,
  131. [TIMER1_OVF_IRQ] = 14,
  132. [TIMER0_COMPA_IRQ] = 15,
  133. [TIMER0_COMPB_IRQ] = 16,
  134. [TIMER0_OVF_IRQ] = 17,
  135. [USART0_RXC_IRQ] = 19,
  136. [USART0_DRE_IRQ] = 20,
  137. [USART0_TXC_IRQ] = 21,
  138. }, irq1280_2560[IRQ_COUNT] = {
  139. [TIMER2_COMPA_IRQ] = 14,
  140. [TIMER2_COMPB_IRQ] = 15,
  141. [TIMER2_OVF_IRQ] = 16,
  142. [TIMER1_CAPT_IRQ] = 17,
  143. [TIMER1_COMPA_IRQ] = 18,
  144. [TIMER1_COMPB_IRQ] = 19,
  145. [TIMER1_COMPC_IRQ] = 20,
  146. [TIMER1_OVF_IRQ] = 21,
  147. [TIMER0_COMPA_IRQ] = 22,
  148. [TIMER0_COMPB_IRQ] = 23,
  149. [TIMER0_OVF_IRQ] = 24,
  150. [USART0_RXC_IRQ] = 26,
  151. [USART0_DRE_IRQ] = 27,
  152. [USART0_TXC_IRQ] = 28,
  153. [TIMER3_CAPT_IRQ] = 32,
  154. [TIMER3_COMPA_IRQ] = 33,
  155. [TIMER3_COMPB_IRQ] = 34,
  156. [TIMER3_COMPC_IRQ] = 35,
  157. [TIMER3_OVF_IRQ] = 36,
  158. [USART1_RXC_IRQ] = 37,
  159. [USART1_DRE_IRQ] = 38,
  160. [USART1_TXC_IRQ] = 39,
  161. [TIMER4_CAPT_IRQ] = 42,
  162. [TIMER4_COMPA_IRQ] = 43,
  163. [TIMER4_COMPB_IRQ] = 44,
  164. [TIMER4_COMPC_IRQ] = 45,
  165. [TIMER4_OVF_IRQ] = 46,
  166. [TIMER5_CAPT_IRQ] = 47,
  167. [TIMER5_COMPA_IRQ] = 48,
  168. [TIMER5_COMPB_IRQ] = 49,
  169. [TIMER5_COMPC_IRQ] = 50,
  170. [TIMER5_OVF_IRQ] = 51,
  171. [USART2_RXC_IRQ] = 52,
  172. [USART2_DRE_IRQ] = 53,
  173. [USART2_TXC_IRQ] = 54,
  174. [USART3_RXC_IRQ] = 55,
  175. [USART3_DRE_IRQ] = 56,
  176. [USART3_TXC_IRQ] = 57,
  177. };
  178. static void connect_peripheral_irq(const AtmegaMcuClass *k,
  179. SysBusDevice *dev, int dev_irqn,
  180. DeviceState *cpu,
  181. unsigned peripheral_index)
  182. {
  183. int cpu_irq = k->irq[peripheral_index];
  184. if (!cpu_irq) {
  185. return;
  186. }
  187. /* FIXME move that to avr_cpu_set_int() once 'sample' board is removed */
  188. assert(cpu_irq >= 2);
  189. cpu_irq -= 2;
  190. sysbus_connect_irq(dev, dev_irqn, qdev_get_gpio_in(cpu, cpu_irq));
  191. }
  192. static void connect_power_reduction_gpio(AtmegaMcuState *s,
  193. const AtmegaMcuClass *k,
  194. DeviceState *cpu,
  195. unsigned peripheral_index)
  196. {
  197. unsigned power_index = k->dev[peripheral_index].power_index;
  198. assert(k->dev[power_index].addr);
  199. sysbus_connect_irq(SYS_BUS_DEVICE(&s->pwr[power_index - POWER0]),
  200. k->dev[peripheral_index].power_bit,
  201. qdev_get_gpio_in(cpu, 0));
  202. }
  203. static void atmega_realize(DeviceState *dev, Error **errp)
  204. {
  205. AtmegaMcuState *s = ATMEGA_MCU(dev);
  206. const AtmegaMcuClass *mc = ATMEGA_MCU_GET_CLASS(dev);
  207. DeviceState *cpudev;
  208. SysBusDevice *sbd;
  209. char *devname;
  210. size_t i;
  211. assert(mc->io_size <= 0x200);
  212. if (!s->xtal_freq_hz) {
  213. error_setg(errp, "\"xtal-frequency-hz\" property must be provided.");
  214. return;
  215. }
  216. /* CPU */
  217. object_initialize_child(OBJECT(dev), "cpu", &s->cpu, mc->cpu_type);
  218. qdev_realize(DEVICE(&s->cpu), NULL, &error_abort);
  219. cpudev = DEVICE(&s->cpu);
  220. /* SRAM */
  221. memory_region_init_ram(&s->sram, OBJECT(dev), "sram", mc->sram_size,
  222. &error_abort);
  223. memory_region_add_subregion(get_system_memory(),
  224. OFFSET_DATA + mc->io_size, &s->sram);
  225. /* Flash */
  226. memory_region_init_rom(&s->flash, OBJECT(dev),
  227. "flash", mc->flash_size, &error_fatal);
  228. memory_region_add_subregion(get_system_memory(), OFFSET_CODE, &s->flash);
  229. /*
  230. * I/O
  231. *
  232. * 0x00 - 0x1f: Registers
  233. * 0x20 - 0x5f: I/O memory
  234. * 0x60 - 0xff: Extended I/O
  235. */
  236. s->io = qdev_new(TYPE_UNIMPLEMENTED_DEVICE);
  237. qdev_prop_set_string(s->io, "name", "I/O");
  238. qdev_prop_set_uint64(s->io, "size", mc->io_size);
  239. sysbus_realize_and_unref(SYS_BUS_DEVICE(s->io), &error_fatal);
  240. sysbus_mmio_map_overlap(SYS_BUS_DEVICE(s->io), 0, OFFSET_DATA, -1234);
  241. /* Power Reduction */
  242. for (i = 0; i < POWER_MAX; i++) {
  243. int idx = POWER(i);
  244. if (!mc->dev[idx].addr) {
  245. continue;
  246. }
  247. devname = g_strdup_printf("power%zu", i);
  248. object_initialize_child(OBJECT(dev), devname, &s->pwr[i],
  249. TYPE_AVR_MASK);
  250. sysbus_realize(SYS_BUS_DEVICE(&s->pwr[i]), &error_abort);
  251. sysbus_mmio_map(SYS_BUS_DEVICE(&s->pwr[i]), 0,
  252. OFFSET_DATA + mc->dev[idx].addr);
  253. g_free(devname);
  254. }
  255. /* GPIO */
  256. for (i = 0; i < GPIO_MAX; i++) {
  257. int idx = GPIO(i);
  258. if (!mc->dev[idx].addr) {
  259. continue;
  260. }
  261. devname = g_strdup_printf("atmega-gpio-%c", 'a' + (char)i);
  262. create_unimplemented_device(devname,
  263. OFFSET_DATA + mc->dev[idx].addr, 3);
  264. g_free(devname);
  265. }
  266. /* USART */
  267. for (i = 0; i < USART_MAX; i++) {
  268. int idx = USART(i);
  269. if (!mc->dev[idx].addr) {
  270. continue;
  271. }
  272. devname = g_strdup_printf("usart%zu", i);
  273. object_initialize_child(OBJECT(dev), devname, &s->usart[i],
  274. TYPE_AVR_USART);
  275. qdev_prop_set_chr(DEVICE(&s->usart[i]), "chardev", serial_hd(i));
  276. sbd = SYS_BUS_DEVICE(&s->usart[i]);
  277. sysbus_realize(sbd, &error_abort);
  278. sysbus_mmio_map(sbd, 0, OFFSET_DATA + mc->dev[USART(i)].addr);
  279. connect_peripheral_irq(mc, sbd, 0, cpudev, USART_RXC_IRQ(i));
  280. connect_peripheral_irq(mc, sbd, 1, cpudev, USART_DRE_IRQ(i));
  281. connect_peripheral_irq(mc, sbd, 2, cpudev, USART_TXC_IRQ(i));
  282. connect_power_reduction_gpio(s, mc, DEVICE(&s->usart[i]), idx);
  283. g_free(devname);
  284. }
  285. /* Timer */
  286. for (i = 0; i < TIMER_MAX; i++) {
  287. int idx = TIMER(i);
  288. if (!mc->dev[idx].addr) {
  289. continue;
  290. }
  291. if (!mc->dev[idx].is_timer16) {
  292. create_unimplemented_device("avr-timer8",
  293. OFFSET_DATA + mc->dev[idx].addr, 5);
  294. create_unimplemented_device("avr-timer8-intmask",
  295. OFFSET_DATA
  296. + mc->dev[idx].intmask_addr, 1);
  297. create_unimplemented_device("avr-timer8-intflag",
  298. OFFSET_DATA
  299. + mc->dev[idx].intflag_addr, 1);
  300. continue;
  301. }
  302. devname = g_strdup_printf("timer%zu", i);
  303. object_initialize_child(OBJECT(dev), devname, &s->timer[i],
  304. TYPE_AVR_TIMER16);
  305. object_property_set_uint(OBJECT(&s->timer[i]), "cpu-frequency-hz",
  306. s->xtal_freq_hz, &error_abort);
  307. sbd = SYS_BUS_DEVICE(&s->timer[i]);
  308. sysbus_realize(sbd, &error_abort);
  309. sysbus_mmio_map(sbd, 0, OFFSET_DATA + mc->dev[idx].addr);
  310. sysbus_mmio_map(sbd, 1, OFFSET_DATA + mc->dev[idx].intmask_addr);
  311. sysbus_mmio_map(sbd, 2, OFFSET_DATA + mc->dev[idx].intflag_addr);
  312. connect_peripheral_irq(mc, sbd, 0, cpudev, TIMER_CAPT_IRQ(i));
  313. connect_peripheral_irq(mc, sbd, 1, cpudev, TIMER_COMPA_IRQ(i));
  314. connect_peripheral_irq(mc, sbd, 2, cpudev, TIMER_COMPB_IRQ(i));
  315. connect_peripheral_irq(mc, sbd, 3, cpudev, TIMER_COMPC_IRQ(i));
  316. connect_peripheral_irq(mc, sbd, 4, cpudev, TIMER_OVF_IRQ(i));
  317. connect_power_reduction_gpio(s, mc, DEVICE(&s->timer[i]), idx);
  318. g_free(devname);
  319. }
  320. create_unimplemented_device("avr-twi", OFFSET_DATA + 0x0b8, 6);
  321. create_unimplemented_device("avr-adc", OFFSET_DATA + 0x078, 8);
  322. create_unimplemented_device("avr-ext-mem-ctrl", OFFSET_DATA + 0x074, 2);
  323. create_unimplemented_device("avr-watchdog", OFFSET_DATA + 0x060, 1);
  324. create_unimplemented_device("avr-spi", OFFSET_DATA + 0x04c, 3);
  325. create_unimplemented_device("avr-eeprom", OFFSET_DATA + 0x03f, 3);
  326. }
  327. static Property atmega_props[] = {
  328. DEFINE_PROP_UINT64("xtal-frequency-hz", AtmegaMcuState,
  329. xtal_freq_hz, 0),
  330. DEFINE_PROP_END_OF_LIST()
  331. };
  332. static void atmega_class_init(ObjectClass *oc, void *data)
  333. {
  334. DeviceClass *dc = DEVICE_CLASS(oc);
  335. dc->realize = atmega_realize;
  336. device_class_set_props(dc, atmega_props);
  337. /* Reason: Mapped at fixed location on the system bus */
  338. dc->user_creatable = false;
  339. }
  340. static void atmega168_class_init(ObjectClass *oc, void *data)
  341. {
  342. AtmegaMcuClass *amc = ATMEGA_MCU_CLASS(oc);
  343. amc->cpu_type = AVR_CPU_TYPE_NAME("avr5");
  344. amc->flash_size = 16 * KiB;
  345. amc->eeprom_size = 512;
  346. amc->sram_size = 1 * KiB;
  347. amc->io_size = 256;
  348. amc->gpio_count = 23;
  349. amc->adc_count = 6;
  350. amc->irq = irq168_328;
  351. amc->dev = dev168_328;
  352. };
  353. static void atmega328_class_init(ObjectClass *oc, void *data)
  354. {
  355. AtmegaMcuClass *amc = ATMEGA_MCU_CLASS(oc);
  356. amc->cpu_type = AVR_CPU_TYPE_NAME("avr5");
  357. amc->flash_size = 32 * KiB;
  358. amc->eeprom_size = 1 * KiB;
  359. amc->sram_size = 2 * KiB;
  360. amc->io_size = 256;
  361. amc->gpio_count = 23;
  362. amc->adc_count = 6;
  363. amc->irq = irq168_328;
  364. amc->dev = dev168_328;
  365. };
  366. static void atmega1280_class_init(ObjectClass *oc, void *data)
  367. {
  368. AtmegaMcuClass *amc = ATMEGA_MCU_CLASS(oc);
  369. amc->cpu_type = AVR_CPU_TYPE_NAME("avr51");
  370. amc->flash_size = 128 * KiB;
  371. amc->eeprom_size = 4 * KiB;
  372. amc->sram_size = 8 * KiB;
  373. amc->io_size = 512;
  374. amc->gpio_count = 86;
  375. amc->adc_count = 16;
  376. amc->irq = irq1280_2560;
  377. amc->dev = dev1280_2560;
  378. };
  379. static void atmega2560_class_init(ObjectClass *oc, void *data)
  380. {
  381. AtmegaMcuClass *amc = ATMEGA_MCU_CLASS(oc);
  382. amc->cpu_type = AVR_CPU_TYPE_NAME("avr6");
  383. amc->flash_size = 256 * KiB;
  384. amc->eeprom_size = 4 * KiB;
  385. amc->sram_size = 8 * KiB;
  386. amc->io_size = 512;
  387. amc->gpio_count = 54;
  388. amc->adc_count = 16;
  389. amc->irq = irq1280_2560;
  390. amc->dev = dev1280_2560;
  391. };
  392. static const TypeInfo atmega_mcu_types[] = {
  393. {
  394. .name = TYPE_ATMEGA168_MCU,
  395. .parent = TYPE_ATMEGA_MCU,
  396. .class_init = atmega168_class_init,
  397. }, {
  398. .name = TYPE_ATMEGA328_MCU,
  399. .parent = TYPE_ATMEGA_MCU,
  400. .class_init = atmega328_class_init,
  401. }, {
  402. .name = TYPE_ATMEGA1280_MCU,
  403. .parent = TYPE_ATMEGA_MCU,
  404. .class_init = atmega1280_class_init,
  405. }, {
  406. .name = TYPE_ATMEGA2560_MCU,
  407. .parent = TYPE_ATMEGA_MCU,
  408. .class_init = atmega2560_class_init,
  409. }, {
  410. .name = TYPE_ATMEGA_MCU,
  411. .parent = TYPE_SYS_BUS_DEVICE,
  412. .instance_size = sizeof(AtmegaMcuState),
  413. .class_size = sizeof(AtmegaMcuClass),
  414. .class_init = atmega_class_init,
  415. .abstract = true,
  416. }
  417. };
  418. DEFINE_TYPES(atmega_mcu_types)