balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Intel XScale PXA255/270 OS Timers. |
| 3 | * |
| 4 | * Copyright (c) 2006 Openedhand Ltd. |
| 5 | * Copyright (c) 2006 Thorsten Zitterell |
| 6 | * |
Matthew Fernandez | 8e31bf3 | 2011-06-26 12:21:35 +1000 | [diff] [blame] | 7 | * This code is licensed under the GPL. |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 8 | */ |
| 9 | |
Peter Maydell | 8ef94f0 | 2016-01-26 18:17:05 +0000 | [diff] [blame] | 10 | #include "qemu/osdep.h" |
Markus Armbruster | 64552b6 | 2019-08-12 07:23:42 +0200 | [diff] [blame] | 11 | #include "hw/irq.h" |
Markus Armbruster | a27bd6c | 2019-08-12 07:23:51 +0200 | [diff] [blame] | 12 | #include "hw/qdev-properties.h" |
Paolo Bonzini | 1de7afc | 2012-12-17 18:20:00 +0100 | [diff] [blame] | 13 | #include "qemu/timer.h" |
Markus Armbruster | 54d3123 | 2019-08-12 07:23:59 +0200 | [diff] [blame] | 14 | #include "sysemu/runstate.h" |
Paolo Bonzini | 0d09e41 | 2013-02-05 17:06:20 +0100 | [diff] [blame] | 15 | #include "hw/arm/pxa.h" |
Paolo Bonzini | 83c9f4c | 2013-02-04 15:40:22 +0100 | [diff] [blame] | 16 | #include "hw/sysbus.h" |
Markus Armbruster | d645427 | 2019-08-12 07:23:45 +0200 | [diff] [blame] | 17 | #include "migration/vmstate.h" |
Philippe Mathieu-Daudé | 2ba63e4 | 2018-01-11 13:25:38 +0000 | [diff] [blame] | 18 | #include "qemu/log.h" |
Markus Armbruster | 0b8fa32 | 2019-05-23 16:35:07 +0200 | [diff] [blame] | 19 | #include "qemu/module.h" |
Eduardo Habkost | db1015e | 2020-09-03 16:43:22 -0400 | [diff] [blame] | 20 | #include "qom/object.h" |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 21 | |
| 22 | #define OSMR0 0x00 |
| 23 | #define OSMR1 0x04 |
| 24 | #define OSMR2 0x08 |
| 25 | #define OSMR3 0x0c |
| 26 | #define OSMR4 0x80 |
| 27 | #define OSMR5 0x84 |
| 28 | #define OSMR6 0x88 |
| 29 | #define OSMR7 0x8c |
| 30 | #define OSMR8 0x90 |
| 31 | #define OSMR9 0x94 |
| 32 | #define OSMR10 0x98 |
| 33 | #define OSMR11 0x9c |
| 34 | #define OSCR 0x10 /* OS Timer Count */ |
| 35 | #define OSCR4 0x40 |
| 36 | #define OSCR5 0x44 |
| 37 | #define OSCR6 0x48 |
| 38 | #define OSCR7 0x4c |
| 39 | #define OSCR8 0x50 |
| 40 | #define OSCR9 0x54 |
| 41 | #define OSCR10 0x58 |
| 42 | #define OSCR11 0x5c |
| 43 | #define OSSR 0x14 /* Timer status register */ |
| 44 | #define OWER 0x18 |
| 45 | #define OIER 0x1c /* Interrupt enable register 3-0 to E3-E0 */ |
| 46 | #define OMCR4 0xc0 /* OS Match Control registers */ |
| 47 | #define OMCR5 0xc4 |
| 48 | #define OMCR6 0xc8 |
| 49 | #define OMCR7 0xcc |
| 50 | #define OMCR8 0xd0 |
| 51 | #define OMCR9 0xd4 |
| 52 | #define OMCR10 0xd8 |
| 53 | #define OMCR11 0xdc |
| 54 | #define OSNR 0x20 |
| 55 | |
| 56 | #define PXA25X_FREQ 3686400 /* 3.6864 MHz */ |
| 57 | #define PXA27X_FREQ 3250000 /* 3.25 MHz */ |
| 58 | |
| 59 | static int pxa2xx_timer4_freq[8] = { |
| 60 | [0] = 0, |
| 61 | [1] = 32768, |
| 62 | [2] = 1000, |
| 63 | [3] = 1, |
| 64 | [4] = 1000000, |
| 65 | /* [5] is the "Externally supplied clock". Assign if necessary. */ |
| 66 | [5 ... 7] = 0, |
| 67 | }; |
| 68 | |
Andreas Färber | feea436 | 2013-07-27 15:20:20 +0200 | [diff] [blame] | 69 | #define TYPE_PXA2XX_TIMER "pxa2xx-timer" |
Eduardo Habkost | 8063396 | 2020-09-16 14:25:19 -0400 | [diff] [blame] | 70 | OBJECT_DECLARE_SIMPLE_TYPE(PXA2xxTimerInfo, PXA2XX_TIMER) |
Andreas Färber | feea436 | 2013-07-27 15:20:20 +0200 | [diff] [blame] | 71 | |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 72 | |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 73 | typedef struct { |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 74 | uint32_t value; |
Andrzej Zaborowski | 5251d19 | 2011-03-03 14:24:25 +0100 | [diff] [blame] | 75 | qemu_irq irq; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 76 | QEMUTimer *qtimer; |
| 77 | int num; |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 78 | PXA2xxTimerInfo *info; |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 79 | } PXA2xxTimer0; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 80 | |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 81 | typedef struct { |
| 82 | PXA2xxTimer0 tm; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 83 | int32_t oldclock; |
| 84 | int32_t clock; |
| 85 | uint64_t lastload; |
| 86 | uint32_t freq; |
| 87 | uint32_t control; |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 88 | } PXA2xxTimer4; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 89 | |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 90 | struct PXA2xxTimerInfo { |
Andreas Färber | feea436 | 2013-07-27 15:20:20 +0200 | [diff] [blame] | 91 | SysBusDevice parent_obj; |
| 92 | |
Benoît Canet | b755bde | 2011-10-30 14:50:16 +0100 | [diff] [blame] | 93 | MemoryRegion iomem; |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 94 | uint32_t flags; |
| 95 | |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 96 | int32_t clock; |
| 97 | int32_t oldclock; |
| 98 | uint64_t lastload; |
| 99 | uint32_t freq; |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 100 | PXA2xxTimer0 timer[4]; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 101 | uint32_t events; |
| 102 | uint32_t irq_enabled; |
| 103 | uint32_t reset3; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 104 | uint32_t snapshot; |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 105 | |
Dmitry Eremin-Solenikov | 4ff927c | 2011-03-04 03:40:59 +0300 | [diff] [blame] | 106 | qemu_irq irq4; |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 107 | PXA2xxTimer4 tm4[8]; |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 108 | }; |
| 109 | |
| 110 | #define PXA2XX_TIMER_HAVE_TM4 0 |
| 111 | |
| 112 | static inline int pxa2xx_timer_has_tm4(PXA2xxTimerInfo *s) |
| 113 | { |
| 114 | return s->flags & (1 << PXA2XX_TIMER_HAVE_TM4); |
| 115 | } |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 116 | |
| 117 | static void pxa2xx_timer_update(void *opaque, uint64_t now_qemu) |
| 118 | { |
Dmitry Eremin-Solenikov | d353eb4 | 2011-02-20 16:50:33 +0300 | [diff] [blame] | 119 | PXA2xxTimerInfo *s = (PXA2xxTimerInfo *) opaque; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 120 | int i; |
| 121 | uint32_t now_vm; |
| 122 | uint64_t new_qemu; |
| 123 | |
| 124 | now_vm = s->clock + |
Rutuja Shah | 73bcb24 | 2016-03-21 21:32:30 +0530 | [diff] [blame] | 125 | muldiv64(now_qemu - s->lastload, s->freq, NANOSECONDS_PER_SECOND); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 126 | |
| 127 | for (i = 0; i < 4; i ++) { |
| 128 | new_qemu = now_qemu + muldiv64((uint32_t) (s->timer[i].value - now_vm), |
Rutuja Shah | 73bcb24 | 2016-03-21 21:32:30 +0530 | [diff] [blame] | 129 | NANOSECONDS_PER_SECOND, s->freq); |
Alex Bligh | bc72ad6 | 2013-08-21 16:03:08 +0100 | [diff] [blame] | 130 | timer_mod(s->timer[i].qtimer, new_qemu); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 131 | } |
| 132 | } |
| 133 | |
| 134 | static void pxa2xx_timer_update4(void *opaque, uint64_t now_qemu, int n) |
| 135 | { |
Dmitry Eremin-Solenikov | d353eb4 | 2011-02-20 16:50:33 +0300 | [diff] [blame] | 136 | PXA2xxTimerInfo *s = (PXA2xxTimerInfo *) opaque; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 137 | uint32_t now_vm; |
| 138 | uint64_t new_qemu; |
| 139 | static const int counters[8] = { 0, 0, 0, 0, 4, 4, 6, 6 }; |
| 140 | int counter; |
| 141 | |
Philippe Mathieu-Daudé | e702fba | 2020-04-22 15:31:52 +0200 | [diff] [blame] | 142 | assert(n < ARRAY_SIZE(counters)); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 143 | if (s->tm4[n].control & (1 << 7)) |
| 144 | counter = n; |
| 145 | else |
| 146 | counter = counters[n]; |
| 147 | |
| 148 | if (!s->tm4[counter].freq) { |
Alex Bligh | bc72ad6 | 2013-08-21 16:03:08 +0100 | [diff] [blame] | 149 | timer_del(s->tm4[n].tm.qtimer); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 150 | return; |
| 151 | } |
| 152 | |
| 153 | now_vm = s->tm4[counter].clock + muldiv64(now_qemu - |
| 154 | s->tm4[counter].lastload, |
Rutuja Shah | 73bcb24 | 2016-03-21 21:32:30 +0530 | [diff] [blame] | 155 | s->tm4[counter].freq, NANOSECONDS_PER_SECOND); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 156 | |
balrog | 3bdd58a | 2007-05-08 22:51:00 +0000 | [diff] [blame] | 157 | new_qemu = now_qemu + muldiv64((uint32_t) (s->tm4[n].tm.value - now_vm), |
Rutuja Shah | 73bcb24 | 2016-03-21 21:32:30 +0530 | [diff] [blame] | 158 | NANOSECONDS_PER_SECOND, s->tm4[counter].freq); |
Alex Bligh | bc72ad6 | 2013-08-21 16:03:08 +0100 | [diff] [blame] | 159 | timer_mod(s->tm4[n].tm.qtimer, new_qemu); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 160 | } |
| 161 | |
Avi Kivity | a8170e5 | 2012-10-23 12:30:10 +0200 | [diff] [blame] | 162 | static uint64_t pxa2xx_timer_read(void *opaque, hwaddr offset, |
Benoît Canet | b755bde | 2011-10-30 14:50:16 +0100 | [diff] [blame] | 163 | unsigned size) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 164 | { |
Dmitry Eremin-Solenikov | d353eb4 | 2011-02-20 16:50:33 +0300 | [diff] [blame] | 165 | PXA2xxTimerInfo *s = (PXA2xxTimerInfo *) opaque; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 166 | int tm = 0; |
| 167 | |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 168 | switch (offset) { |
| 169 | case OSMR3: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 170 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 171 | case OSMR2: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 172 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 173 | case OSMR1: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 174 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 175 | case OSMR0: |
| 176 | return s->timer[tm].value; |
| 177 | case OSMR11: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 178 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 179 | case OSMR10: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 180 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 181 | case OSMR9: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 182 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 183 | case OSMR8: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 184 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 185 | case OSMR7: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 186 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 187 | case OSMR6: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 188 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 189 | case OSMR5: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 190 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 191 | case OSMR4: |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 192 | if (!pxa2xx_timer_has_tm4(s)) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 193 | goto badreg; |
balrog | 3bdd58a | 2007-05-08 22:51:00 +0000 | [diff] [blame] | 194 | return s->tm4[tm].tm.value; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 195 | case OSCR: |
Alex Bligh | bc72ad6 | 2013-08-21 16:03:08 +0100 | [diff] [blame] | 196 | return s->clock + muldiv64(qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL) - |
Rutuja Shah | 73bcb24 | 2016-03-21 21:32:30 +0530 | [diff] [blame] | 197 | s->lastload, s->freq, NANOSECONDS_PER_SECOND); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 198 | case OSCR11: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 199 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 200 | case OSCR10: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 201 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 202 | case OSCR9: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 203 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 204 | case OSCR8: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 205 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 206 | case OSCR7: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 207 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 208 | case OSCR6: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 209 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 210 | case OSCR5: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 211 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 212 | case OSCR4: |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 213 | if (!pxa2xx_timer_has_tm4(s)) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 214 | goto badreg; |
| 215 | |
| 216 | if ((tm == 9 - 4 || tm == 11 - 4) && (s->tm4[tm].control & (1 << 9))) { |
| 217 | if (s->tm4[tm - 1].freq) |
| 218 | s->snapshot = s->tm4[tm - 1].clock + muldiv64( |
Alex Bligh | bc72ad6 | 2013-08-21 16:03:08 +0100 | [diff] [blame] | 219 | qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL) - |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 220 | s->tm4[tm - 1].lastload, |
Rutuja Shah | 73bcb24 | 2016-03-21 21:32:30 +0530 | [diff] [blame] | 221 | s->tm4[tm - 1].freq, NANOSECONDS_PER_SECOND); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 222 | else |
| 223 | s->snapshot = s->tm4[tm - 1].clock; |
| 224 | } |
| 225 | |
| 226 | if (!s->tm4[tm].freq) |
| 227 | return s->tm4[tm].clock; |
Rutuja Shah | 73bcb24 | 2016-03-21 21:32:30 +0530 | [diff] [blame] | 228 | return s->tm4[tm].clock + |
| 229 | muldiv64(qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL) - |
| 230 | s->tm4[tm].lastload, s->tm4[tm].freq, |
| 231 | NANOSECONDS_PER_SECOND); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 232 | case OIER: |
| 233 | return s->irq_enabled; |
| 234 | case OSSR: /* Status register */ |
| 235 | return s->events; |
| 236 | case OWER: |
| 237 | return s->reset3; |
| 238 | case OMCR11: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 239 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 240 | case OMCR10: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 241 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 242 | case OMCR9: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 243 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 244 | case OMCR8: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 245 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 246 | case OMCR7: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 247 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 248 | case OMCR6: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 249 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 250 | case OMCR5: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 251 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 252 | case OMCR4: |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 253 | if (!pxa2xx_timer_has_tm4(s)) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 254 | goto badreg; |
| 255 | return s->tm4[tm].control; |
| 256 | case OSNR: |
| 257 | return s->snapshot; |
| 258 | default: |
Philippe Mathieu-Daudé | 2ba63e4 | 2018-01-11 13:25:38 +0000 | [diff] [blame] | 259 | qemu_log_mask(LOG_UNIMP, |
| 260 | "%s: unknown register 0x%02" HWADDR_PRIx "\n", |
| 261 | __func__, offset); |
| 262 | break; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 263 | badreg: |
Philippe Mathieu-Daudé | 2ba63e4 | 2018-01-11 13:25:38 +0000 | [diff] [blame] | 264 | qemu_log_mask(LOG_GUEST_ERROR, |
| 265 | "%s: incorrect register 0x%02" HWADDR_PRIx "\n", |
| 266 | __func__, offset); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 267 | } |
| 268 | |
| 269 | return 0; |
| 270 | } |
| 271 | |
Avi Kivity | a8170e5 | 2012-10-23 12:30:10 +0200 | [diff] [blame] | 272 | static void pxa2xx_timer_write(void *opaque, hwaddr offset, |
Benoît Canet | b755bde | 2011-10-30 14:50:16 +0100 | [diff] [blame] | 273 | uint64_t value, unsigned size) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 274 | { |
| 275 | int i, tm = 0; |
Dmitry Eremin-Solenikov | d353eb4 | 2011-02-20 16:50:33 +0300 | [diff] [blame] | 276 | PXA2xxTimerInfo *s = (PXA2xxTimerInfo *) opaque; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 277 | |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 278 | switch (offset) { |
| 279 | case OSMR3: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 280 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 281 | case OSMR2: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 282 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 283 | case OSMR1: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 284 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 285 | case OSMR0: |
| 286 | s->timer[tm].value = value; |
Alex Bligh | bc72ad6 | 2013-08-21 16:03:08 +0100 | [diff] [blame] | 287 | pxa2xx_timer_update(s, qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL)); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 288 | break; |
| 289 | case OSMR11: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 290 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 291 | case OSMR10: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 292 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 293 | case OSMR9: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 294 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 295 | case OSMR8: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 296 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 297 | case OSMR7: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 298 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 299 | case OSMR6: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 300 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 301 | case OSMR5: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 302 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 303 | case OSMR4: |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 304 | if (!pxa2xx_timer_has_tm4(s)) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 305 | goto badreg; |
balrog | 3bdd58a | 2007-05-08 22:51:00 +0000 | [diff] [blame] | 306 | s->tm4[tm].tm.value = value; |
Alex Bligh | bc72ad6 | 2013-08-21 16:03:08 +0100 | [diff] [blame] | 307 | pxa2xx_timer_update4(s, qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL), tm); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 308 | break; |
| 309 | case OSCR: |
| 310 | s->oldclock = s->clock; |
Alex Bligh | bc72ad6 | 2013-08-21 16:03:08 +0100 | [diff] [blame] | 311 | s->lastload = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 312 | s->clock = value; |
| 313 | pxa2xx_timer_update(s, s->lastload); |
| 314 | break; |
| 315 | case OSCR11: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 316 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 317 | case OSCR10: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 318 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 319 | case OSCR9: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 320 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 321 | case OSCR8: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 322 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 323 | case OSCR7: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 324 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 325 | case OSCR6: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 326 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 327 | case OSCR5: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 328 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 329 | case OSCR4: |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 330 | if (!pxa2xx_timer_has_tm4(s)) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 331 | goto badreg; |
| 332 | s->tm4[tm].oldclock = s->tm4[tm].clock; |
Alex Bligh | bc72ad6 | 2013-08-21 16:03:08 +0100 | [diff] [blame] | 333 | s->tm4[tm].lastload = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 334 | s->tm4[tm].clock = value; |
| 335 | pxa2xx_timer_update4(s, s->tm4[tm].lastload, tm); |
| 336 | break; |
| 337 | case OIER: |
| 338 | s->irq_enabled = value & 0xfff; |
| 339 | break; |
| 340 | case OSSR: /* Status register */ |
Andrzej Zaborowski | 8034ce7 | 2011-03-10 03:31:02 +0100 | [diff] [blame] | 341 | value &= s->events; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 342 | s->events &= ~value; |
Andrzej Zaborowski | 8034ce7 | 2011-03-10 03:31:02 +0100 | [diff] [blame] | 343 | for (i = 0; i < 4; i ++, value >>= 1) |
| 344 | if (value & 1) |
Andrzej Zaborowski | 5251d19 | 2011-03-03 14:24:25 +0100 | [diff] [blame] | 345 | qemu_irq_lower(s->timer[i].irq); |
Andrzej Zaborowski | 8034ce7 | 2011-03-10 03:31:02 +0100 | [diff] [blame] | 346 | if (pxa2xx_timer_has_tm4(s) && !(s->events & 0xff0) && value) |
| 347 | qemu_irq_lower(s->irq4); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 348 | break; |
| 349 | case OWER: /* XXX: Reset on OSMR3 match? */ |
| 350 | s->reset3 = value; |
| 351 | break; |
| 352 | case OMCR7: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 353 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 354 | case OMCR6: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 355 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 356 | case OMCR5: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 357 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 358 | case OMCR4: |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 359 | if (!pxa2xx_timer_has_tm4(s)) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 360 | goto badreg; |
| 361 | s->tm4[tm].control = value & 0x0ff; |
| 362 | /* XXX Stop if running (shouldn't happen) */ |
| 363 | if ((value & (1 << 7)) || tm == 0) |
| 364 | s->tm4[tm].freq = pxa2xx_timer4_freq[value & 7]; |
| 365 | else { |
| 366 | s->tm4[tm].freq = 0; |
Alex Bligh | bc72ad6 | 2013-08-21 16:03:08 +0100 | [diff] [blame] | 367 | pxa2xx_timer_update4(s, qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL), tm); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 368 | } |
| 369 | break; |
| 370 | case OMCR11: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 371 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 372 | case OMCR10: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 373 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 374 | case OMCR9: tm ++; |
Peter Maydell | de16017 | 2013-01-21 12:50:56 +0000 | [diff] [blame] | 375 | /* fall through */ |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 376 | case OMCR8: tm += 4; |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 377 | if (!pxa2xx_timer_has_tm4(s)) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 378 | goto badreg; |
| 379 | s->tm4[tm].control = value & 0x3ff; |
| 380 | /* XXX Stop if running (shouldn't happen) */ |
| 381 | if ((value & (1 << 7)) || !(tm & 1)) |
| 382 | s->tm4[tm].freq = |
| 383 | pxa2xx_timer4_freq[(value & (1 << 8)) ? 0 : (value & 7)]; |
| 384 | else { |
| 385 | s->tm4[tm].freq = 0; |
Alex Bligh | bc72ad6 | 2013-08-21 16:03:08 +0100 | [diff] [blame] | 386 | pxa2xx_timer_update4(s, qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL), tm); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 387 | } |
| 388 | break; |
| 389 | default: |
Philippe Mathieu-Daudé | 2ba63e4 | 2018-01-11 13:25:38 +0000 | [diff] [blame] | 390 | qemu_log_mask(LOG_UNIMP, |
| 391 | "%s: unknown register 0x%02" HWADDR_PRIx " " |
| 392 | "(value 0x%08" PRIx64 ")\n", __func__, offset, value); |
| 393 | break; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 394 | badreg: |
Philippe Mathieu-Daudé | 2ba63e4 | 2018-01-11 13:25:38 +0000 | [diff] [blame] | 395 | qemu_log_mask(LOG_GUEST_ERROR, |
| 396 | "%s: incorrect register 0x%02" HWADDR_PRIx " " |
| 397 | "(value 0x%08" PRIx64 ")\n", __func__, offset, value); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 398 | } |
| 399 | } |
| 400 | |
Benoît Canet | b755bde | 2011-10-30 14:50:16 +0100 | [diff] [blame] | 401 | static const MemoryRegionOps pxa2xx_timer_ops = { |
| 402 | .read = pxa2xx_timer_read, |
| 403 | .write = pxa2xx_timer_write, |
| 404 | .endianness = DEVICE_NATIVE_ENDIAN, |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 405 | }; |
| 406 | |
| 407 | static void pxa2xx_timer_tick(void *opaque) |
| 408 | { |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 409 | PXA2xxTimer0 *t = (PXA2xxTimer0 *) opaque; |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 410 | PXA2xxTimerInfo *i = t->info; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 411 | |
| 412 | if (i->irq_enabled & (1 << t->num)) { |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 413 | i->events |= 1 << t->num; |
Andrzej Zaborowski | 5251d19 | 2011-03-03 14:24:25 +0100 | [diff] [blame] | 414 | qemu_irq_raise(t->irq); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 415 | } |
| 416 | |
| 417 | if (t->num == 3) |
| 418 | if (i->reset3 & 1) { |
| 419 | i->reset3 = 0; |
Eric Blake | cf83f14 | 2017-05-15 16:41:13 -0500 | [diff] [blame] | 420 | qemu_system_reset_request(SHUTDOWN_CAUSE_GUEST_RESET); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 421 | } |
| 422 | } |
| 423 | |
| 424 | static void pxa2xx_timer_tick4(void *opaque) |
| 425 | { |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 426 | PXA2xxTimer4 *t = (PXA2xxTimer4 *) opaque; |
Dmitry Eremin-Solenikov | d353eb4 | 2011-02-20 16:50:33 +0300 | [diff] [blame] | 427 | PXA2xxTimerInfo *i = (PXA2xxTimerInfo *) t->tm.info; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 428 | |
balrog | 3bdd58a | 2007-05-08 22:51:00 +0000 | [diff] [blame] | 429 | pxa2xx_timer_tick(&t->tm); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 430 | if (t->control & (1 << 3)) |
| 431 | t->clock = 0; |
| 432 | if (t->control & (1 << 6)) |
Alex Bligh | bc72ad6 | 2013-08-21 16:03:08 +0100 | [diff] [blame] | 433 | pxa2xx_timer_update4(i, qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL), t->tm.num - 4); |
Dmitry Eremin-Solenikov | 4ff927c | 2011-03-04 03:40:59 +0300 | [diff] [blame] | 434 | if (i->events & 0xff0) |
| 435 | qemu_irq_raise(i->irq4); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 436 | } |
| 437 | |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 438 | static int pxa25x_timer_post_load(void *opaque, int version_id) |
balrog | aa941b9 | 2007-05-24 18:50:09 +0000 | [diff] [blame] | 439 | { |
Dmitry Eremin-Solenikov | d353eb4 | 2011-02-20 16:50:33 +0300 | [diff] [blame] | 440 | PXA2xxTimerInfo *s = (PXA2xxTimerInfo *) opaque; |
balrog | aa941b9 | 2007-05-24 18:50:09 +0000 | [diff] [blame] | 441 | int64_t now; |
| 442 | int i; |
| 443 | |
Alex Bligh | bc72ad6 | 2013-08-21 16:03:08 +0100 | [diff] [blame] | 444 | now = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL); |
balrog | aa941b9 | 2007-05-24 18:50:09 +0000 | [diff] [blame] | 445 | pxa2xx_timer_update(s, now); |
| 446 | |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 447 | if (pxa2xx_timer_has_tm4(s)) |
| 448 | for (i = 0; i < 8; i ++) |
balrog | aa941b9 | 2007-05-24 18:50:09 +0000 | [diff] [blame] | 449 | pxa2xx_timer_update4(s, now, i); |
balrog | aa941b9 | 2007-05-24 18:50:09 +0000 | [diff] [blame] | 450 | |
| 451 | return 0; |
| 452 | } |
| 453 | |
xiaoqiang.zhao | 5d83e34 | 2016-02-18 14:16:21 +0000 | [diff] [blame] | 454 | static void pxa2xx_timer_init(Object *obj) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 455 | { |
xiaoqiang.zhao | 5d83e34 | 2016-02-18 14:16:21 +0000 | [diff] [blame] | 456 | PXA2xxTimerInfo *s = PXA2XX_TIMER(obj); |
| 457 | SysBusDevice *dev = SYS_BUS_DEVICE(obj); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 458 | |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 459 | s->irq_enabled = 0; |
| 460 | s->oldclock = 0; |
| 461 | s->clock = 0; |
Alex Bligh | bc72ad6 | 2013-08-21 16:03:08 +0100 | [diff] [blame] | 462 | s->lastload = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 463 | s->reset3 = 0; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 464 | |
xiaoqiang.zhao | 5d83e34 | 2016-02-18 14:16:21 +0000 | [diff] [blame] | 465 | memory_region_init_io(&s->iomem, obj, &pxa2xx_timer_ops, s, |
| 466 | "pxa2xx-timer", 0x00001000); |
| 467 | sysbus_init_mmio(dev, &s->iomem); |
| 468 | } |
| 469 | |
| 470 | static void pxa2xx_timer_realize(DeviceState *dev, Error **errp) |
| 471 | { |
| 472 | PXA2xxTimerInfo *s = PXA2XX_TIMER(dev); |
| 473 | SysBusDevice *sbd = SYS_BUS_DEVICE(dev); |
| 474 | int i; |
| 475 | |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 476 | for (i = 0; i < 4; i ++) { |
| 477 | s->timer[i].value = 0; |
xiaoqiang.zhao | 5d83e34 | 2016-02-18 14:16:21 +0000 | [diff] [blame] | 478 | sysbus_init_irq(sbd, &s->timer[i].irq); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 479 | s->timer[i].info = s; |
| 480 | s->timer[i].num = i; |
Alex Bligh | bc72ad6 | 2013-08-21 16:03:08 +0100 | [diff] [blame] | 481 | s->timer[i].qtimer = timer_new_ns(QEMU_CLOCK_VIRTUAL, |
xiaoqiang.zhao | 5d83e34 | 2016-02-18 14:16:21 +0000 | [diff] [blame] | 482 | pxa2xx_timer_tick, &s->timer[i]); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 483 | } |
xiaoqiang.zhao | 5d83e34 | 2016-02-18 14:16:21 +0000 | [diff] [blame] | 484 | |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 485 | if (s->flags & (1 << PXA2XX_TIMER_HAVE_TM4)) { |
xiaoqiang.zhao | 5d83e34 | 2016-02-18 14:16:21 +0000 | [diff] [blame] | 486 | sysbus_init_irq(sbd, &s->irq4); |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 487 | |
| 488 | for (i = 0; i < 8; i ++) { |
| 489 | s->tm4[i].tm.value = 0; |
| 490 | s->tm4[i].tm.info = s; |
| 491 | s->tm4[i].tm.num = i + 4; |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 492 | s->tm4[i].freq = 0; |
| 493 | s->tm4[i].control = 0x0; |
Alex Bligh | bc72ad6 | 2013-08-21 16:03:08 +0100 | [diff] [blame] | 494 | s->tm4[i].tm.qtimer = timer_new_ns(QEMU_CLOCK_VIRTUAL, |
xiaoqiang.zhao | 5d83e34 | 2016-02-18 14:16:21 +0000 | [diff] [blame] | 495 | pxa2xx_timer_tick4, &s->tm4[i]); |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 496 | } |
| 497 | } |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 498 | } |
| 499 | |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 500 | static const VMStateDescription vmstate_pxa2xx_timer0_regs = { |
| 501 | .name = "pxa2xx_timer0", |
Andrzej Zaborowski | 8034ce7 | 2011-03-10 03:31:02 +0100 | [diff] [blame] | 502 | .version_id = 2, |
| 503 | .minimum_version_id = 2, |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 504 | .fields = (VMStateField[]) { |
| 505 | VMSTATE_UINT32(value, PXA2xxTimer0), |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 506 | VMSTATE_END_OF_LIST(), |
| 507 | }, |
| 508 | }; |
| 509 | |
| 510 | static const VMStateDescription vmstate_pxa2xx_timer4_regs = { |
| 511 | .name = "pxa2xx_timer4", |
| 512 | .version_id = 1, |
| 513 | .minimum_version_id = 1, |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 514 | .fields = (VMStateField[]) { |
| 515 | VMSTATE_STRUCT(tm, PXA2xxTimer4, 1, |
| 516 | vmstate_pxa2xx_timer0_regs, PXA2xxTimer0), |
| 517 | VMSTATE_INT32(oldclock, PXA2xxTimer4), |
| 518 | VMSTATE_INT32(clock, PXA2xxTimer4), |
| 519 | VMSTATE_UINT64(lastload, PXA2xxTimer4), |
| 520 | VMSTATE_UINT32(freq, PXA2xxTimer4), |
| 521 | VMSTATE_UINT32(control, PXA2xxTimer4), |
| 522 | VMSTATE_END_OF_LIST(), |
| 523 | }, |
| 524 | }; |
| 525 | |
| 526 | static bool pxa2xx_timer_has_tm4_test(void *opaque, int version_id) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 527 | { |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 528 | return pxa2xx_timer_has_tm4(opaque); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 529 | } |
| 530 | |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 531 | static const VMStateDescription vmstate_pxa2xx_timer_regs = { |
| 532 | .name = "pxa2xx_timer", |
| 533 | .version_id = 1, |
| 534 | .minimum_version_id = 1, |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 535 | .post_load = pxa25x_timer_post_load, |
| 536 | .fields = (VMStateField[]) { |
| 537 | VMSTATE_INT32(clock, PXA2xxTimerInfo), |
| 538 | VMSTATE_INT32(oldclock, PXA2xxTimerInfo), |
| 539 | VMSTATE_UINT64(lastload, PXA2xxTimerInfo), |
| 540 | VMSTATE_STRUCT_ARRAY(timer, PXA2xxTimerInfo, 4, 1, |
| 541 | vmstate_pxa2xx_timer0_regs, PXA2xxTimer0), |
| 542 | VMSTATE_UINT32(events, PXA2xxTimerInfo), |
| 543 | VMSTATE_UINT32(irq_enabled, PXA2xxTimerInfo), |
| 544 | VMSTATE_UINT32(reset3, PXA2xxTimerInfo), |
| 545 | VMSTATE_UINT32(snapshot, PXA2xxTimerInfo), |
| 546 | VMSTATE_STRUCT_ARRAY_TEST(tm4, PXA2xxTimerInfo, 8, |
| 547 | pxa2xx_timer_has_tm4_test, 0, |
| 548 | vmstate_pxa2xx_timer4_regs, PXA2xxTimer4), |
| 549 | VMSTATE_END_OF_LIST(), |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 550 | } |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 551 | }; |
| 552 | |
Anthony Liguori | 999e12b | 2012-01-24 13:12:29 -0600 | [diff] [blame] | 553 | static Property pxa25x_timer_dev_properties[] = { |
| 554 | DEFINE_PROP_UINT32("freq", PXA2xxTimerInfo, freq, PXA25X_FREQ), |
| 555 | DEFINE_PROP_BIT("tm4", PXA2xxTimerInfo, flags, |
Andreas Färber | feea436 | 2013-07-27 15:20:20 +0200 | [diff] [blame] | 556 | PXA2XX_TIMER_HAVE_TM4, false), |
Anthony Liguori | 999e12b | 2012-01-24 13:12:29 -0600 | [diff] [blame] | 557 | DEFINE_PROP_END_OF_LIST(), |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 558 | }; |
| 559 | |
Anthony Liguori | 999e12b | 2012-01-24 13:12:29 -0600 | [diff] [blame] | 560 | static void pxa25x_timer_dev_class_init(ObjectClass *klass, void *data) |
| 561 | { |
Anthony Liguori | 39bffca | 2011-12-07 21:34:16 -0600 | [diff] [blame] | 562 | DeviceClass *dc = DEVICE_CLASS(klass); |
Anthony Liguori | 999e12b | 2012-01-24 13:12:29 -0600 | [diff] [blame] | 563 | |
Anthony Liguori | 39bffca | 2011-12-07 21:34:16 -0600 | [diff] [blame] | 564 | dc->desc = "PXA25x timer"; |
Marc-André Lureau | 4f67d30 | 2020-01-10 19:30:32 +0400 | [diff] [blame] | 565 | device_class_set_props(dc, pxa25x_timer_dev_properties); |
Anthony Liguori | 999e12b | 2012-01-24 13:12:29 -0600 | [diff] [blame] | 566 | } |
| 567 | |
Andreas Färber | 8c43a6f | 2013-01-10 16:19:07 +0100 | [diff] [blame] | 568 | static const TypeInfo pxa25x_timer_dev_info = { |
Anthony Liguori | 39bffca | 2011-12-07 21:34:16 -0600 | [diff] [blame] | 569 | .name = "pxa25x-timer", |
Andreas Färber | feea436 | 2013-07-27 15:20:20 +0200 | [diff] [blame] | 570 | .parent = TYPE_PXA2XX_TIMER, |
Anthony Liguori | 39bffca | 2011-12-07 21:34:16 -0600 | [diff] [blame] | 571 | .instance_size = sizeof(PXA2xxTimerInfo), |
| 572 | .class_init = pxa25x_timer_dev_class_init, |
Anthony Liguori | 999e12b | 2012-01-24 13:12:29 -0600 | [diff] [blame] | 573 | }; |
| 574 | |
| 575 | static Property pxa27x_timer_dev_properties[] = { |
| 576 | DEFINE_PROP_UINT32("freq", PXA2xxTimerInfo, freq, PXA27X_FREQ), |
| 577 | DEFINE_PROP_BIT("tm4", PXA2xxTimerInfo, flags, |
Andreas Färber | feea436 | 2013-07-27 15:20:20 +0200 | [diff] [blame] | 578 | PXA2XX_TIMER_HAVE_TM4, true), |
Anthony Liguori | 999e12b | 2012-01-24 13:12:29 -0600 | [diff] [blame] | 579 | DEFINE_PROP_END_OF_LIST(), |
| 580 | }; |
| 581 | |
| 582 | static void pxa27x_timer_dev_class_init(ObjectClass *klass, void *data) |
| 583 | { |
Anthony Liguori | 39bffca | 2011-12-07 21:34:16 -0600 | [diff] [blame] | 584 | DeviceClass *dc = DEVICE_CLASS(klass); |
Anthony Liguori | 999e12b | 2012-01-24 13:12:29 -0600 | [diff] [blame] | 585 | |
Anthony Liguori | 39bffca | 2011-12-07 21:34:16 -0600 | [diff] [blame] | 586 | dc->desc = "PXA27x timer"; |
Marc-André Lureau | 4f67d30 | 2020-01-10 19:30:32 +0400 | [diff] [blame] | 587 | device_class_set_props(dc, pxa27x_timer_dev_properties); |
Anthony Liguori | 999e12b | 2012-01-24 13:12:29 -0600 | [diff] [blame] | 588 | } |
| 589 | |
Andreas Färber | 8c43a6f | 2013-01-10 16:19:07 +0100 | [diff] [blame] | 590 | static const TypeInfo pxa27x_timer_dev_info = { |
Anthony Liguori | 39bffca | 2011-12-07 21:34:16 -0600 | [diff] [blame] | 591 | .name = "pxa27x-timer", |
Andreas Färber | feea436 | 2013-07-27 15:20:20 +0200 | [diff] [blame] | 592 | .parent = TYPE_PXA2XX_TIMER, |
Anthony Liguori | 39bffca | 2011-12-07 21:34:16 -0600 | [diff] [blame] | 593 | .instance_size = sizeof(PXA2xxTimerInfo), |
| 594 | .class_init = pxa27x_timer_dev_class_init, |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 595 | }; |
| 596 | |
Andreas Färber | feea436 | 2013-07-27 15:20:20 +0200 | [diff] [blame] | 597 | static void pxa2xx_timer_class_init(ObjectClass *oc, void *data) |
| 598 | { |
| 599 | DeviceClass *dc = DEVICE_CLASS(oc); |
Andreas Färber | feea436 | 2013-07-27 15:20:20 +0200 | [diff] [blame] | 600 | |
xiaoqiang.zhao | 5d83e34 | 2016-02-18 14:16:21 +0000 | [diff] [blame] | 601 | dc->realize = pxa2xx_timer_realize; |
Andreas Färber | feea436 | 2013-07-27 15:20:20 +0200 | [diff] [blame] | 602 | dc->vmsd = &vmstate_pxa2xx_timer_regs; |
| 603 | } |
| 604 | |
| 605 | static const TypeInfo pxa2xx_timer_type_info = { |
| 606 | .name = TYPE_PXA2XX_TIMER, |
| 607 | .parent = TYPE_SYS_BUS_DEVICE, |
| 608 | .instance_size = sizeof(PXA2xxTimerInfo), |
xiaoqiang.zhao | 5d83e34 | 2016-02-18 14:16:21 +0000 | [diff] [blame] | 609 | .instance_init = pxa2xx_timer_init, |
Andreas Färber | feea436 | 2013-07-27 15:20:20 +0200 | [diff] [blame] | 610 | .abstract = true, |
| 611 | .class_init = pxa2xx_timer_class_init, |
| 612 | }; |
| 613 | |
Andreas Färber | 83f7d43 | 2012-02-09 15:20:55 +0100 | [diff] [blame] | 614 | static void pxa2xx_timer_register_types(void) |
Dmitry Eremin-Solenikov | 797e954 | 2011-03-03 14:14:44 +0100 | [diff] [blame] | 615 | { |
Andreas Färber | feea436 | 2013-07-27 15:20:20 +0200 | [diff] [blame] | 616 | type_register_static(&pxa2xx_timer_type_info); |
Anthony Liguori | 39bffca | 2011-12-07 21:34:16 -0600 | [diff] [blame] | 617 | type_register_static(&pxa25x_timer_dev_info); |
| 618 | type_register_static(&pxa27x_timer_dev_info); |
Andreas Färber | 83f7d43 | 2012-02-09 15:20:55 +0100 | [diff] [blame] | 619 | } |
| 620 | |
| 621 | type_init(pxa2xx_timer_register_types) |