xref: /openbmc/qemu/rust/hw/timer/hpet/src/hpet.rs (revision 71119ed3651622e1c531d1294839e9f3341adaf5)
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