1269a8f15SZhao Liu // Copyright (C) 2024 Intel Corporation.
2269a8f15SZhao Liu // Author(s): Zhao Liu <zhai1.liu@intel.com>
3269a8f15SZhao Liu // SPDX-License-Identifier: GPL-2.0-or-later
4269a8f15SZhao Liu
56e90a8f8SZhao Liu use std::{
66e90a8f8SZhao Liu ffi::CStr,
7a32b2396SPaolo Bonzini pin::Pin,
86e90a8f8SZhao Liu ptr::{addr_of_mut, null_mut, NonNull},
96e90a8f8SZhao Liu slice::from_ref,
106e90a8f8SZhao Liu };
11269a8f15SZhao Liu
12269a8f15SZhao Liu use qemu_api::{
136e90a8f8SZhao Liu bindings::{
146e90a8f8SZhao Liu address_space_memory, address_space_stl_le, qdev_prop_bit, qdev_prop_bool,
157bda68e8SPaolo Bonzini qdev_prop_uint32, qdev_prop_usize,
166e90a8f8SZhao Liu },
176e90a8f8SZhao Liu c_str,
18269a8f15SZhao Liu cell::{BqlCell, BqlRefCell},
19269a8f15SZhao Liu irq::InterruptSource,
206e90a8f8SZhao Liu memory::{
216e90a8f8SZhao Liu hwaddr, MemoryRegion, MemoryRegionOps, MemoryRegionOpsBuilder, MEMTXATTRS_UNSPECIFIED,
226e90a8f8SZhao Liu },
23269a8f15SZhao Liu prelude::*,
246e90a8f8SZhao Liu qdev::{DeviceImpl, DeviceMethods, DeviceState, Property, ResetType, ResettablePhasesImpl},
25d556226dSPaolo Bonzini qom::{ObjectImpl, ObjectType, ParentField},
266e90a8f8SZhao Liu qom_isa,
273212da00SPaolo Bonzini sysbus::{SysBusDevice, SysBusDeviceImpl},
28269a8f15SZhao Liu timer::{Timer, CLOCK_VIRTUAL},
29269a8f15SZhao Liu };
30269a8f15SZhao Liu
316e90a8f8SZhao Liu use crate::fw_cfg::HPETFwConfig;
326e90a8f8SZhao Liu
33269a8f15SZhao Liu /// Register space for each timer block (`HPET_BASE` is defined in hpet.h).
34269a8f15SZhao Liu const HPET_REG_SPACE_LEN: u64 = 0x400; // 1024 bytes
35269a8f15SZhao Liu
36269a8f15SZhao Liu /// Minimum recommended hardware implementation.
37269a8f15SZhao Liu const HPET_MIN_TIMERS: usize = 3;
38269a8f15SZhao Liu /// Maximum timers in each timer block.
39269a8f15SZhao Liu const HPET_MAX_TIMERS: usize = 32;
40269a8f15SZhao Liu
41269a8f15SZhao Liu /// Flags that HPETState.flags supports.
42269a8f15SZhao Liu const HPET_FLAG_MSI_SUPPORT_SHIFT: usize = 0;
43269a8f15SZhao Liu
44269a8f15SZhao Liu const HPET_NUM_IRQ_ROUTES: usize = 32;
45269a8f15SZhao Liu const HPET_LEGACY_PIT_INT: u32 = 0; // HPET_LEGACY_RTC_INT isn't defined here.
46269a8f15SZhao Liu const RTC_ISA_IRQ: usize = 8;
47269a8f15SZhao Liu
48269a8f15SZhao Liu const HPET_CLK_PERIOD: u64 = 10; // 10 ns
49269a8f15SZhao Liu const FS_PER_NS: u64 = 1000000; // 1000000 femtoseconds == 1 ns
50269a8f15SZhao Liu
51269a8f15SZhao Liu /// Revision ID (bits 0:7). Revision 1 is implemented (refer to v1.0a spec).
52269a8f15SZhao Liu const HPET_CAP_REV_ID_VALUE: u64 = 0x1;
53269a8f15SZhao Liu const HPET_CAP_REV_ID_SHIFT: usize = 0;
54269a8f15SZhao Liu /// Number of Timers (bits 8:12)
55269a8f15SZhao Liu const HPET_CAP_NUM_TIM_SHIFT: usize = 8;
56269a8f15SZhao Liu /// Counter Size (bit 13)
57269a8f15SZhao Liu const HPET_CAP_COUNT_SIZE_CAP_SHIFT: usize = 13;
58269a8f15SZhao Liu /// Legacy Replacement Route Capable (bit 15)
59269a8f15SZhao Liu const HPET_CAP_LEG_RT_CAP_SHIFT: usize = 15;
60269a8f15SZhao Liu /// Vendor ID (bits 16:31)
61269a8f15SZhao Liu const HPET_CAP_VENDER_ID_VALUE: u64 = 0x8086;
62269a8f15SZhao Liu const HPET_CAP_VENDER_ID_SHIFT: usize = 16;
63269a8f15SZhao Liu /// Main Counter Tick Period (bits 32:63)
64269a8f15SZhao Liu const HPET_CAP_CNT_CLK_PERIOD_SHIFT: usize = 32;
65269a8f15SZhao Liu
66269a8f15SZhao Liu /// Overall Enable (bit 0)
67269a8f15SZhao Liu const HPET_CFG_ENABLE_SHIFT: usize = 0;
68269a8f15SZhao Liu /// Legacy Replacement Route (bit 1)
69269a8f15SZhao Liu const HPET_CFG_LEG_RT_SHIFT: usize = 1;
70269a8f15SZhao Liu /// Other bits are reserved.
71269a8f15SZhao Liu const HPET_CFG_WRITE_MASK: u64 = 0x003;
72269a8f15SZhao Liu
73269a8f15SZhao Liu /// bit 0, 7, and bits 16:31 are reserved.
74269a8f15SZhao Liu /// bit 4, 5, 15, and bits 32:64 are read-only.
75269a8f15SZhao Liu const HPET_TN_CFG_WRITE_MASK: u64 = 0x7f4e;
76269a8f15SZhao Liu /// Timer N Interrupt Type (bit 1)
77269a8f15SZhao Liu const HPET_TN_CFG_INT_TYPE_SHIFT: usize = 1;
78269a8f15SZhao Liu /// Timer N Interrupt Enable (bit 2)
79269a8f15SZhao Liu const HPET_TN_CFG_INT_ENABLE_SHIFT: usize = 2;
80269a8f15SZhao Liu /// Timer N Type (Periodic enabled or not, bit 3)
81269a8f15SZhao Liu const HPET_TN_CFG_PERIODIC_SHIFT: usize = 3;
82269a8f15SZhao Liu /// Timer N Periodic Interrupt Capable (support Periodic or not, bit 4)
83269a8f15SZhao Liu const HPET_TN_CFG_PERIODIC_CAP_SHIFT: usize = 4;
84269a8f15SZhao Liu /// Timer N Size (timer size is 64-bits or 32 bits, bit 5)
85269a8f15SZhao Liu const HPET_TN_CFG_SIZE_CAP_SHIFT: usize = 5;
86269a8f15SZhao Liu /// Timer N Value Set (bit 6)
87269a8f15SZhao Liu const HPET_TN_CFG_SETVAL_SHIFT: usize = 6;
88269a8f15SZhao Liu /// Timer N 32-bit Mode (bit 8)
89269a8f15SZhao Liu const HPET_TN_CFG_32BIT_SHIFT: usize = 8;
90269a8f15SZhao Liu /// Timer N Interrupt Rout (bits 9:13)
91269a8f15SZhao Liu const HPET_TN_CFG_INT_ROUTE_MASK: u64 = 0x3e00;
92269a8f15SZhao Liu const HPET_TN_CFG_INT_ROUTE_SHIFT: usize = 9;
93269a8f15SZhao Liu /// Timer N FSB Interrupt Enable (bit 14)
94269a8f15SZhao Liu const HPET_TN_CFG_FSB_ENABLE_SHIFT: usize = 14;
95269a8f15SZhao Liu /// Timer N FSB Interrupt Delivery (bit 15)
96269a8f15SZhao Liu const HPET_TN_CFG_FSB_CAP_SHIFT: usize = 15;
97269a8f15SZhao Liu /// Timer N Interrupt Routing Capability (bits 32:63)
98269a8f15SZhao Liu const HPET_TN_CFG_INT_ROUTE_CAP_SHIFT: usize = 32;
99269a8f15SZhao Liu
100519088b7SPaolo Bonzini #[derive(qemu_api_macros::TryInto)]
101519088b7SPaolo Bonzini #[repr(u64)]
102519088b7SPaolo Bonzini #[allow(non_camel_case_types)]
103519088b7SPaolo Bonzini /// Timer registers, masked by 0x18
104519088b7SPaolo Bonzini enum TimerRegister {
105519088b7SPaolo Bonzini /// Timer N Configuration and Capability Register
106519088b7SPaolo Bonzini CFG = 0,
107519088b7SPaolo Bonzini /// Timer N Comparator Value Register
108519088b7SPaolo Bonzini CMP = 8,
109519088b7SPaolo Bonzini /// Timer N FSB Interrupt Route Register
110519088b7SPaolo Bonzini ROUTE = 16,
111519088b7SPaolo Bonzini }
112269a8f15SZhao Liu
113519088b7SPaolo Bonzini #[derive(qemu_api_macros::TryInto)]
114519088b7SPaolo Bonzini #[repr(u64)]
115519088b7SPaolo Bonzini #[allow(non_camel_case_types)]
116519088b7SPaolo Bonzini /// Global registers
117519088b7SPaolo Bonzini enum GlobalRegister {
118519088b7SPaolo Bonzini /// General Capabilities and ID Register
119519088b7SPaolo Bonzini CAP = 0,
120519088b7SPaolo Bonzini /// General Configuration Register
121519088b7SPaolo Bonzini CFG = 0x10,
122519088b7SPaolo Bonzini /// General Interrupt Status Register
123519088b7SPaolo Bonzini INT_STATUS = 0x20,
124519088b7SPaolo Bonzini /// Main Counter Value Register
125519088b7SPaolo Bonzini COUNTER = 0xF0,
126519088b7SPaolo Bonzini }
127519088b7SPaolo Bonzini
128519088b7SPaolo Bonzini enum HPETRegister<'a> {
129519088b7SPaolo Bonzini /// Global register in the range from `0` to `0xff`
130519088b7SPaolo Bonzini Global(GlobalRegister),
131519088b7SPaolo Bonzini
132519088b7SPaolo Bonzini /// Register in the timer block `0x100`...`0x3ff`
133519088b7SPaolo Bonzini Timer(&'a BqlRefCell<HPETTimer>, TimerRegister),
134519088b7SPaolo Bonzini
135519088b7SPaolo Bonzini /// Invalid address
136519088b7SPaolo Bonzini #[allow(dead_code)]
137519088b7SPaolo Bonzini Unknown(hwaddr),
138519088b7SPaolo Bonzini }
139519088b7SPaolo Bonzini
140519088b7SPaolo Bonzini struct HPETAddrDecode<'a> {
141519088b7SPaolo Bonzini shift: u32,
142519088b7SPaolo Bonzini len: u32,
143519088b7SPaolo Bonzini reg: HPETRegister<'a>,
144519088b7SPaolo Bonzini }
145269a8f15SZhao Liu
hpet_next_wrap(cur_tick: u64) -> u64146269a8f15SZhao Liu const fn hpet_next_wrap(cur_tick: u64) -> u64 {
147269a8f15SZhao Liu (cur_tick | 0xffffffff) + 1
148269a8f15SZhao Liu }
149269a8f15SZhao Liu
hpet_time_after(a: u64, b: u64) -> bool150269a8f15SZhao Liu const fn hpet_time_after(a: u64, b: u64) -> bool {
151269a8f15SZhao Liu ((b - a) as i64) < 0
152269a8f15SZhao Liu }
153269a8f15SZhao Liu
ticks_to_ns(value: u64) -> u64154269a8f15SZhao Liu const fn ticks_to_ns(value: u64) -> u64 {
155269a8f15SZhao Liu value * HPET_CLK_PERIOD
156269a8f15SZhao Liu }
157269a8f15SZhao Liu
ns_to_ticks(value: u64) -> u64158269a8f15SZhao Liu const fn ns_to_ticks(value: u64) -> u64 {
159269a8f15SZhao Liu value / HPET_CLK_PERIOD
160269a8f15SZhao Liu }
161269a8f15SZhao Liu
162269a8f15SZhao Liu // Avoid touching the bits that cannot be written.
hpet_fixup_reg(new: u64, old: u64, mask: u64) -> u64163269a8f15SZhao Liu const fn hpet_fixup_reg(new: u64, old: u64, mask: u64) -> u64 {
164269a8f15SZhao Liu (new & mask) | (old & !mask)
165269a8f15SZhao Liu }
166269a8f15SZhao Liu
activating_bit(old: u64, new: u64, shift: usize) -> bool167269a8f15SZhao Liu const fn activating_bit(old: u64, new: u64, shift: usize) -> bool {
168269a8f15SZhao Liu let mask: u64 = 1 << shift;
169269a8f15SZhao Liu (old & mask == 0) && (new & mask != 0)
170269a8f15SZhao Liu }
171269a8f15SZhao Liu
deactivating_bit(old: u64, new: u64, shift: usize) -> bool172269a8f15SZhao Liu const fn deactivating_bit(old: u64, new: u64, shift: usize) -> bool {
173269a8f15SZhao Liu let mask: u64 = 1 << shift;
174269a8f15SZhao Liu (old & mask != 0) && (new & mask == 0)
175269a8f15SZhao Liu }
176269a8f15SZhao Liu
timer_handler(timer_cell: &BqlRefCell<HPETTimer>)177269a8f15SZhao Liu fn timer_handler(timer_cell: &BqlRefCell<HPETTimer>) {
178269a8f15SZhao Liu timer_cell.borrow_mut().callback()
179269a8f15SZhao Liu }
180269a8f15SZhao Liu
181269a8f15SZhao Liu /// HPET Timer Abstraction
182269a8f15SZhao Liu #[repr(C)]
183e8dc87feSPaolo Bonzini #[derive(Debug, qemu_api_macros::offsets)]
184269a8f15SZhao Liu pub struct HPETTimer {
185269a8f15SZhao Liu /// timer N index within the timer block (`HPETState`)
186269a8f15SZhao Liu #[doc(alias = "tn")]
187269a8f15SZhao Liu index: usize,
188e8dc87feSPaolo Bonzini qemu_timer: Timer,
189269a8f15SZhao Liu /// timer block abstraction containing this timer
190e8dc87feSPaolo Bonzini state: NonNull<HPETState>,
191269a8f15SZhao Liu
192269a8f15SZhao Liu // Memory-mapped, software visible timer registers
193269a8f15SZhao Liu /// Timer N Configuration and Capability Register
194269a8f15SZhao Liu config: u64,
195269a8f15SZhao Liu /// Timer N Comparator Value Register
196269a8f15SZhao Liu cmp: u64,
197269a8f15SZhao Liu /// Timer N FSB Interrupt Route Register
198269a8f15SZhao Liu fsb: u64,
199269a8f15SZhao Liu
200269a8f15SZhao Liu // Hidden register state
201269a8f15SZhao Liu /// comparator (extended to counter width)
202269a8f15SZhao Liu cmp64: u64,
203269a8f15SZhao Liu /// Last value written to comparator
204269a8f15SZhao Liu period: u64,
205269a8f15SZhao Liu /// timer pop will indicate wrap for one-shot 32-bit
206269a8f15SZhao Liu /// mode. Next pop will be actual timer expiration.
207269a8f15SZhao Liu wrap_flag: u8,
208269a8f15SZhao Liu /// last value armed, to avoid timer storms
209269a8f15SZhao Liu last: u64,
210269a8f15SZhao Liu }
211269a8f15SZhao Liu
212269a8f15SZhao Liu impl HPETTimer {
init(&mut self, index: usize, state: &HPETState)213e8dc87feSPaolo Bonzini fn init(&mut self, index: usize, state: &HPETState) {
214e8dc87feSPaolo Bonzini *self = HPETTimer {
215e8dc87feSPaolo Bonzini index,
216a32b2396SPaolo Bonzini // SAFETY: the HPETTimer will only be used after the timer
217a32b2396SPaolo Bonzini // is initialized below.
218a32b2396SPaolo Bonzini qemu_timer: unsafe { Timer::new() },
219e8dc87feSPaolo Bonzini state: NonNull::new(state as *const _ as *mut _).unwrap(),
220e8dc87feSPaolo Bonzini config: 0,
221e8dc87feSPaolo Bonzini cmp: 0,
222e8dc87feSPaolo Bonzini fsb: 0,
223e8dc87feSPaolo Bonzini cmp64: 0,
224e8dc87feSPaolo Bonzini period: 0,
225e8dc87feSPaolo Bonzini wrap_flag: 0,
226e8dc87feSPaolo Bonzini last: 0,
227e8dc87feSPaolo Bonzini };
228269a8f15SZhao Liu
229a32b2396SPaolo Bonzini // SAFETY: HPETTimer is only used as part of HPETState, which is
230a32b2396SPaolo Bonzini // always pinned.
231a32b2396SPaolo Bonzini let qemu_timer = unsafe { Pin::new_unchecked(&mut self.qemu_timer) };
232a32b2396SPaolo Bonzini qemu_timer.init_full(
233269a8f15SZhao Liu None,
234269a8f15SZhao Liu CLOCK_VIRTUAL,
235269a8f15SZhao Liu Timer::NS,
236269a8f15SZhao Liu 0,
237269a8f15SZhao Liu timer_handler,
238e8dc87feSPaolo Bonzini &state.timers[self.index],
239e8dc87feSPaolo Bonzini )
240269a8f15SZhao Liu }
241269a8f15SZhao Liu
get_state(&self) -> &HPETState242269a8f15SZhao Liu fn get_state(&self) -> &HPETState {
243269a8f15SZhao Liu // SAFETY:
244269a8f15SZhao Liu // the pointer is convertible to a reference
245e8dc87feSPaolo Bonzini unsafe { self.state.as_ref() }
246269a8f15SZhao Liu }
247269a8f15SZhao Liu
is_int_active(&self) -> bool248269a8f15SZhao Liu fn is_int_active(&self) -> bool {
249269a8f15SZhao Liu self.get_state().is_timer_int_active(self.index)
250269a8f15SZhao Liu }
251269a8f15SZhao Liu
is_fsb_route_enabled(&self) -> bool252269a8f15SZhao Liu const fn is_fsb_route_enabled(&self) -> bool {
253269a8f15SZhao Liu self.config & (1 << HPET_TN_CFG_FSB_ENABLE_SHIFT) != 0
254269a8f15SZhao Liu }
255269a8f15SZhao Liu
is_periodic(&self) -> bool256269a8f15SZhao Liu const fn is_periodic(&self) -> bool {
257269a8f15SZhao Liu self.config & (1 << HPET_TN_CFG_PERIODIC_SHIFT) != 0
258269a8f15SZhao Liu }
259269a8f15SZhao Liu
is_int_enabled(&self) -> bool260269a8f15SZhao Liu const fn is_int_enabled(&self) -> bool {
261269a8f15SZhao Liu self.config & (1 << HPET_TN_CFG_INT_ENABLE_SHIFT) != 0
262269a8f15SZhao Liu }
263269a8f15SZhao Liu
is_32bit_mod(&self) -> bool264269a8f15SZhao Liu const fn is_32bit_mod(&self) -> bool {
265269a8f15SZhao Liu self.config & (1 << HPET_TN_CFG_32BIT_SHIFT) != 0
266269a8f15SZhao Liu }
267269a8f15SZhao Liu
is_valset_enabled(&self) -> bool268269a8f15SZhao Liu const fn is_valset_enabled(&self) -> bool {
269269a8f15SZhao Liu self.config & (1 << HPET_TN_CFG_SETVAL_SHIFT) != 0
270269a8f15SZhao Liu }
271269a8f15SZhao Liu
clear_valset(&mut self)272269a8f15SZhao Liu fn clear_valset(&mut self) {
273269a8f15SZhao Liu self.config &= !(1 << HPET_TN_CFG_SETVAL_SHIFT);
274269a8f15SZhao Liu }
275269a8f15SZhao Liu
276269a8f15SZhao Liu /// True if timer interrupt is level triggered; otherwise, edge triggered.
is_int_level_triggered(&self) -> bool277269a8f15SZhao Liu const fn is_int_level_triggered(&self) -> bool {
278269a8f15SZhao Liu self.config & (1 << HPET_TN_CFG_INT_TYPE_SHIFT) != 0
279269a8f15SZhao Liu }
280269a8f15SZhao Liu
281269a8f15SZhao Liu /// calculate next value of the general counter that matches the
282269a8f15SZhao Liu /// target (either entirely, or the low 32-bit only depending on
283269a8f15SZhao Liu /// the timer mode).
calculate_cmp64(&self, cur_tick: u64, target: u64) -> u64284269a8f15SZhao Liu fn calculate_cmp64(&self, cur_tick: u64, target: u64) -> u64 {
285269a8f15SZhao Liu if self.is_32bit_mod() {
286269a8f15SZhao Liu let mut result: u64 = cur_tick.deposit(0, 32, target);
287269a8f15SZhao Liu if result < cur_tick {
288269a8f15SZhao Liu result += 0x100000000;
289269a8f15SZhao Liu }
290269a8f15SZhao Liu result
291269a8f15SZhao Liu } else {
292269a8f15SZhao Liu target
293269a8f15SZhao Liu }
294269a8f15SZhao Liu }
295269a8f15SZhao Liu
get_individual_route(&self) -> usize296269a8f15SZhao Liu const fn get_individual_route(&self) -> usize {
297269a8f15SZhao Liu ((self.config & HPET_TN_CFG_INT_ROUTE_MASK) >> HPET_TN_CFG_INT_ROUTE_SHIFT) as usize
298269a8f15SZhao Liu }
299269a8f15SZhao Liu
get_int_route(&self) -> usize300269a8f15SZhao Liu fn get_int_route(&self) -> usize {
301269a8f15SZhao Liu if self.index <= 1 && self.get_state().is_legacy_mode() {
302269a8f15SZhao Liu // If LegacyReplacement Route bit is set, HPET specification requires
303269a8f15SZhao Liu // timer0 be routed to IRQ0 in NON-APIC or IRQ2 in the I/O APIC,
304269a8f15SZhao Liu // timer1 be routed to IRQ8 in NON-APIC or IRQ8 in the I/O APIC.
305269a8f15SZhao Liu //
306269a8f15SZhao Liu // If the LegacyReplacement Route bit is set, the individual routing
307269a8f15SZhao Liu // bits for timers 0 and 1 (APIC or FSB) will have no impact.
308269a8f15SZhao Liu //
309269a8f15SZhao Liu // FIXME: Consider I/O APIC case.
310269a8f15SZhao Liu if self.index == 0 {
311269a8f15SZhao Liu 0
312269a8f15SZhao Liu } else {
313269a8f15SZhao Liu RTC_ISA_IRQ
314269a8f15SZhao Liu }
315269a8f15SZhao Liu } else {
316269a8f15SZhao Liu // (If the LegacyReplacement Route bit is set) Timer 2-n will be
317269a8f15SZhao Liu // routed as per the routing in the timer n config registers.
318269a8f15SZhao Liu // ...
319269a8f15SZhao Liu // If the LegacyReplacement Route bit is not set, the individual
320269a8f15SZhao Liu // routing bits for each of the timers are used.
321269a8f15SZhao Liu self.get_individual_route()
322269a8f15SZhao Liu }
323269a8f15SZhao Liu }
324269a8f15SZhao Liu
set_irq(&mut self, set: bool)325269a8f15SZhao Liu fn set_irq(&mut self, set: bool) {
326269a8f15SZhao Liu let route = self.get_int_route();
327269a8f15SZhao Liu
328269a8f15SZhao Liu if set && self.is_int_enabled() && self.get_state().is_hpet_enabled() {
329269a8f15SZhao Liu if self.is_fsb_route_enabled() {
330269a8f15SZhao Liu // SAFETY:
331269a8f15SZhao Liu // the parameters are valid.
332269a8f15SZhao Liu unsafe {
333269a8f15SZhao Liu address_space_stl_le(
334269a8f15SZhao Liu addr_of_mut!(address_space_memory),
335269a8f15SZhao Liu self.fsb >> 32, // Timer N FSB int addr
336269a8f15SZhao Liu self.fsb as u32, // Timer N FSB int value, truncate!
337269a8f15SZhao Liu MEMTXATTRS_UNSPECIFIED,
338269a8f15SZhao Liu null_mut(),
339269a8f15SZhao Liu );
340269a8f15SZhao Liu }
341269a8f15SZhao Liu } else if self.is_int_level_triggered() {
342269a8f15SZhao Liu self.get_state().irqs[route].raise();
343269a8f15SZhao Liu } else {
344269a8f15SZhao Liu self.get_state().irqs[route].pulse();
345269a8f15SZhao Liu }
346269a8f15SZhao Liu } else if !self.is_fsb_route_enabled() {
347269a8f15SZhao Liu self.get_state().irqs[route].lower();
348269a8f15SZhao Liu }
349269a8f15SZhao Liu }
350269a8f15SZhao Liu
update_irq(&mut self, set: bool)351269a8f15SZhao Liu fn update_irq(&mut self, set: bool) {
352269a8f15SZhao Liu // If Timer N Interrupt Enable bit is 0, "the timer will
353269a8f15SZhao Liu // still operate and generate appropriate status bits, but
354269a8f15SZhao Liu // will not cause an interrupt"
355269a8f15SZhao Liu self.get_state()
356269a8f15SZhao Liu .update_int_status(self.index as u32, set && self.is_int_level_triggered());
357269a8f15SZhao Liu self.set_irq(set);
358269a8f15SZhao Liu }
359269a8f15SZhao Liu
arm_timer(&mut self, tick: u64)360269a8f15SZhao Liu fn arm_timer(&mut self, tick: u64) {
361269a8f15SZhao Liu let mut ns = self.get_state().get_ns(tick);
362269a8f15SZhao Liu
363269a8f15SZhao Liu // Clamp period to reasonable min value (1 us)
364269a8f15SZhao Liu if self.is_periodic() && ns - self.last < 1000 {
365269a8f15SZhao Liu ns = self.last + 1000;
366269a8f15SZhao Liu }
367269a8f15SZhao Liu
368269a8f15SZhao Liu self.last = ns;
369e8dc87feSPaolo Bonzini self.qemu_timer.modify(self.last);
370269a8f15SZhao Liu }
371269a8f15SZhao Liu
set_timer(&mut self)372269a8f15SZhao Liu fn set_timer(&mut self) {
373269a8f15SZhao Liu let cur_tick: u64 = self.get_state().get_ticks();
374269a8f15SZhao Liu
375269a8f15SZhao Liu self.wrap_flag = 0;
376269a8f15SZhao Liu self.cmp64 = self.calculate_cmp64(cur_tick, self.cmp);
377269a8f15SZhao Liu if self.is_32bit_mod() {
378269a8f15SZhao Liu // HPET spec says in one-shot 32-bit mode, generate an interrupt when
379269a8f15SZhao Liu // counter wraps in addition to an interrupt with comparator match.
380269a8f15SZhao Liu if !self.is_periodic() && self.cmp64 > hpet_next_wrap(cur_tick) {
381269a8f15SZhao Liu self.wrap_flag = 1;
382269a8f15SZhao Liu self.arm_timer(hpet_next_wrap(cur_tick));
383269a8f15SZhao Liu return;
384269a8f15SZhao Liu }
385269a8f15SZhao Liu }
386269a8f15SZhao Liu self.arm_timer(self.cmp64);
387269a8f15SZhao Liu }
388269a8f15SZhao Liu
del_timer(&mut self)389269a8f15SZhao Liu fn del_timer(&mut self) {
390269a8f15SZhao Liu // Just remove the timer from the timer_list without destroying
391269a8f15SZhao Liu // this timer instance.
392e8dc87feSPaolo Bonzini self.qemu_timer.delete();
393269a8f15SZhao Liu
394269a8f15SZhao Liu if self.is_int_active() {
395269a8f15SZhao Liu // For level-triggered interrupt, this leaves interrupt status
396269a8f15SZhao Liu // register set but lowers irq.
397269a8f15SZhao Liu self.update_irq(true);
398269a8f15SZhao Liu }
399269a8f15SZhao Liu }
400269a8f15SZhao Liu
401269a8f15SZhao Liu /// Configuration and Capability Register
set_tn_cfg_reg(&mut self, shift: u32, len: u32, val: u64)402269a8f15SZhao Liu fn set_tn_cfg_reg(&mut self, shift: u32, len: u32, val: u64) {
403269a8f15SZhao Liu // TODO: Add trace point - trace_hpet_ram_write_tn_cfg(addr & 4)
404269a8f15SZhao Liu let old_val: u64 = self.config;
405269a8f15SZhao Liu let mut new_val: u64 = old_val.deposit(shift, len, val);
406269a8f15SZhao Liu new_val = hpet_fixup_reg(new_val, old_val, HPET_TN_CFG_WRITE_MASK);
407269a8f15SZhao Liu
408269a8f15SZhao Liu // Switch level-type interrupt to edge-type.
409269a8f15SZhao Liu if deactivating_bit(old_val, new_val, HPET_TN_CFG_INT_TYPE_SHIFT) {
410269a8f15SZhao Liu // Do this before changing timer.config; otherwise, if
411269a8f15SZhao Liu // HPET_TN_FSB is set, update_irq will not lower the qemu_irq.
412269a8f15SZhao Liu self.update_irq(false);
413269a8f15SZhao Liu }
414269a8f15SZhao Liu
415269a8f15SZhao Liu self.config = new_val;
416269a8f15SZhao Liu
417269a8f15SZhao Liu if activating_bit(old_val, new_val, HPET_TN_CFG_INT_ENABLE_SHIFT) && self.is_int_active() {
418269a8f15SZhao Liu self.update_irq(true);
419269a8f15SZhao Liu }
420269a8f15SZhao Liu
421269a8f15SZhao Liu if self.is_32bit_mod() {
422269a8f15SZhao Liu self.cmp = u64::from(self.cmp as u32); // truncate!
423269a8f15SZhao Liu self.period = u64::from(self.period as u32); // truncate!
424269a8f15SZhao Liu }
425269a8f15SZhao Liu
426269a8f15SZhao Liu if self.get_state().is_hpet_enabled() {
427269a8f15SZhao Liu self.set_timer();
428269a8f15SZhao Liu }
429269a8f15SZhao Liu }
430269a8f15SZhao Liu
431269a8f15SZhao Liu /// Comparator Value Register
set_tn_cmp_reg(&mut self, shift: u32, len: u32, val: u64)432269a8f15SZhao Liu fn set_tn_cmp_reg(&mut self, shift: u32, len: u32, val: u64) {
433269a8f15SZhao Liu let mut length = len;
434269a8f15SZhao Liu let mut value = val;
435269a8f15SZhao Liu
436269a8f15SZhao Liu // TODO: Add trace point - trace_hpet_ram_write_tn_cmp(addr & 4)
437269a8f15SZhao Liu if self.is_32bit_mod() {
438269a8f15SZhao Liu // High 32-bits are zero, leave them untouched.
439269a8f15SZhao Liu if shift != 0 {
440269a8f15SZhao Liu // TODO: Add trace point - trace_hpet_ram_write_invalid_tn_cmp()
441269a8f15SZhao Liu return;
442269a8f15SZhao Liu }
443269a8f15SZhao Liu length = 64;
444269a8f15SZhao Liu value = u64::from(value as u32); // truncate!
445269a8f15SZhao Liu }
446269a8f15SZhao Liu
447269a8f15SZhao Liu if !self.is_periodic() || self.is_valset_enabled() {
448269a8f15SZhao Liu self.cmp = self.cmp.deposit(shift, length, value);
449269a8f15SZhao Liu }
450269a8f15SZhao Liu
451269a8f15SZhao Liu if self.is_periodic() {
452269a8f15SZhao Liu self.period = self.period.deposit(shift, length, value);
453269a8f15SZhao Liu }
454269a8f15SZhao Liu
455269a8f15SZhao Liu self.clear_valset();
456269a8f15SZhao Liu if self.get_state().is_hpet_enabled() {
457269a8f15SZhao Liu self.set_timer();
458269a8f15SZhao Liu }
459269a8f15SZhao Liu }
460269a8f15SZhao Liu
461269a8f15SZhao Liu /// FSB Interrupt Route Register
set_tn_fsb_route_reg(&mut self, shift: u32, len: u32, val: u64)462269a8f15SZhao Liu fn set_tn_fsb_route_reg(&mut self, shift: u32, len: u32, val: u64) {
463269a8f15SZhao Liu self.fsb = self.fsb.deposit(shift, len, val);
464269a8f15SZhao Liu }
465269a8f15SZhao Liu
reset(&mut self)466269a8f15SZhao Liu fn reset(&mut self) {
467269a8f15SZhao Liu self.del_timer();
468269a8f15SZhao Liu self.cmp = u64::MAX; // Comparator Match Registers reset to all 1's.
469269a8f15SZhao Liu self.config = (1 << HPET_TN_CFG_PERIODIC_CAP_SHIFT) | (1 << HPET_TN_CFG_SIZE_CAP_SHIFT);
470269a8f15SZhao Liu if self.get_state().has_msi_flag() {
471269a8f15SZhao Liu self.config |= 1 << HPET_TN_CFG_FSB_CAP_SHIFT;
472269a8f15SZhao Liu }
473269a8f15SZhao Liu // advertise availability of ioapic int
474269a8f15SZhao Liu self.config |=
475269a8f15SZhao Liu (u64::from(self.get_state().int_route_cap)) << HPET_TN_CFG_INT_ROUTE_CAP_SHIFT;
476269a8f15SZhao Liu self.period = 0;
477269a8f15SZhao Liu self.wrap_flag = 0;
478269a8f15SZhao Liu }
479269a8f15SZhao Liu
480269a8f15SZhao Liu /// timer expiration callback
callback(&mut self)481269a8f15SZhao Liu fn callback(&mut self) {
482269a8f15SZhao Liu let period: u64 = self.period;
483269a8f15SZhao Liu let cur_tick: u64 = self.get_state().get_ticks();
484269a8f15SZhao Liu
485269a8f15SZhao Liu if self.is_periodic() && period != 0 {
486269a8f15SZhao Liu while hpet_time_after(cur_tick, self.cmp64) {
487269a8f15SZhao Liu self.cmp64 += period;
488269a8f15SZhao Liu }
489269a8f15SZhao Liu if self.is_32bit_mod() {
490269a8f15SZhao Liu self.cmp = u64::from(self.cmp64 as u32); // truncate!
491269a8f15SZhao Liu } else {
492269a8f15SZhao Liu self.cmp = self.cmp64;
493269a8f15SZhao Liu }
494269a8f15SZhao Liu self.arm_timer(self.cmp64);
495269a8f15SZhao Liu } else if self.wrap_flag != 0 {
496269a8f15SZhao Liu self.wrap_flag = 0;
497269a8f15SZhao Liu self.arm_timer(self.cmp64);
498269a8f15SZhao Liu }
499269a8f15SZhao Liu self.update_irq(true);
500269a8f15SZhao Liu }
5016e90a8f8SZhao Liu
read(&self, reg: TimerRegister) -> u64502519088b7SPaolo Bonzini const fn read(&self, reg: TimerRegister) -> u64 {
503519088b7SPaolo Bonzini use TimerRegister::*;
504519088b7SPaolo Bonzini match reg {
505519088b7SPaolo Bonzini CFG => self.config, // including interrupt capabilities
506519088b7SPaolo Bonzini CMP => self.cmp, // comparator register
507519088b7SPaolo Bonzini ROUTE => self.fsb,
5086e90a8f8SZhao Liu }
5096e90a8f8SZhao Liu }
5106e90a8f8SZhao Liu
write(&mut self, reg: TimerRegister, value: u64, shift: u32, len: u32)511519088b7SPaolo Bonzini fn write(&mut self, reg: TimerRegister, value: u64, shift: u32, len: u32) {
512519088b7SPaolo Bonzini use TimerRegister::*;
513519088b7SPaolo Bonzini match reg {
514519088b7SPaolo Bonzini CFG => self.set_tn_cfg_reg(shift, len, value),
515519088b7SPaolo Bonzini CMP => self.set_tn_cmp_reg(shift, len, value),
516519088b7SPaolo Bonzini ROUTE => self.set_tn_fsb_route_reg(shift, len, value),
5176e90a8f8SZhao Liu }
5186e90a8f8SZhao Liu }
519269a8f15SZhao Liu }
520269a8f15SZhao Liu
521269a8f15SZhao Liu /// HPET Event Timer Block Abstraction
522269a8f15SZhao Liu #[repr(C)]
5236e90a8f8SZhao Liu #[derive(qemu_api_macros::Object, qemu_api_macros::offsets)]
524269a8f15SZhao Liu pub struct HPETState {
525269a8f15SZhao Liu parent_obj: ParentField<SysBusDevice>,
526269a8f15SZhao Liu iomem: MemoryRegion,
527269a8f15SZhao Liu
528269a8f15SZhao Liu // HPET block Registers: Memory-mapped, software visible registers
529269a8f15SZhao Liu /// General Capabilities and ID Register
530269a8f15SZhao Liu capability: BqlCell<u64>,
531269a8f15SZhao Liu /// General Configuration Register
532269a8f15SZhao Liu config: BqlCell<u64>,
533269a8f15SZhao Liu /// General Interrupt Status Register
534269a8f15SZhao Liu #[doc(alias = "isr")]
535269a8f15SZhao Liu int_status: BqlCell<u64>,
536269a8f15SZhao Liu /// Main Counter Value Register
537269a8f15SZhao Liu #[doc(alias = "hpet_counter")]
538269a8f15SZhao Liu counter: BqlCell<u64>,
539269a8f15SZhao Liu
540269a8f15SZhao Liu // Internal state
541269a8f15SZhao Liu /// Capabilities that QEMU HPET supports.
542269a8f15SZhao Liu /// bit 0: MSI (or FSB) support.
543269a8f15SZhao Liu flags: u32,
544269a8f15SZhao Liu
545269a8f15SZhao Liu /// Offset of main counter relative to qemu clock.
546269a8f15SZhao Liu hpet_offset: BqlCell<u64>,
547269a8f15SZhao Liu hpet_offset_saved: bool,
548269a8f15SZhao Liu
549269a8f15SZhao Liu irqs: [InterruptSource; HPET_NUM_IRQ_ROUTES],
550269a8f15SZhao Liu rtc_irq_level: BqlCell<u32>,
551269a8f15SZhao Liu pit_enabled: InterruptSource,
552269a8f15SZhao Liu
553269a8f15SZhao Liu /// Interrupt Routing Capability.
554269a8f15SZhao Liu /// This field indicates to which interrupts in the I/O (x) APIC
555269a8f15SZhao Liu /// the timers' interrupt can be routed, and is encoded in the
556269a8f15SZhao Liu /// bits 32:64 of timer N's config register:
557269a8f15SZhao Liu #[doc(alias = "intcap")]
558269a8f15SZhao Liu int_route_cap: u32,
559269a8f15SZhao Liu
560269a8f15SZhao Liu /// HPET timer array managed by this timer block.
561269a8f15SZhao Liu #[doc(alias = "timer")]
562269a8f15SZhao Liu timers: [BqlRefCell<HPETTimer>; HPET_MAX_TIMERS],
563269a8f15SZhao Liu num_timers: BqlCell<usize>,
564269a8f15SZhao Liu
565269a8f15SZhao Liu /// Instance id (HPET timer block ID).
566269a8f15SZhao Liu hpet_id: BqlCell<usize>,
567269a8f15SZhao Liu }
568269a8f15SZhao Liu
569269a8f15SZhao Liu impl HPETState {
has_msi_flag(&self) -> bool570269a8f15SZhao Liu const fn has_msi_flag(&self) -> bool {
571269a8f15SZhao Liu self.flags & (1 << HPET_FLAG_MSI_SUPPORT_SHIFT) != 0
572269a8f15SZhao Liu }
573269a8f15SZhao Liu
is_legacy_mode(&self) -> bool574269a8f15SZhao Liu fn is_legacy_mode(&self) -> bool {
575269a8f15SZhao Liu self.config.get() & (1 << HPET_CFG_LEG_RT_SHIFT) != 0
576269a8f15SZhao Liu }
577269a8f15SZhao Liu
is_hpet_enabled(&self) -> bool578269a8f15SZhao Liu fn is_hpet_enabled(&self) -> bool {
579269a8f15SZhao Liu self.config.get() & (1 << HPET_CFG_ENABLE_SHIFT) != 0
580269a8f15SZhao Liu }
581269a8f15SZhao Liu
is_timer_int_active(&self, index: usize) -> bool582269a8f15SZhao Liu fn is_timer_int_active(&self, index: usize) -> bool {
583269a8f15SZhao Liu self.int_status.get() & (1 << index) != 0
584269a8f15SZhao Liu }
585269a8f15SZhao Liu
get_ticks(&self) -> u64586269a8f15SZhao Liu fn get_ticks(&self) -> u64 {
587269a8f15SZhao Liu ns_to_ticks(CLOCK_VIRTUAL.get_ns() + self.hpet_offset.get())
588269a8f15SZhao Liu }
589269a8f15SZhao Liu
get_ns(&self, tick: u64) -> u64590269a8f15SZhao Liu fn get_ns(&self, tick: u64) -> u64 {
591269a8f15SZhao Liu ticks_to_ns(tick) - self.hpet_offset.get()
592269a8f15SZhao Liu }
593269a8f15SZhao Liu
handle_legacy_irq(&self, irq: u32, level: u32)594269a8f15SZhao Liu fn handle_legacy_irq(&self, irq: u32, level: u32) {
595269a8f15SZhao Liu if irq == HPET_LEGACY_PIT_INT {
596269a8f15SZhao Liu if !self.is_legacy_mode() {
597269a8f15SZhao Liu self.irqs[0].set(level != 0);
598269a8f15SZhao Liu }
599269a8f15SZhao Liu } else {
600269a8f15SZhao Liu self.rtc_irq_level.set(level);
601269a8f15SZhao Liu if !self.is_legacy_mode() {
602269a8f15SZhao Liu self.irqs[RTC_ISA_IRQ].set(level != 0);
603269a8f15SZhao Liu }
604269a8f15SZhao Liu }
605269a8f15SZhao Liu }
606269a8f15SZhao Liu
init_timer(&self)607269a8f15SZhao Liu fn init_timer(&self) {
608269a8f15SZhao Liu for (index, timer) in self.timers.iter().enumerate() {
609e8dc87feSPaolo Bonzini timer.borrow_mut().init(index, self);
610269a8f15SZhao Liu }
611269a8f15SZhao Liu }
612269a8f15SZhao Liu
update_int_status(&self, index: u32, level: bool)613269a8f15SZhao Liu fn update_int_status(&self, index: u32, level: bool) {
614269a8f15SZhao Liu self.int_status
615269a8f15SZhao Liu .set(self.int_status.get().deposit(index, 1, u64::from(level)));
616269a8f15SZhao Liu }
617269a8f15SZhao Liu
618269a8f15SZhao Liu /// General Configuration Register
set_cfg_reg(&self, shift: u32, len: u32, val: u64)619269a8f15SZhao Liu fn set_cfg_reg(&self, shift: u32, len: u32, val: u64) {
620269a8f15SZhao Liu let old_val = self.config.get();
621269a8f15SZhao Liu let mut new_val = old_val.deposit(shift, len, val);
622269a8f15SZhao Liu
623269a8f15SZhao Liu new_val = hpet_fixup_reg(new_val, old_val, HPET_CFG_WRITE_MASK);
624269a8f15SZhao Liu self.config.set(new_val);
625269a8f15SZhao Liu
626269a8f15SZhao Liu if activating_bit(old_val, new_val, HPET_CFG_ENABLE_SHIFT) {
627269a8f15SZhao Liu // Enable main counter and interrupt generation.
628269a8f15SZhao Liu self.hpet_offset
629269a8f15SZhao Liu .set(ticks_to_ns(self.counter.get()) - CLOCK_VIRTUAL.get_ns());
630269a8f15SZhao Liu
631269a8f15SZhao Liu for timer in self.timers.iter().take(self.num_timers.get()) {
632269a8f15SZhao Liu let mut t = timer.borrow_mut();
633269a8f15SZhao Liu
634269a8f15SZhao Liu if t.is_int_enabled() && t.is_int_active() {
635269a8f15SZhao Liu t.update_irq(true);
636269a8f15SZhao Liu }
637269a8f15SZhao Liu t.set_timer();
638269a8f15SZhao Liu }
639269a8f15SZhao Liu } else if deactivating_bit(old_val, new_val, HPET_CFG_ENABLE_SHIFT) {
640269a8f15SZhao Liu // Halt main counter and disable interrupt generation.
641269a8f15SZhao Liu self.counter.set(self.get_ticks());
642269a8f15SZhao Liu
643269a8f15SZhao Liu for timer in self.timers.iter().take(self.num_timers.get()) {
644269a8f15SZhao Liu timer.borrow_mut().del_timer();
645269a8f15SZhao Liu }
646269a8f15SZhao Liu }
647269a8f15SZhao Liu
648269a8f15SZhao Liu // i8254 and RTC output pins are disabled when HPET is in legacy mode
649269a8f15SZhao Liu if activating_bit(old_val, new_val, HPET_CFG_LEG_RT_SHIFT) {
650269a8f15SZhao Liu self.pit_enabled.set(false);
651269a8f15SZhao Liu self.irqs[0].lower();
652269a8f15SZhao Liu self.irqs[RTC_ISA_IRQ].lower();
653269a8f15SZhao Liu } else if deactivating_bit(old_val, new_val, HPET_CFG_LEG_RT_SHIFT) {
654269a8f15SZhao Liu // TODO: Add irq binding: qemu_irq_lower(s->irqs[0])
655269a8f15SZhao Liu self.irqs[0].lower();
656269a8f15SZhao Liu self.pit_enabled.set(true);
657269a8f15SZhao Liu self.irqs[RTC_ISA_IRQ].set(self.rtc_irq_level.get() != 0);
658269a8f15SZhao Liu }
659269a8f15SZhao Liu }
660269a8f15SZhao Liu
661269a8f15SZhao Liu /// General Interrupt Status Register: Read/Write Clear
set_int_status_reg(&self, shift: u32, _len: u32, val: u64)662269a8f15SZhao Liu fn set_int_status_reg(&self, shift: u32, _len: u32, val: u64) {
663269a8f15SZhao Liu let new_val = val << shift;
664269a8f15SZhao Liu let cleared = new_val & self.int_status.get();
665269a8f15SZhao Liu
666269a8f15SZhao Liu for (index, timer) in self.timers.iter().take(self.num_timers.get()).enumerate() {
667269a8f15SZhao Liu if cleared & (1 << index) != 0 {
668269a8f15SZhao Liu timer.borrow_mut().update_irq(false);
669269a8f15SZhao Liu }
670269a8f15SZhao Liu }
671269a8f15SZhao Liu }
672269a8f15SZhao Liu
673269a8f15SZhao Liu /// Main Counter Value Register
set_counter_reg(&self, shift: u32, len: u32, val: u64)674269a8f15SZhao Liu fn set_counter_reg(&self, shift: u32, len: u32, val: u64) {
675269a8f15SZhao Liu if self.is_hpet_enabled() {
676269a8f15SZhao Liu // TODO: Add trace point -
677269a8f15SZhao Liu // trace_hpet_ram_write_counter_write_while_enabled()
678269a8f15SZhao Liu //
679269a8f15SZhao Liu // HPET spec says that writes to this register should only be
680269a8f15SZhao Liu // done while the counter is halted. So this is an undefined
681269a8f15SZhao Liu // behavior. There's no need to forbid it, but when HPET is
682269a8f15SZhao Liu // enabled, the changed counter value will not affect the
683269a8f15SZhao Liu // tick count (i.e., the previously calculated offset will
684269a8f15SZhao Liu // not be changed as well).
685269a8f15SZhao Liu }
686269a8f15SZhao Liu self.counter
687269a8f15SZhao Liu .set(self.counter.get().deposit(shift, len, val));
688269a8f15SZhao Liu }
6896e90a8f8SZhao Liu
init(&mut self)6906e90a8f8SZhao Liu unsafe fn init(&mut self) {
6916e90a8f8SZhao Liu static HPET_RAM_OPS: MemoryRegionOps<HPETState> =
6926e90a8f8SZhao Liu MemoryRegionOpsBuilder::<HPETState>::new()
6936e90a8f8SZhao Liu .read(&HPETState::read)
6946e90a8f8SZhao Liu .write(&HPETState::write)
6956e90a8f8SZhao Liu .native_endian()
6966e90a8f8SZhao Liu .valid_sizes(4, 8)
6976e90a8f8SZhao Liu .impl_sizes(4, 8)
6986e90a8f8SZhao Liu .build();
6996e90a8f8SZhao Liu
7006e90a8f8SZhao Liu // SAFETY:
7016e90a8f8SZhao Liu // self and self.iomem are guaranteed to be valid at this point since callers
7026e90a8f8SZhao Liu // must make sure the `self` reference is valid.
7036e90a8f8SZhao Liu MemoryRegion::init_io(
7046e90a8f8SZhao Liu unsafe { &mut *addr_of_mut!(self.iomem) },
7056e90a8f8SZhao Liu addr_of_mut!(*self),
7066e90a8f8SZhao Liu &HPET_RAM_OPS,
7076e90a8f8SZhao Liu "hpet",
7086e90a8f8SZhao Liu HPET_REG_SPACE_LEN,
7096e90a8f8SZhao Liu );
7106e90a8f8SZhao Liu }
7116e90a8f8SZhao Liu
post_init(&self)7126e90a8f8SZhao Liu fn post_init(&self) {
7136e90a8f8SZhao Liu self.init_mmio(&self.iomem);
7146e90a8f8SZhao Liu for irq in self.irqs.iter() {
7156e90a8f8SZhao Liu self.init_irq(irq);
7166e90a8f8SZhao Liu }
7176e90a8f8SZhao Liu }
7186e90a8f8SZhao Liu
realize(&self)7196e90a8f8SZhao Liu fn realize(&self) {
7206e90a8f8SZhao Liu if self.int_route_cap == 0 {
7216e90a8f8SZhao Liu // TODO: Add error binding: warn_report()
7226e90a8f8SZhao Liu println!("Hpet's hpet-intcap property not initialized");
7236e90a8f8SZhao Liu }
7246e90a8f8SZhao Liu
7256e90a8f8SZhao Liu self.hpet_id.set(HPETFwConfig::assign_hpet_id());
7266e90a8f8SZhao Liu
7276e90a8f8SZhao Liu if self.num_timers.get() < HPET_MIN_TIMERS {
7286e90a8f8SZhao Liu self.num_timers.set(HPET_MIN_TIMERS);
7296e90a8f8SZhao Liu } else if self.num_timers.get() > HPET_MAX_TIMERS {
7306e90a8f8SZhao Liu self.num_timers.set(HPET_MAX_TIMERS);
7316e90a8f8SZhao Liu }
7326e90a8f8SZhao Liu
7336e90a8f8SZhao Liu self.init_timer();
7346e90a8f8SZhao Liu // 64-bit General Capabilities and ID Register; LegacyReplacementRoute.
7356e90a8f8SZhao Liu self.capability.set(
7366e90a8f8SZhao Liu HPET_CAP_REV_ID_VALUE << HPET_CAP_REV_ID_SHIFT |
7376e90a8f8SZhao Liu 1 << HPET_CAP_COUNT_SIZE_CAP_SHIFT |
7386e90a8f8SZhao Liu 1 << HPET_CAP_LEG_RT_CAP_SHIFT |
7396e90a8f8SZhao Liu HPET_CAP_VENDER_ID_VALUE << HPET_CAP_VENDER_ID_SHIFT |
7406e90a8f8SZhao Liu ((self.num_timers.get() - 1) as u64) << HPET_CAP_NUM_TIM_SHIFT | // indicate the last timer
7416e90a8f8SZhao Liu (HPET_CLK_PERIOD * FS_PER_NS) << HPET_CAP_CNT_CLK_PERIOD_SHIFT, // 10 ns
7426e90a8f8SZhao Liu );
7436e90a8f8SZhao Liu
7446e90a8f8SZhao Liu self.init_gpio_in(2, HPETState::handle_legacy_irq);
7456e90a8f8SZhao Liu self.init_gpio_out(from_ref(&self.pit_enabled));
7466e90a8f8SZhao Liu }
7476e90a8f8SZhao Liu
reset_hold(&self, _type: ResetType)7486e90a8f8SZhao Liu fn reset_hold(&self, _type: ResetType) {
7496e90a8f8SZhao Liu for timer in self.timers.iter().take(self.num_timers.get()) {
7506e90a8f8SZhao Liu timer.borrow_mut().reset();
7516e90a8f8SZhao Liu }
7526e90a8f8SZhao Liu
7536e90a8f8SZhao Liu self.counter.set(0);
7546e90a8f8SZhao Liu self.config.set(0);
7556e90a8f8SZhao Liu self.pit_enabled.set(true);
7566e90a8f8SZhao Liu self.hpet_offset.set(0);
7576e90a8f8SZhao Liu
7586e90a8f8SZhao Liu HPETFwConfig::update_hpet_cfg(
7596e90a8f8SZhao Liu self.hpet_id.get(),
7606e90a8f8SZhao Liu self.capability.get() as u32,
76109fda8f5SPaolo Bonzini self.mmio_addr(0).unwrap(),
7626e90a8f8SZhao Liu );
7636e90a8f8SZhao Liu
7646e90a8f8SZhao Liu // to document that the RTC lowers its output on reset as well
7656e90a8f8SZhao Liu self.rtc_irq_level.set(0);
7666e90a8f8SZhao Liu }
7676e90a8f8SZhao Liu
decode(&self, mut addr: hwaddr, size: u32) -> HPETAddrDecode768519088b7SPaolo Bonzini fn decode(&self, mut addr: hwaddr, size: u32) -> HPETAddrDecode {
7696e90a8f8SZhao Liu let shift = ((addr & 4) * 8) as u32;
7706e90a8f8SZhao Liu let len = std::cmp::min(size * 8, 64 - shift);
7716e90a8f8SZhao Liu
772519088b7SPaolo Bonzini addr &= !4;
773519088b7SPaolo Bonzini let reg = if (0..=0xff).contains(&addr) {
774519088b7SPaolo Bonzini GlobalRegister::try_from(addr).map(HPETRegister::Global)
7756e90a8f8SZhao Liu } else {
776519088b7SPaolo Bonzini let timer_id: usize = ((addr - 0x100) / 0x20) as usize;
777519088b7SPaolo Bonzini if timer_id <= self.num_timers.get() {
778519088b7SPaolo Bonzini // TODO: Add trace point - trace_hpet_ram_[read|write]_timer_id(timer_id)
779*64acc23cSPaolo Bonzini TimerRegister::try_from(addr & 0x18)
780519088b7SPaolo Bonzini .map(|reg| HPETRegister::Timer(&self.timers[timer_id], reg))
781519088b7SPaolo Bonzini } else {
782519088b7SPaolo Bonzini // TODO: Add trace point - trace_hpet_timer_id_out_of_range(timer_id)
783519088b7SPaolo Bonzini Err(addr)
7846e90a8f8SZhao Liu }
785519088b7SPaolo Bonzini };
786519088b7SPaolo Bonzini
787519088b7SPaolo Bonzini // reg is now a Result<HPETRegister, hwaddr>
788519088b7SPaolo Bonzini // convert the Err case into HPETRegister as well
789519088b7SPaolo Bonzini let reg = reg.unwrap_or_else(HPETRegister::Unknown);
790519088b7SPaolo Bonzini HPETAddrDecode { shift, len, reg }
791519088b7SPaolo Bonzini }
792519088b7SPaolo Bonzini
read(&self, addr: hwaddr, size: u32) -> u64793519088b7SPaolo Bonzini fn read(&self, addr: hwaddr, size: u32) -> u64 {
794519088b7SPaolo Bonzini // TODO: Add trace point - trace_hpet_ram_read(addr)
795519088b7SPaolo Bonzini let HPETAddrDecode { shift, reg, .. } = self.decode(addr, size);
796519088b7SPaolo Bonzini
797519088b7SPaolo Bonzini use GlobalRegister::*;
798519088b7SPaolo Bonzini use HPETRegister::*;
799519088b7SPaolo Bonzini (match reg {
800519088b7SPaolo Bonzini Timer(timer, tn_reg) => timer.borrow_mut().read(tn_reg),
801519088b7SPaolo Bonzini Global(CAP) => self.capability.get(), /* including HPET_PERIOD 0x004 */
802519088b7SPaolo Bonzini Global(CFG) => self.config.get(),
803519088b7SPaolo Bonzini Global(INT_STATUS) => self.int_status.get(),
804519088b7SPaolo Bonzini Global(COUNTER) => {
805519088b7SPaolo Bonzini // TODO: Add trace point
806519088b7SPaolo Bonzini // trace_hpet_ram_read_reading_counter(addr & 4, cur_tick)
807519088b7SPaolo Bonzini if self.is_hpet_enabled() {
808519088b7SPaolo Bonzini self.get_ticks()
809519088b7SPaolo Bonzini } else {
810519088b7SPaolo Bonzini self.counter.get()
811519088b7SPaolo Bonzini }
812519088b7SPaolo Bonzini }
813519088b7SPaolo Bonzini Unknown(_) => {
814519088b7SPaolo Bonzini // TODO: Add trace point- trace_hpet_ram_read_invalid()
815519088b7SPaolo Bonzini 0
816519088b7SPaolo Bonzini }
817519088b7SPaolo Bonzini }) >> shift
818519088b7SPaolo Bonzini }
819519088b7SPaolo Bonzini
write(&self, addr: hwaddr, value: u64, size: u32)820519088b7SPaolo Bonzini fn write(&self, addr: hwaddr, value: u64, size: u32) {
821519088b7SPaolo Bonzini let HPETAddrDecode { shift, len, reg } = self.decode(addr, size);
822519088b7SPaolo Bonzini
823519088b7SPaolo Bonzini // TODO: Add trace point - trace_hpet_ram_write(addr, value)
824519088b7SPaolo Bonzini use GlobalRegister::*;
825519088b7SPaolo Bonzini use HPETRegister::*;
826519088b7SPaolo Bonzini match reg {
827519088b7SPaolo Bonzini Timer(timer, tn_reg) => timer.borrow_mut().write(tn_reg, value, shift, len),
828519088b7SPaolo Bonzini Global(CAP) => {} // General Capabilities and ID Register: Read Only
829519088b7SPaolo Bonzini Global(CFG) => self.set_cfg_reg(shift, len, value),
830519088b7SPaolo Bonzini Global(INT_STATUS) => self.set_int_status_reg(shift, len, value),
831519088b7SPaolo Bonzini Global(COUNTER) => self.set_counter_reg(shift, len, value),
832519088b7SPaolo Bonzini Unknown(_) => {
833519088b7SPaolo Bonzini // TODO: Add trace point - trace_hpet_ram_write_invalid()
8346e90a8f8SZhao Liu }
8356e90a8f8SZhao Liu }
8366e90a8f8SZhao Liu }
8376e90a8f8SZhao Liu }
8386e90a8f8SZhao Liu
8396e90a8f8SZhao Liu qom_isa!(HPETState: SysBusDevice, DeviceState, Object);
8406e90a8f8SZhao Liu
8416e90a8f8SZhao Liu unsafe impl ObjectType for HPETState {
8426e90a8f8SZhao Liu // No need for HPETClass. Just like OBJECT_DECLARE_SIMPLE_TYPE in C.
8436e90a8f8SZhao Liu type Class = <SysBusDevice as ObjectType>::Class;
8446e90a8f8SZhao Liu const TYPE_NAME: &'static CStr = crate::TYPE_HPET;
8456e90a8f8SZhao Liu }
8466e90a8f8SZhao Liu
8476e90a8f8SZhao Liu impl ObjectImpl for HPETState {
8486e90a8f8SZhao Liu type ParentType = SysBusDevice;
8496e90a8f8SZhao Liu
8506e90a8f8SZhao Liu const INSTANCE_INIT: Option<unsafe fn(&mut Self)> = Some(Self::init);
8516e90a8f8SZhao Liu const INSTANCE_POST_INIT: Option<fn(&Self)> = Some(Self::post_init);
852d556226dSPaolo Bonzini const CLASS_INIT: fn(&mut Self::Class) = Self::Class::class_init::<Self>;
8536e90a8f8SZhao Liu }
8546e90a8f8SZhao Liu
8556e90a8f8SZhao Liu // TODO: Make these properties user-configurable!
8566e90a8f8SZhao Liu qemu_api::declare_properties! {
8576e90a8f8SZhao Liu HPET_PROPERTIES,
8586e90a8f8SZhao Liu qemu_api::define_property!(
8596e90a8f8SZhao Liu c_str!("timers"),
8606e90a8f8SZhao Liu HPETState,
8616e90a8f8SZhao Liu num_timers,
8627bda68e8SPaolo Bonzini unsafe { &qdev_prop_usize },
8637bda68e8SPaolo Bonzini usize,
8646e90a8f8SZhao Liu default = HPET_MIN_TIMERS
8656e90a8f8SZhao Liu ),
8666e90a8f8SZhao Liu qemu_api::define_property!(
8676e90a8f8SZhao Liu c_str!("msi"),
8686e90a8f8SZhao Liu HPETState,
8696e90a8f8SZhao Liu flags,
8706e90a8f8SZhao Liu unsafe { &qdev_prop_bit },
8716e90a8f8SZhao Liu u32,
8726e90a8f8SZhao Liu bit = HPET_FLAG_MSI_SUPPORT_SHIFT as u8,
8736e90a8f8SZhao Liu default = false,
8746e90a8f8SZhao Liu ),
8756e90a8f8SZhao Liu qemu_api::define_property!(
8766e90a8f8SZhao Liu c_str!("hpet-intcap"),
8776e90a8f8SZhao Liu HPETState,
8786e90a8f8SZhao Liu int_route_cap,
8796e90a8f8SZhao Liu unsafe { &qdev_prop_uint32 },
8806e90a8f8SZhao Liu u32,
8816e90a8f8SZhao Liu default = 0
8826e90a8f8SZhao Liu ),
8836e90a8f8SZhao Liu qemu_api::define_property!(
8846e90a8f8SZhao Liu c_str!("hpet-offset-saved"),
8856e90a8f8SZhao Liu HPETState,
8866e90a8f8SZhao Liu hpet_offset_saved,
8876e90a8f8SZhao Liu unsafe { &qdev_prop_bool },
8886e90a8f8SZhao Liu bool,
8896e90a8f8SZhao Liu default = true
8906e90a8f8SZhao Liu ),
8916e90a8f8SZhao Liu }
8926e90a8f8SZhao Liu
8936e90a8f8SZhao Liu impl DeviceImpl for HPETState {
properties() -> &'static [Property]8946e90a8f8SZhao Liu fn properties() -> &'static [Property] {
8956e90a8f8SZhao Liu &HPET_PROPERTIES
8966e90a8f8SZhao Liu }
8976e90a8f8SZhao Liu
8986e90a8f8SZhao Liu const REALIZE: Option<fn(&Self)> = Some(Self::realize);
8996e90a8f8SZhao Liu }
9006e90a8f8SZhao Liu
9016e90a8f8SZhao Liu impl ResettablePhasesImpl for HPETState {
9026e90a8f8SZhao Liu const HOLD: Option<fn(&Self, ResetType)> = Some(Self::reset_hold);
903269a8f15SZhao Liu }
9043212da00SPaolo Bonzini
9053212da00SPaolo Bonzini impl SysBusDeviceImpl for HPETState {}
906