2
0

cmsdk-apb-dualtimer.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559
  1. /*
  2. * ARM CMSDK APB dual-timer emulation
  3. *
  4. * Copyright (c) 2018 Linaro Limited
  5. * Written by Peter Maydell
  6. *
  7. * This program is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License version 2 or
  9. * (at your option) any later version.
  10. */
  11. /*
  12. * This is a model of the "APB dual-input timer" which is part of the Cortex-M
  13. * System Design Kit (CMSDK) and documented in the Cortex-M System
  14. * Design Kit Technical Reference Manual (ARM DDI0479C):
  15. * https://developer.arm.com/products/system-design/system-design-kits/cortex-m-system-design-kit
  16. */
  17. #include "qemu/osdep.h"
  18. #include "qemu/log.h"
  19. #include "trace.h"
  20. #include "qapi/error.h"
  21. #include "qemu/module.h"
  22. #include "hw/sysbus.h"
  23. #include "hw/irq.h"
  24. #include "hw/qdev-properties.h"
  25. #include "hw/registerfields.h"
  26. #include "hw/qdev-clock.h"
  27. #include "hw/timer/cmsdk-apb-dualtimer.h"
  28. #include "migration/vmstate.h"
  29. REG32(TIMER1LOAD, 0x0)
  30. REG32(TIMER1VALUE, 0x4)
  31. REG32(TIMER1CONTROL, 0x8)
  32. FIELD(CONTROL, ONESHOT, 0, 1)
  33. FIELD(CONTROL, SIZE, 1, 1)
  34. FIELD(CONTROL, PRESCALE, 2, 2)
  35. FIELD(CONTROL, INTEN, 5, 1)
  36. FIELD(CONTROL, MODE, 6, 1)
  37. FIELD(CONTROL, ENABLE, 7, 1)
  38. #define R_CONTROL_VALID_MASK (R_CONTROL_ONESHOT_MASK | R_CONTROL_SIZE_MASK | \
  39. R_CONTROL_PRESCALE_MASK | R_CONTROL_INTEN_MASK | \
  40. R_CONTROL_MODE_MASK | R_CONTROL_ENABLE_MASK)
  41. REG32(TIMER1INTCLR, 0xc)
  42. REG32(TIMER1RIS, 0x10)
  43. REG32(TIMER1MIS, 0x14)
  44. REG32(TIMER1BGLOAD, 0x18)
  45. REG32(TIMER2LOAD, 0x20)
  46. REG32(TIMER2VALUE, 0x24)
  47. REG32(TIMER2CONTROL, 0x28)
  48. REG32(TIMER2INTCLR, 0x2c)
  49. REG32(TIMER2RIS, 0x30)
  50. REG32(TIMER2MIS, 0x34)
  51. REG32(TIMER2BGLOAD, 0x38)
  52. REG32(TIMERITCR, 0xf00)
  53. FIELD(TIMERITCR, ENABLE, 0, 1)
  54. #define R_TIMERITCR_VALID_MASK R_TIMERITCR_ENABLE_MASK
  55. REG32(TIMERITOP, 0xf04)
  56. FIELD(TIMERITOP, TIMINT1, 0, 1)
  57. FIELD(TIMERITOP, TIMINT2, 1, 1)
  58. #define R_TIMERITOP_VALID_MASK (R_TIMERITOP_TIMINT1_MASK | \
  59. R_TIMERITOP_TIMINT2_MASK)
  60. REG32(PID4, 0xfd0)
  61. REG32(PID5, 0xfd4)
  62. REG32(PID6, 0xfd8)
  63. REG32(PID7, 0xfdc)
  64. REG32(PID0, 0xfe0)
  65. REG32(PID1, 0xfe4)
  66. REG32(PID2, 0xfe8)
  67. REG32(PID3, 0xfec)
  68. REG32(CID0, 0xff0)
  69. REG32(CID1, 0xff4)
  70. REG32(CID2, 0xff8)
  71. REG32(CID3, 0xffc)
  72. /* PID/CID values */
  73. static const int timer_id[] = {
  74. 0x04, 0x00, 0x00, 0x00, /* PID4..PID7 */
  75. 0x23, 0xb8, 0x1b, 0x00, /* PID0..PID3 */
  76. 0x0d, 0xf0, 0x05, 0xb1, /* CID0..CID3 */
  77. };
  78. static bool cmsdk_dualtimermod_intstatus(CMSDKAPBDualTimerModule *m)
  79. {
  80. /* Return masked interrupt status for the timer module */
  81. return m->intstatus && (m->control & R_CONTROL_INTEN_MASK);
  82. }
  83. static void cmsdk_apb_dualtimer_update(CMSDKAPBDualTimer *s)
  84. {
  85. bool timint1, timint2, timintc;
  86. if (s->timeritcr) {
  87. /* Integration test mode: outputs driven directly from TIMERITOP bits */
  88. timint1 = s->timeritop & R_TIMERITOP_TIMINT1_MASK;
  89. timint2 = s->timeritop & R_TIMERITOP_TIMINT2_MASK;
  90. } else {
  91. timint1 = cmsdk_dualtimermod_intstatus(&s->timermod[0]);
  92. timint2 = cmsdk_dualtimermod_intstatus(&s->timermod[1]);
  93. }
  94. timintc = timint1 || timint2;
  95. qemu_set_irq(s->timermod[0].timerint, timint1);
  96. qemu_set_irq(s->timermod[1].timerint, timint2);
  97. qemu_set_irq(s->timerintc, timintc);
  98. }
  99. static int cmsdk_dualtimermod_divisor(CMSDKAPBDualTimerModule *m)
  100. {
  101. /* Return the divisor set by the current CONTROL.PRESCALE value */
  102. switch (FIELD_EX32(m->control, CONTROL, PRESCALE)) {
  103. case 0:
  104. return 1;
  105. case 1:
  106. return 16;
  107. case 2:
  108. case 3: /* UNDEFINED, we treat like 2 (and complained when it was set) */
  109. return 256;
  110. default:
  111. g_assert_not_reached();
  112. }
  113. }
  114. static void cmsdk_dualtimermod_write_control(CMSDKAPBDualTimerModule *m,
  115. uint32_t newctrl)
  116. {
  117. /* Handle a write to the CONTROL register */
  118. uint32_t changed;
  119. ptimer_transaction_begin(m->timer);
  120. newctrl &= R_CONTROL_VALID_MASK;
  121. changed = m->control ^ newctrl;
  122. if (changed & ~newctrl & R_CONTROL_ENABLE_MASK) {
  123. /* ENABLE cleared, stop timer before any further changes */
  124. ptimer_stop(m->timer);
  125. }
  126. if (changed & R_CONTROL_PRESCALE_MASK) {
  127. int divisor;
  128. switch (FIELD_EX32(newctrl, CONTROL, PRESCALE)) {
  129. case 0:
  130. divisor = 1;
  131. break;
  132. case 1:
  133. divisor = 16;
  134. break;
  135. case 2:
  136. divisor = 256;
  137. break;
  138. case 3:
  139. /* UNDEFINED; complain, and arbitrarily treat like 2 */
  140. qemu_log_mask(LOG_GUEST_ERROR,
  141. "CMSDK APB dual-timer: CONTROL.PRESCALE==0b11"
  142. " is undefined behaviour\n");
  143. divisor = 256;
  144. break;
  145. default:
  146. g_assert_not_reached();
  147. }
  148. ptimer_set_period_from_clock(m->timer, m->parent->timclk, divisor);
  149. }
  150. if (changed & R_CONTROL_MODE_MASK) {
  151. uint32_t load;
  152. if (newctrl & R_CONTROL_MODE_MASK) {
  153. /* Periodic: the limit is the LOAD register value */
  154. load = m->load;
  155. } else {
  156. /* Free-running: counter wraps around */
  157. load = ptimer_get_limit(m->timer);
  158. if (!(m->control & R_CONTROL_SIZE_MASK)) {
  159. load = deposit32(m->load, 0, 16, load);
  160. }
  161. m->load = load;
  162. load = 0xffffffff;
  163. }
  164. if (!(m->control & R_CONTROL_SIZE_MASK)) {
  165. load &= 0xffff;
  166. }
  167. ptimer_set_limit(m->timer, load, 0);
  168. }
  169. if (changed & R_CONTROL_SIZE_MASK) {
  170. /* Timer switched between 16 and 32 bit count */
  171. uint32_t value, load;
  172. value = ptimer_get_count(m->timer);
  173. load = ptimer_get_limit(m->timer);
  174. if (newctrl & R_CONTROL_SIZE_MASK) {
  175. /* 16 -> 32, top half of VALUE is in struct field */
  176. value = deposit32(m->value, 0, 16, value);
  177. } else {
  178. /* 32 -> 16: save top half to struct field and truncate */
  179. m->value = value;
  180. value &= 0xffff;
  181. }
  182. if (newctrl & R_CONTROL_MODE_MASK) {
  183. /* Periodic, timer limit has LOAD value */
  184. if (newctrl & R_CONTROL_SIZE_MASK) {
  185. load = deposit32(m->load, 0, 16, load);
  186. } else {
  187. m->load = load;
  188. load &= 0xffff;
  189. }
  190. } else {
  191. /* Free-running, timer limit is set to give wraparound */
  192. if (newctrl & R_CONTROL_SIZE_MASK) {
  193. load = 0xffffffff;
  194. } else {
  195. load = 0xffff;
  196. }
  197. }
  198. ptimer_set_count(m->timer, value);
  199. ptimer_set_limit(m->timer, load, 0);
  200. }
  201. if (newctrl & R_CONTROL_ENABLE_MASK) {
  202. /*
  203. * ENABLE is set; start the timer after all other changes.
  204. * We start it even if the ENABLE bit didn't actually change,
  205. * in case the timer was an expired one-shot timer that has
  206. * now been changed into a free-running or periodic timer.
  207. */
  208. ptimer_run(m->timer, !!(newctrl & R_CONTROL_ONESHOT_MASK));
  209. }
  210. m->control = newctrl;
  211. ptimer_transaction_commit(m->timer);
  212. }
  213. static uint64_t cmsdk_apb_dualtimer_read(void *opaque, hwaddr offset,
  214. unsigned size)
  215. {
  216. CMSDKAPBDualTimer *s = CMSDK_APB_DUALTIMER(opaque);
  217. uint64_t r;
  218. if (offset >= A_TIMERITCR) {
  219. switch (offset) {
  220. case A_TIMERITCR:
  221. r = s->timeritcr;
  222. break;
  223. case A_PID4 ... A_CID3:
  224. r = timer_id[(offset - A_PID4) / 4];
  225. break;
  226. default:
  227. bad_offset:
  228. qemu_log_mask(LOG_GUEST_ERROR,
  229. "CMSDK APB dual-timer read: bad offset %x\n",
  230. (int) offset);
  231. r = 0;
  232. break;
  233. }
  234. } else {
  235. int timer = offset >> 5;
  236. CMSDKAPBDualTimerModule *m;
  237. if (timer >= ARRAY_SIZE(s->timermod)) {
  238. goto bad_offset;
  239. }
  240. m = &s->timermod[timer];
  241. switch (offset & 0x1F) {
  242. case A_TIMER1LOAD:
  243. case A_TIMER1BGLOAD:
  244. if (m->control & R_CONTROL_MODE_MASK) {
  245. /*
  246. * Periodic: the ptimer limit is the LOAD register value, (or
  247. * just the low 16 bits of it if the timer is in 16-bit mode)
  248. */
  249. r = ptimer_get_limit(m->timer);
  250. if (!(m->control & R_CONTROL_SIZE_MASK)) {
  251. r = deposit32(m->load, 0, 16, r);
  252. }
  253. } else {
  254. /* Free-running: LOAD register value is just in m->load */
  255. r = m->load;
  256. }
  257. break;
  258. case A_TIMER1VALUE:
  259. r = ptimer_get_count(m->timer);
  260. if (!(m->control & R_CONTROL_SIZE_MASK)) {
  261. r = deposit32(m->value, 0, 16, r);
  262. }
  263. break;
  264. case A_TIMER1CONTROL:
  265. r = m->control;
  266. break;
  267. case A_TIMER1RIS:
  268. r = m->intstatus;
  269. break;
  270. case A_TIMER1MIS:
  271. r = cmsdk_dualtimermod_intstatus(m);
  272. break;
  273. default:
  274. goto bad_offset;
  275. }
  276. }
  277. trace_cmsdk_apb_dualtimer_read(offset, r, size);
  278. return r;
  279. }
  280. static void cmsdk_apb_dualtimer_write(void *opaque, hwaddr offset,
  281. uint64_t value, unsigned size)
  282. {
  283. CMSDKAPBDualTimer *s = CMSDK_APB_DUALTIMER(opaque);
  284. trace_cmsdk_apb_dualtimer_write(offset, value, size);
  285. if (offset >= A_TIMERITCR) {
  286. switch (offset) {
  287. case A_TIMERITCR:
  288. s->timeritcr = value & R_TIMERITCR_VALID_MASK;
  289. cmsdk_apb_dualtimer_update(s);
  290. break;
  291. case A_TIMERITOP:
  292. s->timeritop = value & R_TIMERITOP_VALID_MASK;
  293. cmsdk_apb_dualtimer_update(s);
  294. break;
  295. default:
  296. bad_offset:
  297. qemu_log_mask(LOG_GUEST_ERROR,
  298. "CMSDK APB dual-timer write: bad offset %x\n",
  299. (int) offset);
  300. break;
  301. }
  302. } else {
  303. int timer = offset >> 5;
  304. CMSDKAPBDualTimerModule *m;
  305. if (timer >= ARRAY_SIZE(s->timermod)) {
  306. goto bad_offset;
  307. }
  308. m = &s->timermod[timer];
  309. switch (offset & 0x1F) {
  310. case A_TIMER1LOAD:
  311. /* Set the limit, and immediately reload the count from it */
  312. m->load = value;
  313. m->value = value;
  314. if (!(m->control & R_CONTROL_SIZE_MASK)) {
  315. value &= 0xffff;
  316. }
  317. ptimer_transaction_begin(m->timer);
  318. if (!(m->control & R_CONTROL_MODE_MASK)) {
  319. /*
  320. * In free-running mode this won't set the limit but will
  321. * still change the current count value.
  322. */
  323. ptimer_set_count(m->timer, value);
  324. } else {
  325. if (!value) {
  326. ptimer_stop(m->timer);
  327. }
  328. ptimer_set_limit(m->timer, value, 1);
  329. if (value && (m->control & R_CONTROL_ENABLE_MASK)) {
  330. /* Force possibly-expired oneshot timer to restart */
  331. ptimer_run(m->timer, 1);
  332. }
  333. }
  334. ptimer_transaction_commit(m->timer);
  335. break;
  336. case A_TIMER1BGLOAD:
  337. /* Set the limit, but not the current count */
  338. m->load = value;
  339. if (!(m->control & R_CONTROL_MODE_MASK)) {
  340. /* In free-running mode there is no limit */
  341. break;
  342. }
  343. if (!(m->control & R_CONTROL_SIZE_MASK)) {
  344. value &= 0xffff;
  345. }
  346. ptimer_transaction_begin(m->timer);
  347. ptimer_set_limit(m->timer, value, 0);
  348. ptimer_transaction_commit(m->timer);
  349. break;
  350. case A_TIMER1CONTROL:
  351. cmsdk_dualtimermod_write_control(m, value);
  352. cmsdk_apb_dualtimer_update(s);
  353. break;
  354. case A_TIMER1INTCLR:
  355. m->intstatus = 0;
  356. cmsdk_apb_dualtimer_update(s);
  357. break;
  358. default:
  359. goto bad_offset;
  360. }
  361. }
  362. }
  363. static const MemoryRegionOps cmsdk_apb_dualtimer_ops = {
  364. .read = cmsdk_apb_dualtimer_read,
  365. .write = cmsdk_apb_dualtimer_write,
  366. .endianness = DEVICE_LITTLE_ENDIAN,
  367. /* byte/halfword accesses are just zero-padded on reads and writes */
  368. .impl.min_access_size = 4,
  369. .impl.max_access_size = 4,
  370. .valid.min_access_size = 1,
  371. .valid.max_access_size = 4,
  372. };
  373. static void cmsdk_dualtimermod_tick(void *opaque)
  374. {
  375. CMSDKAPBDualTimerModule *m = opaque;
  376. m->intstatus = 1;
  377. cmsdk_apb_dualtimer_update(m->parent);
  378. }
  379. static void cmsdk_dualtimermod_reset(CMSDKAPBDualTimerModule *m)
  380. {
  381. m->control = R_CONTROL_INTEN_MASK;
  382. m->intstatus = 0;
  383. m->load = 0;
  384. m->value = 0xffffffff;
  385. ptimer_transaction_begin(m->timer);
  386. ptimer_stop(m->timer);
  387. /*
  388. * We start in free-running mode, with VALUE at 0xffffffff, and
  389. * in 16-bit counter mode. This means that the ptimer count and
  390. * limit must both be set to 0xffff, so we wrap at 16 bits.
  391. */
  392. ptimer_set_limit(m->timer, 0xffff, 1);
  393. ptimer_set_period_from_clock(m->timer, m->parent->timclk,
  394. cmsdk_dualtimermod_divisor(m));
  395. ptimer_transaction_commit(m->timer);
  396. }
  397. static void cmsdk_apb_dualtimer_reset(DeviceState *dev)
  398. {
  399. CMSDKAPBDualTimer *s = CMSDK_APB_DUALTIMER(dev);
  400. int i;
  401. trace_cmsdk_apb_dualtimer_reset();
  402. for (i = 0; i < ARRAY_SIZE(s->timermod); i++) {
  403. cmsdk_dualtimermod_reset(&s->timermod[i]);
  404. }
  405. s->timeritcr = 0;
  406. s->timeritop = 0;
  407. }
  408. static void cmsdk_apb_dualtimer_clk_update(void *opaque, ClockEvent event)
  409. {
  410. CMSDKAPBDualTimer *s = CMSDK_APB_DUALTIMER(opaque);
  411. int i;
  412. for (i = 0; i < ARRAY_SIZE(s->timermod); i++) {
  413. CMSDKAPBDualTimerModule *m = &s->timermod[i];
  414. ptimer_transaction_begin(m->timer);
  415. ptimer_set_period_from_clock(m->timer, m->parent->timclk,
  416. cmsdk_dualtimermod_divisor(m));
  417. ptimer_transaction_commit(m->timer);
  418. }
  419. }
  420. static void cmsdk_apb_dualtimer_init(Object *obj)
  421. {
  422. SysBusDevice *sbd = SYS_BUS_DEVICE(obj);
  423. CMSDKAPBDualTimer *s = CMSDK_APB_DUALTIMER(obj);
  424. int i;
  425. memory_region_init_io(&s->iomem, obj, &cmsdk_apb_dualtimer_ops,
  426. s, "cmsdk-apb-dualtimer", 0x1000);
  427. sysbus_init_mmio(sbd, &s->iomem);
  428. sysbus_init_irq(sbd, &s->timerintc);
  429. for (i = 0; i < ARRAY_SIZE(s->timermod); i++) {
  430. sysbus_init_irq(sbd, &s->timermod[i].timerint);
  431. }
  432. s->timclk = qdev_init_clock_in(DEVICE(s), "TIMCLK",
  433. cmsdk_apb_dualtimer_clk_update, s,
  434. ClockUpdate);
  435. }
  436. static void cmsdk_apb_dualtimer_realize(DeviceState *dev, Error **errp)
  437. {
  438. CMSDKAPBDualTimer *s = CMSDK_APB_DUALTIMER(dev);
  439. int i;
  440. if (!clock_has_source(s->timclk)) {
  441. error_setg(errp, "CMSDK APB dualtimer: TIMCLK clock must be connected");
  442. return;
  443. }
  444. for (i = 0; i < ARRAY_SIZE(s->timermod); i++) {
  445. CMSDKAPBDualTimerModule *m = &s->timermod[i];
  446. m->parent = s;
  447. m->timer = ptimer_init(cmsdk_dualtimermod_tick, m,
  448. PTIMER_POLICY_WRAP_AFTER_ONE_PERIOD |
  449. PTIMER_POLICY_TRIGGER_ONLY_ON_DECREMENT |
  450. PTIMER_POLICY_NO_IMMEDIATE_RELOAD |
  451. PTIMER_POLICY_NO_COUNTER_ROUND_DOWN);
  452. }
  453. }
  454. static const VMStateDescription cmsdk_dualtimermod_vmstate = {
  455. .name = "cmsdk-apb-dualtimer-module",
  456. .version_id = 1,
  457. .minimum_version_id = 1,
  458. .fields = (VMStateField[]) {
  459. VMSTATE_PTIMER(timer, CMSDKAPBDualTimerModule),
  460. VMSTATE_UINT32(load, CMSDKAPBDualTimerModule),
  461. VMSTATE_UINT32(value, CMSDKAPBDualTimerModule),
  462. VMSTATE_UINT32(control, CMSDKAPBDualTimerModule),
  463. VMSTATE_UINT32(intstatus, CMSDKAPBDualTimerModule),
  464. VMSTATE_END_OF_LIST()
  465. }
  466. };
  467. static const VMStateDescription cmsdk_apb_dualtimer_vmstate = {
  468. .name = "cmsdk-apb-dualtimer",
  469. .version_id = 2,
  470. .minimum_version_id = 2,
  471. .fields = (VMStateField[]) {
  472. VMSTATE_CLOCK(timclk, CMSDKAPBDualTimer),
  473. VMSTATE_STRUCT_ARRAY(timermod, CMSDKAPBDualTimer,
  474. CMSDK_APB_DUALTIMER_NUM_MODULES,
  475. 1, cmsdk_dualtimermod_vmstate,
  476. CMSDKAPBDualTimerModule),
  477. VMSTATE_UINT32(timeritcr, CMSDKAPBDualTimer),
  478. VMSTATE_UINT32(timeritop, CMSDKAPBDualTimer),
  479. VMSTATE_END_OF_LIST()
  480. }
  481. };
  482. static void cmsdk_apb_dualtimer_class_init(ObjectClass *klass, void *data)
  483. {
  484. DeviceClass *dc = DEVICE_CLASS(klass);
  485. dc->realize = cmsdk_apb_dualtimer_realize;
  486. dc->vmsd = &cmsdk_apb_dualtimer_vmstate;
  487. dc->reset = cmsdk_apb_dualtimer_reset;
  488. }
  489. static const TypeInfo cmsdk_apb_dualtimer_info = {
  490. .name = TYPE_CMSDK_APB_DUALTIMER,
  491. .parent = TYPE_SYS_BUS_DEVICE,
  492. .instance_size = sizeof(CMSDKAPBDualTimer),
  493. .instance_init = cmsdk_apb_dualtimer_init,
  494. .class_init = cmsdk_apb_dualtimer_class_init,
  495. };
  496. static void cmsdk_apb_dualtimer_register_types(void)
  497. {
  498. type_register_static(&cmsdk_apb_dualtimer_info);
  499. }
  500. type_init(cmsdk_apb_dualtimer_register_types);