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