2 * QEMU lowRISC Ibex Timer device
4 * Copyright (c) 2021 Western Digital
6 * For details check the documentation here:
7 * https://docs.opentitan.org/hw/ip/rv_timer/doc/
9 * Permission is hereby granted, free of charge, to any person obtaining a copy
10 * of this software and associated documentation files (the "Software"), to deal
11 * in the Software without restriction, including without limitation the rights
12 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
13 * copies of the Software, and to permit persons to whom the Software is
14 * furnished to do so, subject to the following conditions:
16 * The above copyright notice and this permission notice shall be included in
17 * all copies or substantial portions of the Software.
19 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
20 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
21 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
22 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
23 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
24 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
28 #include "qemu/osdep.h"
30 #include "qemu/timer.h"
31 #include "hw/timer/ibex_timer.h"
33 #include "hw/qdev-properties.h"
34 #include "target/riscv/cpu.h"
35 #include "migration/vmstate.h"
37 REG32(ALERT_TEST
, 0x00)
38 FIELD(ALERT_TEST
, FATAL_FAULT
, 0, 1)
40 FIELD(CTRL
, ACTIVE
, 0, 1)
42 FIELD(CFG0
, PRESCALE
, 0, 12)
43 FIELD(CFG0
, STEP
, 16, 8)
46 REG32(COMPARE_LOWER0
, 0x10C)
47 REG32(COMPARE_UPPER0
, 0x110)
48 REG32(INTR_ENABLE
, 0x114)
49 FIELD(INTR_ENABLE
, IE_0
, 0, 1)
50 REG32(INTR_STATE
, 0x118)
51 FIELD(INTR_STATE
, IS_0
, 0, 1)
52 REG32(INTR_TEST
, 0x11C)
53 FIELD(INTR_TEST
, T_0
, 0, 1)
55 static uint64_t cpu_riscv_read_rtc(uint32_t timebase_freq
)
57 return muldiv64(qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL
),
58 timebase_freq
, NANOSECONDS_PER_SECOND
);
61 static void ibex_timer_update_irqs(IbexTimerState
*s
)
63 CPUState
*cs
= qemu_get_cpu(0);
64 RISCVCPU
*cpu
= RISCV_CPU(cs
);
65 uint64_t value
= s
->timer_compare_lower0
|
66 ((uint64_t)s
->timer_compare_upper0
<< 32);
68 uint64_t now
= cpu_riscv_read_rtc(s
->timebase_freq
);
70 if (!(s
->timer_ctrl
& R_CTRL_ACTIVE_MASK
)) {
71 /* Timer isn't active */
75 /* Update the CPUs mtimecmp */
76 cpu
->env
.timecmp
= value
;
78 if (cpu
->env
.timecmp
<= now
) {
80 * If the mtimecmp was in the past raise the interrupt now.
82 qemu_irq_raise(s
->m_timer_irq
);
83 if (s
->timer_intr_enable
& R_INTR_ENABLE_IE_0_MASK
) {
84 s
->timer_intr_state
|= R_INTR_STATE_IS_0_MASK
;
85 qemu_set_irq(s
->irq
, true);
90 /* Setup a timer to trigger the interrupt in the future */
91 qemu_irq_lower(s
->m_timer_irq
);
92 qemu_set_irq(s
->irq
, false);
94 diff
= cpu
->env
.timecmp
- now
;
95 next
= qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL
) +
97 NANOSECONDS_PER_SECOND
,
100 if (next
< qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL
)) {
101 /* We overflowed the timer, just set it as large as we can */
102 timer_mod(cpu
->env
.timer
, 0x7FFFFFFFFFFFFFFF);
104 timer_mod(cpu
->env
.timer
, next
);
108 static void ibex_timer_cb(void *opaque
)
110 IbexTimerState
*s
= opaque
;
112 qemu_irq_raise(s
->m_timer_irq
);
113 if (s
->timer_intr_enable
& R_INTR_ENABLE_IE_0_MASK
) {
114 s
->timer_intr_state
|= R_INTR_STATE_IS_0_MASK
;
115 qemu_set_irq(s
->irq
, true);
119 static void ibex_timer_reset(DeviceState
*dev
)
121 IbexTimerState
*s
= IBEX_TIMER(dev
);
123 CPUState
*cpu
= qemu_get_cpu(0);
124 CPURISCVState
*env
= cpu
->env_ptr
;
125 env
->timer
= timer_new_ns(QEMU_CLOCK_VIRTUAL
,
129 s
->timer_ctrl
= 0x00000000;
130 s
->timer_cfg0
= 0x00010000;
131 s
->timer_compare_lower0
= 0xFFFFFFFF;
132 s
->timer_compare_upper0
= 0xFFFFFFFF;
133 s
->timer_intr_enable
= 0x00000000;
134 s
->timer_intr_state
= 0x00000000;
136 ibex_timer_update_irqs(s
);
139 static uint64_t ibex_timer_read(void *opaque
, hwaddr addr
,
142 IbexTimerState
*s
= opaque
;
143 uint64_t now
= cpu_riscv_read_rtc(s
->timebase_freq
);
144 uint64_t retvalue
= 0;
148 qemu_log_mask(LOG_GUEST_ERROR
,
149 "Attempted to read ALERT_TEST, a write only register");
152 retvalue
= s
->timer_ctrl
;
155 retvalue
= s
->timer_cfg0
;
161 retvalue
= now
>> 32;
163 case R_COMPARE_LOWER0
:
164 retvalue
= s
->timer_compare_lower0
;
166 case R_COMPARE_UPPER0
:
167 retvalue
= s
->timer_compare_upper0
;
170 retvalue
= s
->timer_intr_enable
;
173 retvalue
= s
->timer_intr_state
;
176 qemu_log_mask(LOG_GUEST_ERROR
,
177 "Attempted to read INTR_TEST, a write only register");
180 qemu_log_mask(LOG_GUEST_ERROR
,
181 "%s: Bad offset 0x%"HWADDR_PRIx
"\n", __func__
, addr
);
188 static void ibex_timer_write(void *opaque
, hwaddr addr
,
189 uint64_t val64
, unsigned int size
)
191 IbexTimerState
*s
= opaque
;
192 uint32_t val
= val64
;
196 qemu_log_mask(LOG_UNIMP
, "Alert triggering not supported");
202 qemu_log_mask(LOG_UNIMP
, "Changing prescale or step not supported");
206 qemu_log_mask(LOG_UNIMP
, "Changing timer value is not supported");
209 qemu_log_mask(LOG_UNIMP
, "Changing timer value is not supported");
211 case R_COMPARE_LOWER0
:
212 s
->timer_compare_lower0
= val
;
213 ibex_timer_update_irqs(s
);
215 case R_COMPARE_UPPER0
:
216 s
->timer_compare_upper0
= val
;
217 ibex_timer_update_irqs(s
);
220 s
->timer_intr_enable
= val
;
223 /* Write 1 to clear */
224 s
->timer_intr_state
&= ~val
;
227 if (s
->timer_intr_enable
& val
& R_INTR_ENABLE_IE_0_MASK
) {
228 s
->timer_intr_state
|= R_INTR_STATE_IS_0_MASK
;
229 qemu_set_irq(s
->irq
, true);
233 qemu_log_mask(LOG_GUEST_ERROR
,
234 "%s: Bad offset 0x%"HWADDR_PRIx
"\n", __func__
, addr
);
238 static const MemoryRegionOps ibex_timer_ops
= {
239 .read
= ibex_timer_read
,
240 .write
= ibex_timer_write
,
241 .endianness
= DEVICE_NATIVE_ENDIAN
,
242 .impl
.min_access_size
= 4,
243 .impl
.max_access_size
= 4,
246 static int ibex_timer_post_load(void *opaque
, int version_id
)
248 IbexTimerState
*s
= opaque
;
250 ibex_timer_update_irqs(s
);
254 static const VMStateDescription vmstate_ibex_timer
= {
255 .name
= TYPE_IBEX_TIMER
,
257 .minimum_version_id
= 2,
258 .post_load
= ibex_timer_post_load
,
259 .fields
= (VMStateField
[]) {
260 VMSTATE_UINT32(timer_ctrl
, IbexTimerState
),
261 VMSTATE_UINT32(timer_cfg0
, IbexTimerState
),
262 VMSTATE_UINT32(timer_compare_lower0
, IbexTimerState
),
263 VMSTATE_UINT32(timer_compare_upper0
, IbexTimerState
),
264 VMSTATE_UINT32(timer_intr_enable
, IbexTimerState
),
265 VMSTATE_UINT32(timer_intr_state
, IbexTimerState
),
266 VMSTATE_END_OF_LIST()
270 static Property ibex_timer_properties
[] = {
271 DEFINE_PROP_UINT32("timebase-freq", IbexTimerState
, timebase_freq
, 10000),
272 DEFINE_PROP_END_OF_LIST(),
275 static void ibex_timer_init(Object
*obj
)
277 IbexTimerState
*s
= IBEX_TIMER(obj
);
279 sysbus_init_irq(SYS_BUS_DEVICE(obj
), &s
->irq
);
281 memory_region_init_io(&s
->mmio
, obj
, &ibex_timer_ops
, s
,
282 TYPE_IBEX_TIMER
, 0x400);
283 sysbus_init_mmio(SYS_BUS_DEVICE(obj
), &s
->mmio
);
286 static void ibex_timer_realize(DeviceState
*dev
, Error
**errp
)
288 IbexTimerState
*s
= IBEX_TIMER(dev
);
290 qdev_init_gpio_out(dev
, &s
->m_timer_irq
, 1);
294 static void ibex_timer_class_init(ObjectClass
*klass
, void *data
)
296 DeviceClass
*dc
= DEVICE_CLASS(klass
);
298 dc
->reset
= ibex_timer_reset
;
299 dc
->vmsd
= &vmstate_ibex_timer
;
300 dc
->realize
= ibex_timer_realize
;
301 device_class_set_props(dc
, ibex_timer_properties
);
304 static const TypeInfo ibex_timer_info
= {
305 .name
= TYPE_IBEX_TIMER
,
306 .parent
= TYPE_SYS_BUS_DEVICE
,
307 .instance_size
= sizeof(IbexTimerState
),
308 .instance_init
= ibex_timer_init
,
309 .class_init
= ibex_timer_class_init
,
312 static void ibex_timer_register_types(void)
314 type_register_static(&ibex_timer_info
);
317 type_init(ibex_timer_register_types
)