VirtualBox

source: vbox/trunk/src/VBox/Devices/PC/DevHPET.cpp@ 88824

Last change on this file since 88824 was 88354, checked in by vboxsync, 4 years ago

DevHPET: scm fix

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 70.5 KB
Line 
1/* $Id: DevHPET.cpp 88354 2021-04-02 22:00:23Z vboxsync $ */
2/** @file
3 * HPET virtual device - High Precision Event Timer emulation.
4 *
5 * This implementation is based on the (generic) Intel IA-PC HPET specification
6 * and the Intel ICH9 datasheet.
7 *
8 * Typical windows 1809 usage (efi, smp) is to do repated one-shots and
9 * a variable rate. The reprogramming sequence is as follows (all accesses
10 * are 32-bit):
11 * -# counter register read.
12 * -# timer 0: config register read.
13 * -# timer 0: write 0x134 to config register.
14 * -# timer 0: write comparator register.
15 * -# timer 0: write 0x134 to config register.
16 * -# timer 0: read comparator register.
17 * -# counter register read.
18 *
19 * Typical linux will configure the timer at Hz but not necessarily enable
20 * interrupts (HPET_TN_ENABLE not set). It would be nice to emulate this
21 * mode without using timers.
22 *
23 */
24
25/*
26 * Copyright (C) 2009-2020 Oracle Corporation
27 *
28 * This file is part of VirtualBox Open Source Edition (OSE), as
29 * available from http://www.virtualbox.org. This file is free software;
30 * you can redistribute it and/or modify it under the terms of the GNU
31 * General Public License (GPL) as published by the Free Software
32 * Foundation, in version 2 as it comes in the "COPYING" file of the
33 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
34 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
35 */
36
37
38/*********************************************************************************************************************************
39* Header Files *
40*********************************************************************************************************************************/
41#define LOG_GROUP LOG_GROUP_DEV_HPET
42#include <VBox/vmm/pdmdev.h>
43#include <VBox/vmm/stam.h>
44#include <VBox/log.h>
45#include <VBox/AssertGuest.h>
46#include <iprt/asm-math.h>
47#include <iprt/string.h>
48
49#include "VBoxDD.h"
50
51
52/*********************************************************************************************************************************
53* Defined Constants And Macros *
54*********************************************************************************************************************************/
55/*
56 * Current limitations:
57 * - not entirely correct time of interrupt, i.e. never
58 * schedule interrupt earlier than in 1ms
59 * - statistics not implemented
60 * - level-triggered mode not implemented
61 */
62
63/** Base address for MMIO.
64 * On ICH9, it is 0xFED0x000 where 'x' is 0-3, default 0. We do not support
65 * relocation as the platform firmware is responsible for configuring the
66 * HPET base address and the OS isn't expected to move it.
67 * WARNING: This has to match the ACPI tables! */
68#define HPET_BASE 0xfed00000
69
70/** HPET reserves a 1K range. */
71#define HPET_BAR_SIZE 0x1000
72
73/** The number of timers for PIIX4 / PIIX3. */
74#define HPET_NUM_TIMERS_PIIX 3 /* Minimal implementation. */
75/** The number of timers for ICH9. */
76#define HPET_NUM_TIMERS_ICH9 4
77
78/** HPET clock period for PIIX4 / PIIX3.
79 * 10000000 femtoseconds == 10ns.
80 */
81#define HPET_CLK_PERIOD_PIIX UINT32_C(10000000)
82
83/** HPET clock period for ICH9.
84 * 69841279 femtoseconds == 69.84 ns (1 / 14.31818MHz).
85 */
86#define HPET_CLK_PERIOD_ICH9 UINT32_C(69841279)
87
88/**
89 * Femtosecods in a nanosecond
90 */
91#define FS_PER_NS 1000000
92
93/**
94 * Femtoseconds in a day. Still fits within int64_t.
95 */
96#define FS_PER_DAY (1000000LL * 60 * 60 * 24 * FS_PER_NS)
97
98/**
99 * Number of HPET ticks in 100 years, ICH9 frequency.
100 */
101#define HPET_TICKS_IN_100YR_ICH9 (FS_PER_DAY / HPET_CLK_PERIOD_ICH9 * 365 * 100)
102
103/**
104 * Number of HPET ticks in 100 years, made-up PIIX frequency.
105 */
106#define HPET_TICKS_IN_100YR_PIIX (FS_PER_DAY / HPET_CLK_PERIOD_PIIX * 365 * 100)
107
108/** @name Interrupt type
109 * @{ */
110#define HPET_TIMER_TYPE_LEVEL (1 << 1)
111#define HPET_TIMER_TYPE_EDGE (0 << 1)
112/** @} */
113
114/** @name Delivery mode
115 * @{ */
116#define HPET_TIMER_DELIVERY_APIC 0 /**< Delivery through APIC. */
117#define HPET_TIMER_DELIVERY_FSB 1 /**< Delivery through FSB. */
118/** @} */
119
120#define HPET_TIMER_CAP_FSB_INT_DEL (1 << 15)
121#define HPET_TIMER_CAP_PER_INT (1 << 4)
122
123#define HPET_CFG_ENABLE 0x001 /**< ENABLE_CNF */
124#define HPET_CFG_LEGACY 0x002 /**< LEG_RT_CNF */
125
126/** @name Register offsets in HPET space.
127 * @{ */
128#define HPET_ID 0x000 /**< Device ID. */
129#define HPET_PERIOD 0x004 /**< Clock period in femtoseconds. */
130#define HPET_CFG 0x010 /**< Configuration register. */
131#define HPET_STATUS 0x020 /**< Status register. */
132#define HPET_COUNTER 0x0f0 /**< Main HPET counter. */
133/** @} */
134
135/** @name Timer N offsets (within each timer's space).
136 * @{ */
137#define HPET_TN_CFG 0x000 /**< Timer N configuration. */
138#define HPET_TN_CMP 0x008 /**< Timer N comparator. */
139#define HPET_TN_ROUTE 0x010 /**< Timer N interrupt route. */
140/** @} */
141
142#define HPET_CFG_WRITE_MASK 0x3
143
144#define HPET_TN_INT_TYPE RT_BIT_64(1)
145#define HPET_TN_ENABLE RT_BIT_64(2)
146#define HPET_TN_PERIODIC RT_BIT_64(3)
147#define HPET_TN_PERIODIC_CAP RT_BIT_64(4)
148#define HPET_TN_SIZE_CAP RT_BIT_64(5)
149#define HPET_TN_SETVAL RT_BIT_64(6) /**< Periodic timers only: Change COMPARATOR as well as ACCUMULATOR. */
150#define HPET_TN_32BIT RT_BIT_64(8)
151#define HPET_TN_INT_ROUTE_MASK UINT64_C(0x3e00)
152#define HPET_TN_CFG_WRITE_MASK UINT64_C(0x3e46)
153#define HPET_TN_INT_ROUTE_SHIFT 9
154#define HPET_TN_INT_ROUTE_CAP_SHIFT 32
155
156#define HPET_TN_CFG_BITS_READONLY_OR_RESERVED 0xffff80b1U
157
158/** Extract the timer count from the capabilities. */
159#define HPET_CAP_GET_TIMERS(a_u32) ((((a_u32) >> 8) + 1) & 0x1f)
160/** Revision ID. */
161#define HPET_CAP_GET_REV_ID(a_u32) ((a_u32) & 0xff)
162/** Counter size. */
163#define HPET_CAP_HAS_64BIT_COUNT_SIZE(a_u32) RT_BOOL((a_u32) & RT_BIT(13))
164/** Legacy Replacement Route. */
165#define HPET_CAP_HAS_LEG_RT(a_u32) RT_BOOL((a_u32) & RT_BIT(15))
166
167
168/** The version of the saved state. */
169#define HPET_SAVED_STATE_VERSION 3
170/** The version of the saved state prior to the off-by-1 timer count fix. */
171#define HPET_SAVED_STATE_VERSION_PRE_TIMER 2
172/** Empty saved state */
173#define HPET_SAVED_STATE_VERSION_EMPTY 1
174
175
176/**
177 * Acquires the HPET lock or returns.
178 */
179#define DEVHPET_LOCK_RETURN(a_pDevIns, a_pThis, a_rcBusy) \
180 do { \
181 int rcLock = PDMDevHlpCritSectEnter((a_pDevIns), &(a_pThis)->CritSect, (a_rcBusy)); \
182 if (RT_LIKELY(rcLock == VINF_SUCCESS)) \
183 { /* likely */ } \
184 else \
185 return rcLock; \
186 } while (0)
187
188/**
189 * Releases the HPET lock.
190 */
191#define DEVHPET_UNLOCK(a_pDevIns, a_pThis) \
192 do { PDMDevHlpCritSectLeave((a_pDevIns), &(a_pThis)->CritSect); } while (0)
193
194
195/**
196 * Acquires the TM lock and HPET lock, returns on failure.
197 * @todo r=bird: Aren't the timers using the same critsect?!?
198 */
199#define DEVHPET_LOCK_BOTH_RETURN(a_pDevIns, a_pThis, a_rcBusy) \
200 do { \
201 VBOXSTRICTRC rcLock = PDMDevHlpTimerLockClock2((a_pDevIns), (a_pThis)->aTimers[0].hTimer, &(a_pThis)->CritSect, (a_rcBusy)); \
202 if (RT_LIKELY(rcLock == VINF_SUCCESS)) \
203 { /* likely */ } \
204 else \
205 return rcLock; \
206 } while (0)
207
208
209/**
210 * Releases the HPET lock and TM lock.
211 */
212#define DEVHPET_UNLOCK_BOTH(a_pDevIns, a_pThis) \
213 PDMDevHlpTimerUnlockClock2((a_pDevIns), (a_pThis)->aTimers[0].hTimer, &(a_pThis)->CritSect)
214
215
216/*********************************************************************************************************************************
217* Structures and Typedefs *
218*********************************************************************************************************************************/
219/**
220 * A HPET timer.
221 *
222 * @note To avoid excessive locking, we many of the updates atomically.
223 */
224typedef struct HPETTIMER
225{
226 /** The HPET timer. */
227 TMTIMERHANDLE hTimer;
228
229 /** Timer index. */
230 uint8_t idxTimer;
231 /** Wrap. */
232 uint8_t u8Wrap;
233 /** Explicit padding. */
234 uint8_t abPadding[6];
235
236 /** @name Memory-mapped, software visible timer registers.
237 * @{ */
238 /** Configuration/capabilities. */
239 uint64_t u64Config;
240 /** Comparator. */
241 uint64_t u64Cmp;
242 /** FSB route, not supported now. */
243 uint64_t u64Fsb;
244 /** @} */
245
246 /** @name Hidden register state.
247 * @{ */
248 /** Accumulator / Last value written to comparator. */
249 uint64_t u64Period;
250 /** @} */
251
252 STAMCOUNTER StatSetIrq;
253 STAMCOUNTER StatSetTimer;
254} HPETTIMER;
255AssertCompileMemberAlignment(HPETTIMER, u64Config, sizeof(uint64_t));
256AssertCompileSizeAlignment(HPETTIMER, 64);
257/** Pointer to the shared state of an HPET timer. */
258typedef HPETTIMER *PHPETTIMER;
259/** Const pointer to the shared state of an HPET timer. */
260typedef HPETTIMER const *PCHPETTIMER;
261
262
263/**
264 * The shared HPET device state.
265 */
266typedef struct HPET
267{
268 /** Timer structures. */
269 HPETTIMER aTimers[RT_MAX(HPET_NUM_TIMERS_PIIX, HPET_NUM_TIMERS_ICH9)];
270
271 /** Offset realtive to the virtual sync clock. */
272 uint64_t u64HpetOffset;
273
274 /** @name Memory-mapped, software visible registers
275 * @{ */
276 /** Capabilities. */
277 uint32_t u32Capabilities;
278 /** Used to be u32Period. We only implement two period values depending on
279 * fIch9, and since we usually would have to RT_MIN(u32Period,1) we could
280 * just as well select between HPET_CLK_PERIOD_ICH9 and HPET_CLK_PERIOD_PIIX. */
281 uint32_t u32Padding;
282 /** Configuration. */
283 uint64_t u64HpetConfig;
284 /** Interrupt status register. */
285 uint64_t u64Isr;
286 /** Main counter. */
287 uint64_t u64HpetCounter;
288 /** @} */
289
290 /** Whether we emulate ICH9 HPET (different frequency & timer count). */
291 bool fIch9;
292 /** Size alignment padding. */
293 uint8_t abPadding0[7+8];
294
295 /** The handle of the MMIO region. */
296 IOMMMIOHANDLE hMmio;
297
298 /** Global device lock. */
299 PDMCRITSECT CritSect;
300
301 STAMCOUNTER StatCounterRead4Byte;
302 STAMCOUNTER StatCounterRead8Byte;
303 STAMCOUNTER StatCounterWriteLow;
304 STAMCOUNTER StatCounterWriteHigh;
305 STAMCOUNTER StatZeroDeltaHack;
306} HPET;
307AssertCompileMemberAlignment(HPET, aTimers, 64);
308AssertCompileMemberAlignment(HPET, CritSect, 64);
309/** Pointer to the shared HPET device state. */
310typedef HPET *PHPET;
311/** Const pointer to the shared HPET device state. */
312typedef const HPET *PCHPET;
313
314
315/**
316 * The ring-3 specific HPET device state.
317 */
318typedef struct HPETR3
319{
320 /** The HPET helpers. */
321 PCPDMHPETHLPR3 pHpetHlp;
322} HPETR3;
323/** Pointer to the ring-3 specific HPET device state. */
324typedef HPETR3 *PHPETR3;
325
326
327/**
328 * The ring-0 specific HPET device state.
329 */
330typedef struct HPETR0
331{
332 /** The HPET helpers. */
333 PCPDMHPETHLPR0 pHpetHlp;
334} HPETR0;
335/** Pointer to the ring-0 specific HPET device state. */
336typedef HPETR0 *PHPETR0;
337
338
339/**
340 * The raw-mode specific HPET device state.
341 */
342typedef struct HPETRC
343{
344 /** The HPET helpers. */
345 PCPDMHPETHLPRC pHpetHlp;
346} HPETRC;
347/** Pointer to the raw-mode specific HPET device state. */
348typedef HPETRC *PHPETRC;
349
350
351/** The HPET device state specific to the current context. */
352typedef CTX_SUFF(HPET) HPETCC;
353/** Pointer to the HPET device state specific to the current context. */
354typedef CTX_SUFF(PHPET) PHPETCC;
355
356
357#ifndef VBOX_DEVICE_STRUCT_TESTCASE
358
359DECLINLINE(bool) hpet32bitTimerEx(uint64_t fConfig)
360{
361 return !(fConfig & HPET_TN_SIZE_CAP)
362 || (fConfig & HPET_TN_32BIT);
363}
364
365
366DECLINLINE(bool) hpet32bitTimer(PHPETTIMER pHpetTimer)
367{
368 return hpet32bitTimerEx(ASMAtomicUoReadU64(&pHpetTimer->u64Config));
369}
370
371
372DECLINLINE(uint64_t) hpetInvalidValue(PHPETTIMER pHpetTimer)
373{
374 return hpet32bitTimer(pHpetTimer) ? UINT32_MAX : UINT64_MAX;
375}
376
377DECLINLINE(uint64_t) hpetTicksToNs(PHPET pThis, uint64_t value)
378{
379 return ASMMultU64ByU32DivByU32(value, pThis->fIch9 ? HPET_CLK_PERIOD_ICH9 : HPET_CLK_PERIOD_PIIX, FS_PER_NS);
380}
381
382DECLINLINE(uint64_t) nsToHpetTicks(PCHPET pThis, uint64_t u64Value)
383{
384 return ASMMultU64ByU32DivByU32(u64Value, FS_PER_NS, pThis->fIch9 ? HPET_CLK_PERIOD_ICH9 : HPET_CLK_PERIOD_PIIX);
385}
386
387DECLINLINE(uint64_t) hpetGetTicksEx(PCHPET pThis, uint64_t tsNow)
388{
389 return nsToHpetTicks(pThis, tsNow + pThis->u64HpetOffset);
390}
391
392DECLINLINE(uint64_t) hpetUpdateMasked(uint64_t u64NewValue, uint64_t u64OldValue, uint64_t u64Mask)
393{
394 u64NewValue &= u64Mask;
395 u64NewValue |= (u64OldValue & ~u64Mask);
396 return u64NewValue;
397}
398
399DECLINLINE(bool) hpetBitJustSet(uint64_t u64OldValue, uint64_t u64NewValue, uint64_t u64Mask)
400{
401 return !(u64OldValue & u64Mask)
402 && !!(u64NewValue & u64Mask);
403}
404
405DECLINLINE(bool) hpetBitJustCleared(uint64_t u64OldValue, uint64_t u64NewValue, uint64_t u64Mask)
406{
407 return !!(u64OldValue & u64Mask)
408 && !(u64NewValue & u64Mask);
409}
410
411DECLINLINE(uint64_t) hpetComputeDiff(uint64_t fConfig, uint64_t uCmp, uint64_t uHpetNow)
412{
413 if (hpet32bitTimerEx(fConfig))
414 {
415 uint32_t u32Diff = (uint32_t)uCmp - (uint32_t)uHpetNow;
416 if ((int32_t)u32Diff > 0)
417 return u32Diff;
418 }
419 else
420 {
421 uint64_t u64Diff = uCmp - uHpetNow;
422 if ((int64_t)u64Diff > 0)
423 return u64Diff;
424 }
425 return 0;
426}
427
428
429DECLINLINE(uint64_t) hpetAdjustComparator(PHPETTIMER pHpetTimer, uint64_t fConfig, uint64_t uCmp,
430 uint64_t uPeriod, uint64_t uHpetNow)
431{
432 if (fConfig & HPET_TN_PERIODIC)
433 {
434 if (uPeriod)
435 {
436 uint64_t cPeriods = (uHpetNow - uCmp) / uPeriod;
437 uCmp += (cPeriods + 1) * uPeriod;
438 ASMAtomicWriteU64(&pHpetTimer->u64Cmp, uCmp);
439 }
440 }
441 return uCmp;
442}
443
444
445/**
446 * Sets the frequency hint if it's a periodic timer.
447 *
448 * @param pDevIns The device instance.
449 * @param pThis The shared HPET state.
450 * @param pHpetTimer The timer.
451 * @param fConfig Already read config value.
452 * @param uPeriod Already read period value.
453 */
454DECLINLINE(void) hpetTimerSetFrequencyHint(PPDMDEVINS pDevIns, PHPET pThis, PHPETTIMER pHpetTimer,
455 uint64_t fConfig, uint64_t uPeriod)
456{
457 if (fConfig & HPET_TN_PERIODIC)
458 {
459 uint64_t const nsPeriod = hpetTicksToNs(pThis, uPeriod);
460 if (nsPeriod < RT_NS_100MS)
461 PDMDevHlpTimerSetFrequencyHint(pDevIns, pHpetTimer->hTimer, RT_NS_1SEC / (uint32_t)nsPeriod);
462 }
463}
464
465
466/**
467 * Programs an HPET timer, arming hTimer for the next IRQ.
468 *
469 * @param pDevIns The device instance.
470 * @param pThis The HPET instance data.
471 * @param pHpetTimer The HPET timer to program. The wrap-around indicator is
472 * updates, and for periodic timer the comparator.
473 * @param tsNow The current virtual sync clock time.
474 * @note Caller must both the virtual sync (timer) and HPET locks.
475 */
476static void hpetProgramTimer(PPDMDEVINS pDevIns, PHPET pThis, PHPETTIMER pHpetTimer, uint64_t const tsNow)
477{
478 /*
479 * Calculate the number of HPET ticks to the next timer IRQ, but
480 * first updating comparator if periodic timer.
481 */
482 uint64_t const fConfig = pHpetTimer->u64Config;
483 uint64_t const uPeriod = pHpetTimer->u64Period;
484 uint64_t uCmp = pHpetTimer->u64Cmp;
485 uint64_t const uHpetNow = hpetGetTicksEx(pThis, tsNow);
486 uCmp = hpetAdjustComparator(pHpetTimer, fConfig, uCmp, uPeriod, uHpetNow);
487 uint64_t uHpetDelta = hpetComputeDiff(fConfig, uCmp, uHpetNow);
488
489 /*
490 * HPET spec says in one-shot 32-bit mode, generate an interrupt when
491 * counter wraps in addition to an interrupt with comparator match.
492 */
493 bool fWrap = false;
494 if ( hpet32bitTimerEx(fConfig)
495 && !(fConfig & HPET_TN_PERIODIC))
496 {
497 uint32_t cHpetTicksTillWrap = UINT32_MAX - (uint32_t)uHpetNow + 1;
498 if (cHpetTicksTillWrap < (uint32_t)uHpetDelta)
499 {
500 Log(("HPET[%u]: wrap: till=%u ticks=%lld diff64=%lld\n",
501 pHpetTimer->idxTimer, cHpetTicksTillWrap, uHpetNow, uHpetDelta));
502 uHpetDelta = cHpetTicksTillWrap;
503 fWrap = true;
504 }
505 }
506 pHpetTimer->u8Wrap = fWrap;
507
508 /*
509 * HACK ALERT! Avoid killing VM with interrupts.
510 */
511#if 1 /** @todo HACK, rethink, may have negative impact on the guest */
512 if (uHpetDelta != 0)
513 { /* likely? */ }
514 else
515 {
516 Log(("HPET[%u]: Applying zero delta hack!\n", pHpetTimer->idxTimer));
517 STAM_REL_COUNTER_INC(&pThis->StatZeroDeltaHack);
518/** @todo lower this. */
519 uHpetDelta = pThis->fIch9 ? 14318 : 100000; /* 1 millisecond */
520 }
521#endif
522
523 /*
524 * Arm the timer.
525 */
526 uint64_t u64TickLimit = pThis->fIch9 ? HPET_TICKS_IN_100YR_ICH9 : HPET_TICKS_IN_100YR_PIIX;
527 if (uHpetDelta <= u64TickLimit)
528 {
529 uint64_t const tsDeadline = tsNow + hpetTicksToNs(pThis, uHpetDelta);
530 Log4(("HPET[%u]: next IRQ in %lld ticks (at %llu)\n", pHpetTimer->idxTimer, uHpetDelta, tsDeadline));
531 PDMDevHlpTimerSet(pDevIns, pHpetTimer->hTimer, tsDeadline);
532 hpetTimerSetFrequencyHint(pDevIns, pThis, pHpetTimer, fConfig, uPeriod);
533 STAM_REL_COUNTER_INC(&pHpetTimer->StatSetTimer);
534 }
535 else
536 LogRelMax(10, ("HPET[%u]: Not scheduling an interrupt more than 100 years in the future.\n", pHpetTimer->idxTimer));
537}
538
539
540/* -=-=-=-=-=- Timer register accesses -=-=-=-=-=- */
541
542
543/**
544 * Reads a HPET timer register.
545 *
546 * @returns The register value.
547 * @param pThis The HPET instance.
548 * @param iTimerNo The timer index.
549 * @param iTimerReg The index of the timer register to read.
550 *
551 * @note No locking required.
552 */
553static uint32_t hpetTimerRegRead32(PHPET pThis, uint32_t iTimerNo, uint32_t iTimerReg)
554{
555 uint32_t u32Value;
556 if ( iTimerNo < HPET_CAP_GET_TIMERS(pThis->u32Capabilities)
557 && iTimerNo < RT_ELEMENTS(pThis->aTimers) )
558 {
559 PHPETTIMER pHpetTimer = &pThis->aTimers[iTimerNo];
560 switch (iTimerReg)
561 {
562 case HPET_TN_CFG:
563 u32Value = (uint32_t)ASMAtomicReadU64(&pHpetTimer->u64Config);
564 Log(("HPET[%u]: read32 HPET_TN_CFG: %#x\n", iTimerNo, u32Value));
565 break;
566
567 case HPET_TN_CFG + 4:
568 u32Value = (uint32_t)(ASMAtomicReadU64(&pHpetTimer->u64Config) >> 32);
569 Log(("HPET[%u]: read32 HPET_TN_CFG+4: %#x\n", iTimerNo, u32Value));
570 break;
571
572 case HPET_TN_CMP:
573 {
574 uint64_t uCmp = ASMAtomicReadU64(&pHpetTimer->u64Cmp);
575 u32Value = (uint32_t)uCmp;
576 Log(("HPET[%u]: read32 HPET_TN_CMP: %#x (%#RX64)\n", pHpetTimer->idxTimer, u32Value, uCmp));
577 break;
578 }
579
580 case HPET_TN_CMP + 4:
581 {
582 uint64_t uCmp = ASMAtomicReadU64(&pHpetTimer->u64Cmp);
583 u32Value = (uint32_t)(uCmp >> 32);
584 Log(("HPET[%u]: read32 HPET_TN_CMP+4: %#x (%#RX64)\n", pHpetTimer->idxTimer, u32Value, uCmp));
585 break;
586 }
587
588 case HPET_TN_ROUTE:
589 u32Value = (uint32_t)(pHpetTimer->u64Fsb >> 32); /** @todo Looks wrong, but since it's not supported, who cares. */
590 Log(("HPET[%u]: read32 HPET_TN_ROUTE: %#x\n", iTimerNo, u32Value));
591 break;
592
593 default:
594 LogRelMax(10, ("HPET[%u]: Invalid HPET register read: %d\n", iTimerNo, iTimerReg));
595 u32Value = 0;
596 break;
597 }
598 }
599 else
600 {
601 LogRelMax(10, ("HPET: Using timer above configured range: %d\n", iTimerNo));
602 u32Value = 0;
603 }
604 return u32Value;
605}
606
607
608/**
609 * Reads a HPET timer register, 64-bit access.
610 *
611 * @returns The register value.
612 * @param pThis The HPET instance.
613 * @param iTimerNo The timer index.
614 * @param iTimerReg The index of the timer register to read.
615 */
616static uint64_t hpetTimerRegRead64(PHPET pThis, uint32_t iTimerNo, uint32_t iTimerReg)
617{
618 uint64_t u64Value;
619 if ( iTimerNo < HPET_CAP_GET_TIMERS(pThis->u32Capabilities)
620 && iTimerNo < RT_ELEMENTS(pThis->aTimers) )
621 {
622 PHPETTIMER pHpetTimer = &pThis->aTimers[iTimerNo];
623 switch (iTimerReg)
624 {
625 case HPET_TN_CFG:
626 u64Value = ASMAtomicReadU64(&pHpetTimer->u64Config);
627 Log(("HPET[%u]: read64 HPET_TN_CFG: %#RX64\n", iTimerNo, u64Value));
628 break;
629
630 case HPET_TN_CMP:
631 u64Value = ASMAtomicReadU64(&pHpetTimer->u64Config);
632 Log(("HPET[%u]: read64 HPET_TN_CMP: %#RX64\n", iTimerNo, u64Value));
633 break;
634
635 case HPET_TN_ROUTE:
636 u64Value = (uint32_t)(pHpetTimer->u64Fsb >> 32); /** @todo Looks wrong, but since it's not supported, who cares. */
637 Log(("HPET[%u]: read64 HPET_TN_ROUTE: %#RX64\n", iTimerNo, u64Value));
638 break;
639
640 default:
641 LogRelMax(10, ("HPET[%u]: Invalid 64-bit HPET register read64: %d\n", iTimerNo, iTimerReg));
642 u64Value = 0;
643 break;
644 }
645 }
646 else
647 {
648 LogRelMax(10, ("HPET: Using timer above configured range: %d\n", iTimerNo));
649 u64Value = 0;
650 }
651 return u64Value;
652}
653
654
655/**
656 * 32-bit write to a HPET timer register.
657 *
658 * @returns Strict VBox status code.
659 *
660 * @param pDevIns The device instance.
661 * @param pThis The shared HPET state.
662 * @param iTimerNo The timer being written to.
663 * @param iTimerReg The register being written to.
664 * @param u32NewValue The value being written.
665 *
666 * @remarks The caller should not hold any locks.
667 */
668static VBOXSTRICTRC hpetTimerRegWrite32(PPDMDEVINS pDevIns, PHPET pThis, uint32_t iTimerNo,
669 uint32_t iTimerReg, uint32_t u32NewValue)
670{
671 Assert(!PDMDevHlpCritSectIsOwner(pDevIns, &pThis->CritSect));
672 Assert(!PDMDevHlpTimerIsLockOwner(pDevIns, pThis->aTimers[0].hTimer));
673
674 if ( iTimerNo < HPET_CAP_GET_TIMERS(pThis->u32Capabilities)
675 && iTimerNo < RT_ELEMENTS(pThis->aTimers) ) /* Parfait - see above. */
676 {
677 PHPETTIMER pHpetTimer = &pThis->aTimers[iTimerNo];
678
679 switch (iTimerReg)
680 {
681 case HPET_TN_CFG:
682 {
683 /*
684 * Calculate the writable mask and see if anything actually changed
685 * before doing any locking. Windows 10 (1809) does two CFG writes
686 * with the same value (0x134) when reprogramming the HPET#0 timer.
687 */
688 uint64_t fConfig = ASMAtomicUoReadU64(&pHpetTimer->u64Config);
689 uint64_t const fMask = HPET_TN_CFG_WRITE_MASK
690 | (fConfig & HPET_TN_PERIODIC_CAP ? HPET_TN_PERIODIC : 0)
691 | (fConfig & HPET_TN_SIZE_CAP ? HPET_TN_32BIT : 0);
692 if ((u32NewValue & fMask) == (fConfig & fMask))
693 Log(("HPET[%u]: write32 HPET_TN_CFG: %#x - no change (%#RX64)\n", iTimerNo, u32NewValue, fConfig));
694 else
695 {
696#ifndef IN_RING3
697 /* Return to ring-3 (where LogRel works) to complain about level-triggered interrupts. */
698 if ((u32NewValue & HPET_TN_INT_TYPE) == HPET_TIMER_TYPE_LEVEL)
699 return VINF_IOM_R3_MMIO_WRITE;
700#endif
701 DEVHPET_LOCK_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_WRITE);
702
703 fConfig = ASMAtomicUoReadU64(&pHpetTimer->u64Config);
704 uint64_t const fConfigNew = hpetUpdateMasked(u32NewValue, fConfig, fMask);
705 Log(("HPET[%u]: write HPET_TN_CFG: %#RX64 -> %#RX64\n", iTimerNo, fConfig, fConfigNew));
706
707 if ((fConfigNew & HPET_TN_32BIT) == (fConfig & HPET_TN_32BIT))
708 { /* likely it stays the same */ }
709 else if (fConfigNew & HPET_TN_32BIT)
710 {
711 Log(("HPET[%u]: Changing timer to 32-bit mode.\n", iTimerNo));
712 /* Clear the top bits of the comparator and period to be on the safe side. */
713 ASMAtomicUoWriteU64(&pHpetTimer->u64Cmp, (uint32_t)pHpetTimer->u64Cmp);
714 ASMAtomicUoWriteU64(&pHpetTimer->u64Period, (uint32_t)pHpetTimer->u64Period);
715 }
716 else
717 Log(("HPET[%u]: Changing timer to 64-bit mode.\n", iTimerNo));
718 ASMAtomicWriteU64(&pHpetTimer->u64Config, fConfigNew);
719
720 DEVHPET_UNLOCK(pDevIns, pThis);
721
722 if (RT_LIKELY((fConfigNew & HPET_TN_INT_TYPE) != HPET_TIMER_TYPE_LEVEL))
723 { /* likely */ }
724 else
725 {
726 LogRelMax(10, ("HPET[%u]: Level-triggered config not yet supported\n", iTimerNo));
727 ASSERT_GUEST_MSG_FAILED(("Level-triggered config not yet supported"));
728 }
729 }
730 break;
731 }
732
733 case HPET_TN_CFG + 4: /* Interrupt capabilities - read only. */
734 Log(("HPET[%u]: write32 HPET_TN_CFG + 4 (ignored)\n", iTimerNo));
735 break;
736
737 case HPET_TN_CMP: /* lower bits of comparator register */
738 {
739 DEVHPET_LOCK_BOTH_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_WRITE);
740 uint64_t fConfig = ASMAtomicUoReadU64(&pHpetTimer->u64Config);
741 Log(("HPET[%u]: write32 HPET_TN_CMP: %#x (fCfg=%#RX32)\n", iTimerNo, u32NewValue, (uint32_t)fConfig));
742
743 if (fConfig & HPET_TN_PERIODIC)
744 ASMAtomicUoWriteU64(&pHpetTimer->u64Period, RT_MAKE_U64(u32NewValue, RT_HI_U32(pHpetTimer->u64Period)));
745
746 if (!(fConfig & HPET_TN_PERIODIC) || (fConfig & HPET_TN_SETVAL))
747 ASMAtomicUoWriteU64(&pHpetTimer->u64Cmp, RT_MAKE_U64(u32NewValue, RT_HI_U32(pHpetTimer->u64Cmp)));
748
749 ASMAtomicAndU64(&pHpetTimer->u64Config, ~HPET_TN_SETVAL);
750 Log2(("HPET[%u]: after32 HPET_TN_CMP cmp=%#llx per=%#llx\n", iTimerNo, pHpetTimer->u64Cmp, pHpetTimer->u64Period));
751
752 if (pThis->u64HpetConfig & HPET_CFG_ENABLE)
753 hpetProgramTimer(pDevIns, pThis, pHpetTimer, PDMDevHlpTimerGet(pDevIns, pHpetTimer->hTimer));
754 DEVHPET_UNLOCK_BOTH(pDevIns, pThis);
755 break;
756 }
757
758 /** @todo figure out how exactly it behaves wrt to HPET_TN_SETVAL */
759 case HPET_TN_CMP + 4: /* upper bits of comparator register */
760 {
761 DEVHPET_LOCK_BOTH_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_WRITE);
762 uint64_t fConfig = ASMAtomicUoReadU64(&pHpetTimer->u64Config);
763
764 if (!hpet32bitTimerEx(fConfig))
765 {
766 Log(("HPET[%u]: write32 HPET_TN_CMP + 4: %#x (fCfg=%#RX32)\n", iTimerNo, u32NewValue, (uint32_t)fConfig));
767 if (fConfig & HPET_TN_PERIODIC)
768 ASMAtomicUoWriteU64(&pHpetTimer->u64Period, RT_MAKE_U64(RT_LO_U32(pHpetTimer->u64Period), u32NewValue));
769
770 if (!(fConfig & HPET_TN_PERIODIC) || (fConfig & HPET_TN_SETVAL))
771 ASMAtomicUoWriteU64(&pHpetTimer->u64Cmp, RT_MAKE_U64(RT_LO_U32(pHpetTimer->u64Cmp), u32NewValue));
772
773 ASMAtomicAndU64(&pHpetTimer->u64Config, ~HPET_TN_SETVAL);
774 Log2(("HPET[%u]: after32 HPET_TN_CMP+4: cmp=%#llx per=%#llx\n", iTimerNo, pHpetTimer->u64Cmp, pHpetTimer->u64Period));
775
776 if (pThis->u64HpetConfig & HPET_CFG_ENABLE)
777 hpetProgramTimer(pDevIns, pThis, pHpetTimer, PDMDevHlpTimerGet(pDevIns, pHpetTimer->hTimer));
778 }
779 else
780 Log(("HPET[%u]: write32 HPET_TN_CMP + 4: %#x - but timer is 32-bit!! (fCfg=%#RX32)\n", iTimerNo, u32NewValue, (uint32_t)fConfig));
781 DEVHPET_UNLOCK_BOTH(pDevIns, pThis);
782 break;
783 }
784
785 case HPET_TN_ROUTE:
786 Log(("HPET[%u]: write32 HPET_TN_ROUTE (ignored)\n", iTimerNo));
787 break;
788
789 case HPET_TN_ROUTE + 4:
790 Log(("HPET[%u]: write32 HPET_TN_ROUTE + 4 (ignored)\n", iTimerNo));
791 break;
792
793 default:
794 LogRelMax(10, ("HPET[%u]: Invalid timer register write: %d\n", iTimerNo, iTimerReg));
795 break;
796 }
797 }
798 else
799 LogRelMax(10, ("HPET: Using timer above configured range: %d (reg %#x)\n", iTimerNo, iTimerReg));
800 return VINF_SUCCESS;
801}
802
803
804/**
805 * 32-bit write to a HPET timer register.
806 *
807 * @returns Strict VBox status code.
808 *
809 * @param pDevIns The device instance.
810 * @param pThis The shared HPET state.
811 * @param iTimerNo The timer being written to.
812 * @param iTimerReg The register being written to.
813 * @param u64NewValue The value being written.
814 *
815 * @remarks The caller should not hold any locks.
816 */
817static VBOXSTRICTRC hpetTimerRegWrite64(PPDMDEVINS pDevIns, PHPET pThis, uint32_t iTimerNo,
818 uint32_t iTimerReg, uint64_t u64NewValue)
819{
820 Assert(!PDMDevHlpCritSectIsOwner(pDevIns, &pThis->CritSect));
821 Assert(!PDMDevHlpTimerIsLockOwner(pDevIns, pThis->aTimers[0].hTimer));
822 Assert(!(iTimerReg & 7));
823
824 if ( iTimerNo < HPET_CAP_GET_TIMERS(pThis->u32Capabilities)
825 && iTimerNo < RT_ELEMENTS(pThis->aTimers) ) /* Parfait - see above. */
826 {
827 PHPETTIMER pHpetTimer = &pThis->aTimers[iTimerNo];
828
829 switch (iTimerReg)
830 {
831 case HPET_TN_CFG:
832 /* The upper 32 bits are not writable, so join paths with the 32-bit version. */
833 return hpetTimerRegWrite32(pDevIns, pThis, iTimerNo, iTimerReg, (uint32_t)u64NewValue);
834
835 case HPET_TN_CMP:
836 {
837 DEVHPET_LOCK_BOTH_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_WRITE);
838 uint64_t fConfig = ASMAtomicUoReadU64(&pHpetTimer->u64Config);
839 Log(("HPET[%u]: write64 HPET_TN_CMP: %#RX64 (fCfg=%#RX64)\n", iTimerNo, u64NewValue, (uint32_t)fConfig));
840
841 /** @todo not sure if this is right, but it is consistent with the 32-bit config
842 * change behaviour and defensive wrt mixups. */
843 if (!hpet32bitTimerEx(fConfig))
844 { /* likely */ }
845 else
846 u64NewValue = (uint32_t)u64NewValue;
847
848 if (fConfig & HPET_TN_PERIODIC)
849 ASMAtomicUoWriteU64(&pHpetTimer->u64Period, u64NewValue);
850
851 if (!(fConfig & HPET_TN_PERIODIC) || (fConfig & HPET_TN_SETVAL))
852 ASMAtomicUoWriteU64(&pHpetTimer->u64Cmp, u64NewValue);
853
854 ASMAtomicAndU64(&pHpetTimer->u64Config, ~HPET_TN_SETVAL);
855 Log2(("HPET[%u]: after64 HPET_TN_CMP cmp=%#llx per=%#llx\n", iTimerNo, pHpetTimer->u64Cmp, pHpetTimer->u64Period));
856
857 if (pThis->u64HpetConfig & HPET_CFG_ENABLE)
858 hpetProgramTimer(pDevIns, pThis, pHpetTimer, PDMDevHlpTimerGet(pDevIns, pHpetTimer->hTimer));
859 DEVHPET_UNLOCK_BOTH(pDevIns, pThis);
860 break;
861 }
862
863 case HPET_TN_ROUTE:
864 Log(("HPET[%u]: write64 HPET_TN_ROUTE (ignored)\n", iTimerNo));
865 break;
866
867 default:
868 LogRelMax(10, ("HPET[%u]: Invalid timer register write: %d\n", iTimerNo, iTimerReg));
869 break;
870 }
871 }
872 else
873 LogRelMax(10, ("HPET: Using timer above configured range: %d (reg %#x)\n", iTimerNo, iTimerReg));
874 return VINF_SUCCESS;
875}
876
877
878/* -=-=-=-=-=- Non-timer register accesses -=-=-=-=-=- */
879
880
881/**
882 * Read a 32-bit HPET register.
883 *
884 * @returns Strict VBox status code.
885 * @param pDevIns The device instance.
886 * @param pThis The shared HPET state.
887 * @param idxReg The register to read.
888 * @param pu32Value Where to return the register value.
889 *
890 * @remarks The caller must not own the device lock if HPET_COUNTER is read.
891 */
892static VBOXSTRICTRC hpetConfigRegRead32(PPDMDEVINS pDevIns, PHPET pThis, uint32_t idxReg, uint32_t *pu32Value)
893{
894 Assert(!PDMDevHlpCritSectIsOwner(pDevIns, &pThis->CritSect) || (idxReg != HPET_COUNTER && idxReg != HPET_COUNTER + 4));
895
896 uint32_t u32Value;
897 switch (idxReg)
898 {
899 case HPET_ID:
900 DEVHPET_LOCK_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_READ);
901 u32Value = pThis->u32Capabilities;
902 DEVHPET_UNLOCK(pDevIns, pThis);
903 Log(("read HPET_ID: %#x\n", u32Value));
904 break;
905
906 case HPET_PERIOD:
907 u32Value = pThis->fIch9 ? HPET_CLK_PERIOD_ICH9 : HPET_CLK_PERIOD_PIIX;
908 Log(("read HPET_PERIOD: %#x\n", u32Value));
909 break;
910
911 case HPET_CFG:
912 DEVHPET_LOCK_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_READ);
913 u32Value = (uint32_t)pThis->u64HpetConfig;
914 DEVHPET_UNLOCK(pDevIns, pThis);
915 Log(("read HPET_CFG: %#x\n", u32Value));
916 break;
917
918 case HPET_CFG + 4:
919 DEVHPET_LOCK_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_READ);
920 u32Value = (uint32_t)(pThis->u64HpetConfig >> 32);
921 DEVHPET_UNLOCK(pDevIns, pThis);
922 Log(("read of HPET_CFG + 4: %#x\n", u32Value));
923 break;
924
925 case HPET_COUNTER:
926 case HPET_COUNTER + 4:
927 {
928 /** @todo We don't technically need to sit on the virtualsync lock here to
929 * read it, but it helps wrt quality... */
930 DEVHPET_LOCK_BOTH_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_READ);
931
932 uint64_t u64Ticks;
933 if (pThis->u64HpetConfig & HPET_CFG_ENABLE)
934 {
935 uint64_t const tsNow = PDMDevHlpTimerGet(pDevIns, pThis->aTimers[0].hTimer);
936 PDMDevHlpTimerUnlockClock(pDevIns, pThis->aTimers[0].hTimer);
937 u64Ticks = hpetGetTicksEx(pThis, tsNow);
938 }
939 else
940 {
941 PDMDevHlpTimerUnlockClock(pDevIns, pThis->aTimers[0].hTimer);
942 u64Ticks = pThis->u64HpetCounter;
943 }
944
945 STAM_REL_COUNTER_INC(&pThis->StatCounterRead4Byte);
946 DEVHPET_UNLOCK(pDevIns, pThis);
947
948 /** @todo is it correct? */
949 u32Value = idxReg == HPET_COUNTER ? (uint32_t)u64Ticks : (uint32_t)(u64Ticks >> 32);
950 Log(("read HPET_COUNTER: %s part value %x (%#llx)\n", (idxReg == HPET_COUNTER) ? "low" : "high", u32Value, u64Ticks));
951 break;
952 }
953
954 case HPET_STATUS:
955 DEVHPET_LOCK_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_READ);
956 u32Value = (uint32_t)pThis->u64Isr;
957 DEVHPET_UNLOCK(pDevIns, pThis);
958 Log(("read HPET_STATUS: %#x\n", u32Value));
959 break;
960
961 default:
962 Log(("invalid HPET register read: %x\n", idxReg));
963 u32Value = 0;
964 break;
965 }
966
967 *pu32Value = u32Value;
968 return VINF_SUCCESS;
969}
970
971
972/**
973 * 32-bit write to a config register.
974 *
975 * @returns Strict VBox status code.
976 *
977 * @param pDevIns The device instance.
978 * @param pThis The shared HPET state.
979 * @param idxReg The register being written to.
980 * @param u32NewValue The value being written.
981 *
982 * @remarks The caller should not hold the device lock, unless it also holds
983 * the TM lock.
984 */
985static VBOXSTRICTRC hpetConfigRegWrite32(PPDMDEVINS pDevIns, PHPET pThis, uint32_t idxReg, uint32_t u32NewValue)
986{
987 Assert(!PDMDevHlpCritSectIsOwner(pDevIns, &pThis->CritSect) || PDMDevHlpTimerIsLockOwner(pDevIns, pThis->aTimers[0].hTimer));
988
989 VBOXSTRICTRC rc = VINF_SUCCESS;
990 switch (idxReg)
991 {
992 case HPET_ID:
993 case HPET_ID + 4:
994 {
995 Log(("write HPET_ID, useless\n"));
996 break;
997 }
998
999 case HPET_CFG:
1000 {
1001 DEVHPET_LOCK_BOTH_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_WRITE);
1002 uint32_t const iOldValue = (uint32_t)(pThis->u64HpetConfig);
1003 Log(("write HPET_CFG: %x (old %x)\n", u32NewValue, iOldValue));
1004
1005 /*
1006 * This check must be here, before actual update, as hpetLegacyMode
1007 * may request retry in R3 - so we must keep state intact.
1008 */
1009 if ((iOldValue ^ u32NewValue) & HPET_CFG_LEGACY)
1010 {
1011#ifdef IN_RING3
1012 PHPETCC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PHPETCC);
1013 if (pThisCC->pHpetHlp != NULL)
1014 {
1015 rc = pThisCC->pHpetHlp->pfnSetLegacyMode(pDevIns, RT_BOOL(u32NewValue & HPET_CFG_LEGACY));
1016 if (rc != VINF_SUCCESS)
1017 {
1018 DEVHPET_UNLOCK_BOTH(pDevIns, pThis);
1019 break;
1020 }
1021 }
1022#else
1023 rc = VINF_IOM_R3_MMIO_WRITE;
1024 DEVHPET_UNLOCK_BOTH(pDevIns, pThis);
1025 break;
1026#endif
1027 }
1028
1029 /* Updating it using an atomic write just to be on the safe side. */
1030 ASMAtomicWriteU64(&pThis->u64HpetConfig, hpetUpdateMasked(u32NewValue, iOldValue, HPET_CFG_WRITE_MASK));
1031
1032 uint32_t const cTimers = RT_MIN(HPET_CAP_GET_TIMERS(pThis->u32Capabilities), RT_ELEMENTS(pThis->aTimers));
1033 if (hpetBitJustSet(iOldValue, u32NewValue, HPET_CFG_ENABLE))
1034 {
1035 /*
1036 * Enable main counter and interrupt generation.
1037 */
1038 uint64_t u64TickLimit = pThis->fIch9 ? HPET_TICKS_IN_100YR_ICH9 : HPET_TICKS_IN_100YR_PIIX;
1039 if (pThis->u64HpetCounter <= u64TickLimit)
1040 pThis->u64HpetOffset = hpetTicksToNs(pThis, pThis->u64HpetCounter);
1041 else
1042 {
1043 LogRelMax(10, ("HPET: Counter set more than 100 years in the future, reducing.\n"));
1044 pThis->u64HpetOffset = 1000000LL * 60 * 60 * 24 * 365 * 100;
1045 }
1046
1047 uint64_t const tsNow = PDMDevHlpTimerGet(pDevIns, pThis->aTimers[0].hTimer);
1048 pThis->u64HpetOffset -= tsNow;
1049
1050 for (uint32_t i = 0; i < cTimers; i++)
1051 if (pThis->aTimers[i].u64Cmp != hpetInvalidValue(&pThis->aTimers[i]))
1052 hpetProgramTimer(pDevIns, pThis, &pThis->aTimers[i], tsNow);
1053 }
1054 else if (hpetBitJustCleared(iOldValue, u32NewValue, HPET_CFG_ENABLE))
1055 {
1056 /*
1057 * Halt main counter and disable interrupt generation.
1058 */
1059 pThis->u64HpetCounter = hpetGetTicksEx(pThis, PDMDevHlpTimerGet(pDevIns, pThis->aTimers[0].hTimer));
1060 for (uint32_t i = 0; i < cTimers; i++)
1061 PDMDevHlpTimerStop(pDevIns, pThis->aTimers[i].hTimer);
1062 }
1063
1064 DEVHPET_UNLOCK_BOTH(pDevIns, pThis);
1065 break;
1066 }
1067
1068 case HPET_CFG + 4:
1069 {
1070/** @todo r=bird: Is the whole upper part of the config register really
1071 * writable? Only 2 bits are writable in the lower part... */
1072 DEVHPET_LOCK_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_WRITE);
1073 pThis->u64HpetConfig = hpetUpdateMasked((uint64_t)u32NewValue << 32,
1074 pThis->u64HpetConfig,
1075 UINT64_C(0xffffffff00000000));
1076 Log(("write HPET_CFG + 4: %x -> %#llx\n", u32NewValue, pThis->u64HpetConfig));
1077 DEVHPET_UNLOCK(pDevIns, pThis);
1078 break;
1079 }
1080
1081 case HPET_STATUS:
1082 {
1083 DEVHPET_LOCK_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_WRITE);
1084 /* Clear ISR for all set bits in u32NewValue, see p. 14 of the HPET spec. */
1085 pThis->u64Isr &= ~((uint64_t)u32NewValue);
1086 Log(("write HPET_STATUS: %x -> ISR=%#llx\n", u32NewValue, pThis->u64Isr));
1087 DEVHPET_UNLOCK(pDevIns, pThis);
1088 break;
1089 }
1090
1091 case HPET_STATUS + 4:
1092 {
1093 Log(("write HPET_STATUS + 4: %x\n", u32NewValue));
1094 if (u32NewValue != 0)
1095 LogRelMax(10, ("HPET: Writing HPET_STATUS + 4 with non-zero, ignored\n"));
1096 break;
1097 }
1098
1099 case HPET_COUNTER:
1100 {
1101 STAM_REL_COUNTER_INC(&pThis->StatCounterWriteLow);
1102 DEVHPET_LOCK_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_WRITE);
1103 pThis->u64HpetCounter = RT_MAKE_U64(u32NewValue, RT_HI_U32(pThis->u64HpetCounter));
1104/** @todo how is this supposed to work if the HPET is enabled? */
1105 Log(("write HPET_COUNTER: %#x -> %llx\n", u32NewValue, pThis->u64HpetCounter));
1106 DEVHPET_UNLOCK(pDevIns, pThis);
1107 break;
1108 }
1109
1110 case HPET_COUNTER + 4:
1111 {
1112 STAM_REL_COUNTER_INC(&pThis->StatCounterWriteHigh);
1113 DEVHPET_LOCK_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_WRITE);
1114 pThis->u64HpetCounter = RT_MAKE_U64(RT_LO_U32(pThis->u64HpetCounter), u32NewValue);
1115 Log(("write HPET_COUNTER + 4: %#x -> %llx\n", u32NewValue, pThis->u64HpetCounter));
1116 DEVHPET_UNLOCK(pDevIns, pThis);
1117 break;
1118 }
1119
1120 default:
1121 LogRelMax(10, ("HPET: Invalid HPET config write: %x\n", idxReg));
1122 break;
1123 }
1124
1125 return rc;
1126}
1127
1128
1129/* -=-=-=-=-=- MMIO callbacks -=-=-=-=-=- */
1130
1131
1132/**
1133 * @callback_method_impl{FNIOMMMIONEWREAD}
1134 */
1135static DECLCALLBACK(VBOXSTRICTRC) hpetMMIORead(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS off, void *pv, unsigned cb)
1136{
1137 HPET *pThis = PDMDEVINS_2_DATA(pDevIns, HPET*);
1138 NOREF(pvUser);
1139 Assert(cb == 4 || cb == 8);
1140 Assert(!(off & (cb - 1)));
1141
1142 LogFlow(("hpetMMIORead (%d): %RGp\n", cb, off));
1143
1144 VBOXSTRICTRC rc;
1145 if (cb == 4)
1146 {
1147 /*
1148 * 4-byte access.
1149 */
1150 if (off >= 0x100 && off < 0x400)
1151 {
1152 *(uint32_t *)pv = hpetTimerRegRead32(pThis,
1153 (uint32_t)(off - 0x100) / 0x20,
1154 (uint32_t)(off - 0x100) % 0x20);
1155 rc = VINF_SUCCESS;
1156 }
1157 else
1158 rc = hpetConfigRegRead32(pDevIns, pThis, off, (uint32_t *)pv);
1159 }
1160 else
1161 {
1162 /*
1163 * 8-byte access - Split the access except for timing sensitive registers.
1164 * The others assume the protection of the lock.
1165 */
1166 PRTUINT64U pValue = (PRTUINT64U)pv;
1167 if (off == HPET_COUNTER)
1168 {
1169 /** @todo We don't technically need to sit on the virtualsync lock here to
1170 * read it, but it helps wrt quality... */
1171 /* When reading HPET counter we must read it in a single read,
1172 to avoid unexpected time jumps on 32-bit overflow. */
1173 DEVHPET_LOCK_BOTH_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_READ);
1174
1175 if (pThis->u64HpetConfig & HPET_CFG_ENABLE)
1176 {
1177 uint64_t const tsNow = PDMDevHlpTimerGet(pDevIns, pThis->aTimers[0].hTimer);
1178 PDMDevHlpTimerUnlockClock(pDevIns, pThis->aTimers[0].hTimer);
1179 pValue->u = hpetGetTicksEx(pThis, tsNow);
1180 }
1181 else
1182 {
1183 PDMDevHlpTimerUnlockClock(pDevIns, pThis->aTimers[0].hTimer);
1184 pValue->u = pThis->u64HpetCounter;
1185 }
1186
1187 STAM_REL_COUNTER_INC(&pThis->StatCounterRead8Byte);
1188 DEVHPET_UNLOCK(pDevIns, pThis);
1189 rc = VINF_SUCCESS;
1190 }
1191 else
1192 {
1193 if (off >= 0x100 && off < 0x400)
1194 {
1195 uint32_t iTimer = (uint32_t)(off - 0x100) / 0x20;
1196 uint32_t iTimerReg = (uint32_t)(off - 0x100) % 0x20;
1197 Assert(!(iTimerReg & 7));
1198 pValue->u = hpetTimerRegRead64(pThis, iTimer, iTimerReg);
1199 rc = VINF_SUCCESS;
1200 }
1201 else
1202 {
1203 /* for most 8-byte accesses we just split them, happens under lock anyway. */
1204 DEVHPET_LOCK_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_READ);
1205 rc = hpetConfigRegRead32(pDevIns, pThis, off, &pValue->s.Lo);
1206 if (rc == VINF_SUCCESS)
1207 rc = hpetConfigRegRead32(pDevIns, pThis, off + 4, &pValue->s.Hi);
1208 DEVHPET_UNLOCK(pDevIns, pThis);
1209 }
1210 }
1211 }
1212 return rc;
1213}
1214
1215
1216/**
1217 * @callback_method_impl{FNIOMMMIONEWWRITE}
1218 */
1219static DECLCALLBACK(VBOXSTRICTRC) hpetMMIOWrite(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS off, void const *pv, unsigned cb)
1220{
1221 HPET *pThis = PDMDEVINS_2_DATA(pDevIns, HPET*);
1222 LogFlow(("hpetMMIOWrite: cb=%u reg=%RGp val=%llx\n",
1223 cb, off, cb == 4 ? *(uint32_t *)pv : cb == 8 ? *(uint64_t *)pv : 0xdeadbeef));
1224 NOREF(pvUser);
1225 Assert(cb == 4 || cb == 8);
1226 Assert(!(off & (cb - 1)));
1227
1228 VBOXSTRICTRC rc;
1229 if (cb == 4)
1230 {
1231 if (off >= 0x100 && off < 0x400)
1232 rc = hpetTimerRegWrite32(pDevIns, pThis,
1233 (uint32_t)(off - 0x100) / 0x20,
1234 (uint32_t)(off - 0x100) % 0x20,
1235 *(uint32_t const *)pv);
1236 else
1237 rc = hpetConfigRegWrite32(pDevIns, pThis, off, *(uint32_t const *)pv);
1238 }
1239 else
1240 {
1241 /*
1242 * 8-byte access.
1243 */
1244 if (off >= 0x100 && off < 0x400)
1245 rc = hpetTimerRegWrite64(pDevIns, pThis,
1246 (uint32_t)(off - 0x100) / 0x20,
1247 (uint32_t)(off - 0x100) % 0x20,
1248 *(uint64_t const *)pv);
1249 else
1250 {
1251 /* Split the access and rely on the locking to prevent trouble. */
1252 RTUINT64U uValue;
1253 uValue.u = *(uint64_t const *)pv;
1254 DEVHPET_LOCK_BOTH_RETURN(pDevIns, pThis, VINF_IOM_R3_MMIO_WRITE);
1255 rc = hpetConfigRegWrite32(pDevIns, pThis, off, uValue.s.Lo);
1256 if (RT_LIKELY(rc == VINF_SUCCESS))
1257 rc = hpetConfigRegWrite32(pDevIns, pThis, off + 4, uValue.s.Hi);
1258 DEVHPET_UNLOCK_BOTH(pDevIns, pThis);
1259 }
1260 }
1261
1262 return rc;
1263}
1264
1265#ifdef IN_RING3
1266
1267/* -=-=-=-=-=- Timer Callback Processing -=-=-=-=-=- */
1268
1269/**
1270 * Gets the IRQ of an HPET timer.
1271 *
1272 * @returns IRQ number.
1273 * @param pThis The shared HPET state.
1274 * @param pHpetTimer The HPET timer.
1275 * @param fConfig The HPET timer config value.
1276 */
1277DECLINLINE(uint32_t) hpetR3TimerGetIrq(PHPET pThis, PCHPETTIMER pHpetTimer, uint64_t fConfig)
1278{
1279 /*
1280 * Per spec, in legacy mode the HPET timers are wired as follows:
1281 * timer 0: IRQ0 for PIC and IRQ2 for APIC
1282 * timer 1: IRQ8 for both PIC and APIC
1283 *
1284 * ISA IRQ delivery logic will take care of correct delivery
1285 * to the different ICs.
1286 */
1287 if ( pHpetTimer->idxTimer <= 1
1288 && (pThis->u64HpetConfig & HPET_CFG_LEGACY))
1289 return pHpetTimer->idxTimer == 0 ? 0 : 8;
1290
1291 return (fConfig & HPET_TN_INT_ROUTE_MASK) >> HPET_TN_INT_ROUTE_SHIFT;
1292}
1293
1294
1295/**
1296 * @callback_method_impl{FNTMTIMERDEV, Device timer callback function.}
1297 *
1298 * @note Only the virtual sync lock is held when called.
1299 */
1300static DECLCALLBACK(void) hpetR3Timer(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, void *pvUser)
1301{
1302 PHPET pThis = PDMDEVINS_2_DATA(pDevIns, PHPET);
1303 PHPETTIMER pHpetTimer = (HPETTIMER *)pvUser;
1304
1305 /*
1306 * Read the timer configuration values we need first.
1307 *
1308 * The comparator and period are only written while owning the virtual sync
1309 * lock, so we don't run any risk there. The configuration register is
1310 * written with only the device lock, so must be a bit more careful with it.
1311 */
1312 uint64_t uCmp = ASMAtomicUoReadU64(&pHpetTimer->u64Cmp);
1313 uint64_t const uPeriod = ASMAtomicUoReadU64(&pHpetTimer->u64Period);
1314 uint64_t const fConfig = ASMAtomicUoReadU64(&pHpetTimer->u64Config);
1315 Assert(hTimer == pHpetTimer->hTimer);
1316
1317 if (fConfig & HPET_TN_PERIODIC)
1318 {
1319 if (uPeriod)
1320 {
1321 uint64_t const tsNow = PDMDevHlpTimerGet(pDevIns, pHpetTimer->hTimer);
1322 uint64_t const uHpetNow = hpetGetTicksEx(pThis, tsNow);
1323 uCmp = hpetAdjustComparator(pHpetTimer, fConfig, uCmp, uPeriod, uHpetNow);
1324 uint64_t const cTicksDiff = hpetComputeDiff(fConfig, uCmp, uHpetNow);
1325 uint64_t const u64TickLimit = pThis->fIch9 ? HPET_TICKS_IN_100YR_ICH9 : HPET_TICKS_IN_100YR_PIIX;
1326 if (cTicksDiff <= u64TickLimit)
1327 {
1328 uint64_t const tsDeadline = tsNow + hpetTicksToNs(pThis, cTicksDiff);
1329 Log4(("HPET: periodic: next in %llu\n", tsDeadline));
1330 PDMDevHlpTimerSet(pDevIns, hTimer, tsDeadline);
1331 STAM_REL_COUNTER_INC(&pHpetTimer->StatSetTimer);
1332 }
1333 else
1334 {
1335 LogRelMax(10, ("HPET: Not scheduling periodic interrupt more than 100 years in the future.\n"));
1336 }
1337 }
1338 }
1339 /* For 32-bit non-periodic timers, generate wrap-around interrupts. */
1340 else if (pHpetTimer->u8Wrap && hpet32bitTimerEx(fConfig))
1341 {
1342 pHpetTimer->u8Wrap = 0; /* (only modified while owning the virtual sync lock) */
1343 uint64_t const tsNow = PDMDevHlpTimerGet(pDevIns, hTimer);
1344 uint64_t const uHpetNow = nsToHpetTicks(pThis, tsNow + pThis->u64HpetOffset);
1345 uint64_t const cTicksDiff = hpetComputeDiff(fConfig, uCmp, uHpetNow);
1346 uint64_t const tsDeadline = tsNow + hpetTicksToNs(pThis, cTicksDiff);
1347 Log4(("HPET: post-wrap deadline: %llu\n", tsDeadline));
1348 PDMDevHlpTimerSet(pDevIns, pHpetTimer->hTimer, tsDeadline);
1349 }
1350
1351 /*
1352 * IRQ update.
1353 */
1354 if ( (fConfig & HPET_TN_ENABLE)
1355 && (pThis->u64HpetConfig & HPET_CFG_ENABLE))
1356 {
1357 AssertCompile(HPET_TN_INT_TYPE == 2);
1358
1359 /* We trigger flip/flop in edge-triggered mode and do nothing in
1360 level-triggered mode yet. */
1361 if ((fConfig & HPET_TN_INT_TYPE) == HPET_TIMER_TYPE_EDGE)
1362 {
1363 PHPETCC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PHPETCC);
1364 AssertReturnVoid(pThisCC);
1365
1366 uint32_t const uIrq = hpetR3TimerGetIrq(pThis, pHpetTimer, fConfig);
1367 Log4(("HPET: raising IRQ %u\n", uIrq));
1368
1369 pThisCC->pHpetHlp->pfnSetIrq(pDevIns, uIrq, PDM_IRQ_LEVEL_FLIP_FLOP);
1370 STAM_REL_COUNTER_INC(&pHpetTimer->StatSetIrq);
1371 }
1372 /* ISR bits are only set in level-triggered mode. */
1373 else
1374 {
1375 Assert((fConfig & HPET_TN_INT_TYPE) == HPET_TIMER_TYPE_LEVEL);
1376 ASMAtomicOrU64(&pThis->u64Isr, RT_BIT_64(pHpetTimer->idxTimer));
1377 /** @todo implement IRQs in level-triggered mode */
1378 }
1379 }
1380
1381}
1382
1383
1384/* -=-=-=-=-=- DBGF Info Handlers -=-=-=-=-=- */
1385
1386
1387/**
1388 * @callback_method_impl{FNDBGFHANDLERDEV}
1389 */
1390static DECLCALLBACK(void) hpetR3Info(PPDMDEVINS pDevIns, PCDBGFINFOHLP pHlp, const char *pszArgs)
1391{
1392 PHPET pThis = PDMDEVINS_2_DATA(pDevIns, PHPET);
1393 NOREF(pszArgs);
1394
1395 pHlp->pfnPrintf(pHlp,
1396 "HPET status:\n"
1397 " config=%016RX64 isr=%016RX64\n"
1398 " offset=%016RX64 counter=%016RX64 frequency=%u fs\n"
1399 " legacy-mode=%s timer-count=%u\n",
1400 pThis->u64HpetConfig, pThis->u64Isr,
1401 pThis->u64HpetOffset, pThis->u64HpetCounter, pThis->fIch9 ? HPET_CLK_PERIOD_ICH9 : HPET_CLK_PERIOD_PIIX,
1402 !!(pThis->u64HpetConfig & HPET_CFG_LEGACY) ? "on " : "off",
1403 HPET_CAP_GET_TIMERS(pThis->u32Capabilities));
1404 pHlp->pfnPrintf(pHlp,
1405 "Timers:\n");
1406 for (unsigned i = 0; i < RT_ELEMENTS(pThis->aTimers); i++)
1407 {
1408 static const struct
1409 {
1410 const char *psz;
1411 uint32_t cch;
1412 uint32_t fFlags;
1413 } s_aFlags[] =
1414 {
1415 { RT_STR_TUPLE(" lvl"), HPET_TN_INT_TYPE },
1416 { RT_STR_TUPLE(" en"), HPET_TN_ENABLE },
1417 { RT_STR_TUPLE(" per"), HPET_TN_PERIODIC },
1418 { RT_STR_TUPLE(" cap_per"), HPET_TN_PERIODIC_CAP },
1419 { RT_STR_TUPLE(" cap_64"), HPET_TN_SIZE_CAP },
1420 { RT_STR_TUPLE(" setval"), HPET_TN_SETVAL },
1421 { RT_STR_TUPLE(" 32b"), HPET_TN_32BIT },
1422 };
1423 char szTmp[64];
1424 uint64_t fCfg = pThis->aTimers[i].u64Config;
1425 size_t off = 0;
1426 for (unsigned j = 0; j < RT_ELEMENTS(s_aFlags); j++)
1427 if (fCfg & s_aFlags[j].fFlags)
1428 {
1429 memcpy(&szTmp[off], s_aFlags[j].psz, s_aFlags[j].cch);
1430 off += s_aFlags[j].cch;
1431 fCfg &= ~(uint64_t)s_aFlags[j].fFlags;
1432 }
1433 szTmp[off] = '\0';
1434 Assert(off < sizeof(szTmp));
1435
1436 pHlp->pfnPrintf(pHlp,
1437 " %d: comparator=%016RX64 accumulator=%016RX64 (%RU64 ns)\n"
1438 " config=%016RX64 irq=%d%s\n",
1439 pThis->aTimers[i].idxTimer,
1440 pThis->aTimers[i].u64Cmp,
1441 pThis->aTimers[i].u64Period,
1442 hpetTicksToNs(pThis, pThis->aTimers[i].u64Period),
1443 pThis->aTimers[i].u64Config,
1444 hpetR3TimerGetIrq(pThis, &pThis->aTimers[i], pThis->aTimers[i].u64Config),
1445 szTmp);
1446 }
1447}
1448
1449
1450/* -=-=-=-=-=- Saved State -=-=-=-=-=- */
1451
1452
1453/**
1454 * @callback_method_impl{FNSSMDEVLIVEEXEC}
1455 */
1456static DECLCALLBACK(int) hpetR3LiveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uPass)
1457{
1458 PHPET pThis = PDMDEVINS_2_DATA(pDevIns, PHPET);
1459 PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3;
1460 NOREF(uPass);
1461
1462 pHlp->pfnSSMPutU8(pSSM, HPET_CAP_GET_TIMERS(pThis->u32Capabilities));
1463
1464 return VINF_SSM_DONT_CALL_AGAIN;
1465}
1466
1467
1468/**
1469 * @callback_method_impl{FNSSMDEVSAVEEXEC}
1470 */
1471static DECLCALLBACK(int) hpetR3SaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
1472{
1473 PHPET pThis = PDMDEVINS_2_DATA(pDevIns, PHPET);
1474 PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3;
1475
1476 /*
1477 * The config.
1478 */
1479 hpetR3LiveExec(pDevIns, pSSM, SSM_PASS_FINAL);
1480
1481 /*
1482 * The state.
1483 */
1484 uint32_t const cTimers = HPET_CAP_GET_TIMERS(pThis->u32Capabilities);
1485 AssertReturn(cTimers <= RT_ELEMENTS(pThis->aTimers), VERR_OUT_OF_RANGE);
1486 for (uint32_t iTimer = 0; iTimer < cTimers; iTimer++)
1487 {
1488 PHPETTIMER pHpetTimer = &pThis->aTimers[iTimer];
1489 PDMDevHlpTimerSave(pDevIns, pHpetTimer->hTimer, pSSM);
1490 pHlp->pfnSSMPutU8(pSSM, pHpetTimer->u8Wrap);
1491 pHlp->pfnSSMPutU64(pSSM, pHpetTimer->u64Config);
1492 pHlp->pfnSSMPutU64(pSSM, pHpetTimer->u64Cmp);
1493 pHlp->pfnSSMPutU64(pSSM, pHpetTimer->u64Fsb);
1494 pHlp->pfnSSMPutU64(pSSM, pHpetTimer->u64Period);
1495 }
1496
1497 pHlp->pfnSSMPutU64(pSSM, pThis->u64HpetOffset);
1498 uint64_t u64CapPer = RT_MAKE_U64(pThis->u32Capabilities, pThis->fIch9 ? HPET_CLK_PERIOD_ICH9 : HPET_CLK_PERIOD_PIIX);
1499 pHlp->pfnSSMPutU64(pSSM, u64CapPer);
1500 pHlp->pfnSSMPutU64(pSSM, pThis->u64HpetConfig);
1501 pHlp->pfnSSMPutU64(pSSM, pThis->u64Isr);
1502 return pHlp->pfnSSMPutU64(pSSM, pThis->u64HpetCounter);
1503}
1504
1505
1506/**
1507 * @callback_method_impl{FNSSMDEVLOADEXEC}
1508 */
1509static DECLCALLBACK(int) hpetR3LoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
1510{
1511 PHPET pThis = PDMDEVINS_2_DATA(pDevIns, PHPET);
1512 PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3;
1513
1514 /*
1515 * Version checks.
1516 */
1517 if (uVersion == HPET_SAVED_STATE_VERSION_EMPTY)
1518 return VINF_SUCCESS;
1519 if ( uVersion != HPET_SAVED_STATE_VERSION
1520 && uVersion != HPET_SAVED_STATE_VERSION_PRE_TIMER)
1521 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
1522
1523 /*
1524 * The config.
1525 */
1526 uint8_t cTimers;
1527 int rc = pHlp->pfnSSMGetU8(pSSM, &cTimers);
1528 AssertRCReturn(rc, rc);
1529 if (cTimers > RT_ELEMENTS(pThis->aTimers))
1530 return pHlp->pfnSSMSetCfgError(pSSM, RT_SRC_POS, N_("Config mismatch - too many timers: saved=%#x config=%#x"),
1531 cTimers, RT_ELEMENTS(pThis->aTimers));
1532
1533 if (uPass != SSM_PASS_FINAL)
1534 return VINF_SUCCESS;
1535
1536 /*
1537 * The state.
1538 */
1539 for (uint32_t iTimer = 0; iTimer < cTimers; iTimer++)
1540 {
1541 PHPETTIMER pHpetTimer = &pThis->aTimers[iTimer];
1542 PDMDevHlpTimerLoad(pDevIns, pHpetTimer->hTimer, pSSM);
1543 pHlp->pfnSSMGetU8(pSSM, &pHpetTimer->u8Wrap);
1544 pHlp->pfnSSMGetU64(pSSM, &pHpetTimer->u64Config);
1545 pHlp->pfnSSMGetU64(pSSM, &pHpetTimer->u64Cmp);
1546 pHlp->pfnSSMGetU64(pSSM, &pHpetTimer->u64Fsb);
1547 pHlp->pfnSSMGetU64(pSSM, &pHpetTimer->u64Period);
1548 }
1549
1550 pHlp->pfnSSMGetU64(pSSM, &pThis->u64HpetOffset);
1551 uint64_t u64CapPer;
1552 pHlp->pfnSSMGetU64(pSSM, &u64CapPer);
1553 pHlp->pfnSSMGetU64(pSSM, &pThis->u64HpetConfig);
1554 pHlp->pfnSSMGetU64(pSSM, &pThis->u64Isr);
1555 rc = pHlp->pfnSSMGetU64(pSSM, &pThis->u64HpetCounter);
1556 if (RT_FAILURE(rc))
1557 return rc;
1558
1559 /* Older saved state have an off-by-1 timer count bug. */
1560 uint8_t cCapTimers = HPET_CAP_GET_TIMERS(RT_LO_U32(u64CapPer));
1561 if ( uVersion <= HPET_SAVED_STATE_VERSION_PRE_TIMER
1562 && cCapTimers > 0 /* Paranoia */)
1563 --cCapTimers;
1564
1565 /* Verify capability reported timer count matches timer count in the saved state field. */
1566 if (cCapTimers != cTimers)
1567 return pHlp->pfnSSMSetCfgError(pSSM, RT_SRC_POS, N_("Capabilities does not match timer count: cTimers=%#x caps=%#x"),
1568 cTimers, cCapTimers);
1569 if (HPET_CAP_GET_TIMERS(RT_LO_U32(u64CapPer)) > RT_ELEMENTS(pThis->aTimers))
1570 return pHlp->pfnSSMSetCfgError(pSSM, RT_SRC_POS, N_("Config mismatch - too many timers in capability register: CAP=%#x => %u times, max %u"),
1571 RT_LO_U32(u64CapPer), (unsigned)HPET_CAP_GET_TIMERS(RT_LO_U32(u64CapPer)), RT_ELEMENTS(pThis->aTimers));
1572
1573 pThis->u32Capabilities = RT_LO_U32(u64CapPer);
1574 uint32_t const uExpectedPeriod = pThis->fIch9 ? HPET_CLK_PERIOD_ICH9 : HPET_CLK_PERIOD_PIIX;
1575 if (RT_HI_U32(u64CapPer) != uExpectedPeriod)
1576 return pHlp->pfnSSMSetCfgError(pSSM, RT_SRC_POS, N_("Config mismatch - Expected period %RU32 fs, loaded %RU32 fs"),
1577 uExpectedPeriod, RT_HI_U32(u64CapPer));
1578
1579 /*
1580 * Set the timer frequency hints.
1581 */
1582 PDMDevHlpCritSectEnter(pDevIns, &pThis->CritSect, VERR_IGNORED);
1583 for (uint32_t iTimer = 0; iTimer < cTimers; iTimer++)
1584 {
1585 PHPETTIMER pHpetTimer = &pThis->aTimers[iTimer];
1586 if (PDMDevHlpTimerIsActive(pDevIns, pHpetTimer->hTimer))
1587 hpetTimerSetFrequencyHint(pDevIns, pThis, pHpetTimer, pHpetTimer->u64Config, pHpetTimer->u64Period);
1588 }
1589 PDMDevHlpCritSectLeave(pDevIns, &pThis->CritSect);
1590 return VINF_SUCCESS;
1591}
1592
1593
1594/* -=-=-=-=-=- PDMDEVREG -=-=-=-=-=- */
1595
1596
1597/**
1598 * @interface_method_impl{PDMDEVREG,pfnRelocate}
1599 */
1600static DECLCALLBACK(void) hpetR3Relocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
1601{
1602 PHPETRC pThisRC = PDMINS_2_DATA_RC(pDevIns, PHPETRC);
1603 LogFlow(("hpetR3Relocate:\n"));
1604
1605 pThisRC->pHpetHlp += offDelta;
1606}
1607
1608
1609/**
1610 * @interface_method_impl{PDMDEVREG,pfnReset}
1611 */
1612static DECLCALLBACK(void) hpetR3Reset(PPDMDEVINS pDevIns)
1613{
1614 PHPET pThis = PDMDEVINS_2_DATA(pDevIns, PHPET);
1615 PHPETCC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PHPETCC);
1616 LogFlow(("hpetR3Reset:\n"));
1617
1618 /*
1619 * The timers first.
1620 */
1621 PDMDevHlpTimerLockClock(pDevIns, pThis->aTimers[0].hTimer, VERR_IGNORED);
1622 for (unsigned i = 0; i < RT_ELEMENTS(pThis->aTimers); i++)
1623 {
1624 PHPETTIMER pHpetTimer = &pThis->aTimers[i];
1625 Assert(pHpetTimer->idxTimer == i);
1626 PDMDevHlpTimerStop(pDevIns, pHpetTimer->hTimer);
1627
1628 /* capable of periodic operations and 64-bits */
1629 uint64_t fConfig;
1630 if (pThis->fIch9)
1631 fConfig = i == 0 ? HPET_TN_PERIODIC_CAP | HPET_TN_SIZE_CAP : 0;
1632 else
1633 fConfig = HPET_TN_PERIODIC_CAP | HPET_TN_SIZE_CAP;
1634
1635 /* We can do all IRQs */
1636 uint32_t u32RoutingCap = 0xffffffff;
1637 fConfig |= ((uint64_t)u32RoutingCap) << HPET_TN_INT_ROUTE_CAP_SHIFT;
1638 ASMAtomicWriteU64(&pHpetTimer->u64Config, fConfig);
1639 pHpetTimer->u64Period = 0;
1640 pHpetTimer->u8Wrap = 0;
1641 pHpetTimer->u64Cmp = hpetInvalidValue(pHpetTimer);
1642 }
1643 PDMDevHlpTimerUnlockClock(pDevIns, pThis->aTimers[0].hTimer);
1644
1645 /*
1646 * The shared HPET state.
1647 */
1648 pThis->u64HpetConfig = 0;
1649 pThis->u64HpetCounter = 0;
1650 pThis->u64HpetOffset = 0;
1651
1652 /* 64-bit main counter; 3 timers supported; LegacyReplacementRoute. */
1653 pThis->u32Capabilities = (1 << 15) /* LEG_RT_CAP - LegacyReplacementRoute capable. */
1654 | (1 << 13) /* COUNTER_SIZE_CAP - Main counter is 64-bit capable. */
1655 | 1; /* REV_ID - Revision, must not be 0 */
1656 if (pThis->fIch9) /* NUM_TIM_CAP - Number of timers -1. */
1657 pThis->u32Capabilities |= (HPET_NUM_TIMERS_ICH9 - 1) << 8;
1658 else
1659 pThis->u32Capabilities |= (HPET_NUM_TIMERS_PIIX - 1) << 8;
1660 pThis->u32Capabilities |= UINT32_C(0x80860000); /* VENDOR */
1661 AssertCompile(HPET_NUM_TIMERS_ICH9 <= RT_ELEMENTS(pThis->aTimers));
1662 AssertCompile(HPET_NUM_TIMERS_PIIX <= RT_ELEMENTS(pThis->aTimers));
1663
1664
1665 /*
1666 * Notify the PIT/RTC devices.
1667 */
1668 if (pThisCC->pHpetHlp)
1669 pThisCC->pHpetHlp->pfnSetLegacyMode(pDevIns, false /*fActive*/);
1670}
1671
1672
1673/**
1674 * @interface_method_impl{PDMDEVREG,pfnConstruct}
1675 */
1676static DECLCALLBACK(int) hpetR3Construct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
1677{
1678 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
1679 PHPET pThis = PDMDEVINS_2_DATA(pDevIns, PHPET);
1680 PHPETCC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PHPETCC);
1681 PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3;
1682
1683 /* Only one HPET device now, as we use fixed MMIO region. */
1684 Assert(iInstance == 0); RT_NOREF(iInstance);
1685
1686 /*
1687 * Initialize the device state.
1688 */
1689
1690 /* Init the HPET timers (init all regardless of how many we expose). */
1691 for (unsigned i = 0; i < RT_ELEMENTS(pThis->aTimers); i++)
1692 {
1693 PHPETTIMER pHpetTimer = &pThis->aTimers[i];
1694 pHpetTimer->idxTimer = i;
1695 pHpetTimer->hTimer = NIL_TMTIMERHANDLE;
1696 }
1697
1698 /*
1699 * Validate and read the configuration.
1700 */
1701 PDMDEV_VALIDATE_CONFIG_RETURN(pDevIns, "ICH9", "");
1702
1703 int rc = pHlp->pfnCFGMQueryBoolDef(pCfg, "ICH9", &pThis->fIch9, false);
1704 if (RT_FAILURE(rc))
1705 return PDMDEV_SET_ERROR(pDevIns, rc, N_("Configuration error: failed to read ICH9 as boolean"));
1706
1707
1708 /*
1709 * Create critsect and timers.
1710 * Note! We don't use the default critical section of the device, but our own.
1711 */
1712 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->CritSect, RT_SRC_POS, "HPET");
1713 AssertRCReturn(rc, rc);
1714
1715 rc = PDMDevHlpSetDeviceCritSect(pDevIns, PDMDevHlpCritSectGetNop(pDevIns));
1716 AssertRCReturn(rc, rc);
1717
1718 /* Init the HPET timers (init all regardless of how many we expose). */
1719 static const char * const s_apszTimerNames[] =
1720 { "HPET Timer 0", "HPET Timer 1", "HPET Timer 2", "HPET Timer 3" };
1721 AssertCompile(RT_ELEMENTS(pThis->aTimers) == RT_ELEMENTS(s_apszTimerNames));
1722 for (unsigned i = 0; i < RT_ELEMENTS(pThis->aTimers); i++)
1723 {
1724 PHPETTIMER pHpetTimer = &pThis->aTimers[i];
1725 rc = PDMDevHlpTimerCreate(pDevIns, TMCLOCK_VIRTUAL_SYNC, hpetR3Timer, pHpetTimer,
1726 TMTIMER_FLAGS_NO_CRIT_SECT | TMTIMER_FLAGS_RING0,
1727 s_apszTimerNames[i], &pThis->aTimers[i].hTimer);
1728 AssertRCReturn(rc, rc);
1729 uint64_t const cTicksPerSec = PDMDevHlpTimerGetFreq(pDevIns, pThis->aTimers[i].hTimer);
1730 if (cTicksPerSec != RT_NS_1SEC)
1731 return PDMDevHlpVMSetError(pDevIns, VERR_INTERNAL_ERROR_2, RT_SRC_POS,
1732 "Unexpected timer resolution %RU64, code assumes nanonsecond resolution!", cTicksPerSec);
1733 }
1734
1735 /*
1736 * This must be done prior to registering the HPET, right?
1737 */
1738 hpetR3Reset(pDevIns);
1739
1740 uint32_t const fCaps = pThis->u32Capabilities;
1741 LogRel(("HPET: Capabilities=%#RX32 (LegacyRt=%RTbool CounterSize=%s Timers=%u Revision=%#x)\n",
1742 fCaps, HPET_CAP_HAS_LEG_RT(fCaps), HPET_CAP_HAS_64BIT_COUNT_SIZE(fCaps) ? "64-bit" : "32-bit",
1743 HPET_CAP_GET_TIMERS(fCaps), HPET_CAP_GET_REV_ID(fCaps)));
1744
1745 /*
1746 * Register the HPET and get helpers.
1747 */
1748 PDMHPETREG HpetReg;
1749 HpetReg.u32Version = PDM_HPETREG_VERSION;
1750 rc = PDMDevHlpHpetRegister(pDevIns, &HpetReg, &pThisCC->pHpetHlp);
1751 AssertRCReturn(rc, rc);
1752
1753 /*
1754 * Register the MMIO range, PDM API requests page aligned
1755 * addresses and sizes.
1756 */
1757 rc = PDMDevHlpMmioCreateAndMap(pDevIns, HPET_BASE, HPET_BAR_SIZE, hpetMMIOWrite, hpetMMIORead,
1758 IOMMMIO_FLAGS_READ_DWORD_QWORD | IOMMMIO_FLAGS_WRITE_ONLY_DWORD_QWORD,
1759 "HPET Memory", &pThis->hMmio);
1760 AssertRCReturn(rc, rc);
1761
1762 /*
1763 * Register SSM state, info item and statistics.
1764 */
1765 rc = PDMDevHlpSSMRegister3(pDevIns, HPET_SAVED_STATE_VERSION, sizeof(*pThis), hpetR3LiveExec, hpetR3SaveExec, hpetR3LoadExec);
1766 AssertRCReturn(rc, rc);
1767
1768 PDMDevHlpDBGFInfoRegister(pDevIns, "hpet", "Display HPET status. (no arguments)", hpetR3Info);
1769
1770 /* Statistics: */
1771 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatCounterRead4Byte, STAMTYPE_COUNTER,
1772 "ReadCounter32bit", STAMUNIT_OCCURENCES, "HPET_COUNTER 32-bit reads");
1773 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatCounterRead8Byte, STAMTYPE_COUNTER,
1774 "ReadCounter64bit", STAMUNIT_OCCURENCES, "HPET_COUNTER 64-bit reads");
1775 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatCounterWriteLow, STAMTYPE_COUNTER,
1776 "WriteCounterLow", STAMUNIT_OCCURENCES, "Low HPET_COUNTER writes");
1777 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatCounterWriteHigh, STAMTYPE_COUNTER,
1778 "WriteCounterHigh", STAMUNIT_OCCURENCES, "High HPET_COUNTER writes");
1779 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatZeroDeltaHack, STAMTYPE_COUNTER,
1780 "ZeroDeltaHacks", STAMUNIT_OCCURENCES, "High HPET_COUNTER writes");
1781
1782 for (unsigned i = 0; i < RT_ELEMENTS(pThis->aTimers); i++)
1783 {
1784 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->aTimers[i].StatSetIrq, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS,
1785 STAMUNIT_OCCURENCES, "Number of times the IRQ has been set.", "timer%u/SetIrq", i);
1786 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->aTimers[i].StatSetTimer, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS,
1787 STAMUNIT_OCCURENCES, "Number of times the timer has be programmed.", "timer%u/SetTimer", i);
1788 }
1789
1790 return VINF_SUCCESS;
1791}
1792
1793#else /* !IN_RING3 */
1794
1795/**
1796 * @callback_method_impl{PDMDEVREGR0,pfnConstruct}
1797 */
1798static DECLCALLBACK(int) hpetRZConstruct(PPDMDEVINS pDevIns)
1799{
1800 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
1801 PHPET pThis = PDMDEVINS_2_DATA(pDevIns, PHPET);
1802 PHPETCC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PHPETCC);
1803
1804 int rc = PDMDevHlpSetDeviceCritSect(pDevIns, PDMDevHlpCritSectGetNop(pDevIns));
1805 AssertRCReturn(rc, rc);
1806
1807 PDMHPETREG HpetReg;
1808 HpetReg.u32Version = PDM_HPETREG_VERSION;
1809 rc = PDMDevHlpHpetSetUpContext(pDevIns, &HpetReg, &pThisCC->pHpetHlp);
1810 AssertRCReturn(rc, rc);
1811
1812 rc = PDMDevHlpMmioSetUpContext(pDevIns, pThis->hMmio, hpetMMIOWrite, hpetMMIORead, NULL /*pvUser*/);
1813 AssertRCReturn(rc, rc);
1814
1815 return VINF_SUCCESS;
1816}
1817
1818#endif /* !IN_RING3 */
1819
1820/**
1821 * The device registration structure.
1822 */
1823const PDMDEVREG g_DeviceHPET =
1824{
1825 /* .u32Version = */ PDM_DEVREG_VERSION,
1826 /* .uReserved0 = */ 0,
1827 /* .szName = */ "hpet",
1828 /* .fFlags = */ PDM_DEVREG_FLAGS_DEFAULT_BITS | PDM_DEVREG_FLAGS_RZ | PDM_DEVREG_FLAGS_NEW_STYLE,
1829 /* .fClass = */ PDM_DEVREG_CLASS_PIT,
1830 /* .cMaxInstances = */ 1,
1831 /* .uSharedVersion = */ 42,
1832 /* .cbInstanceShared = */ sizeof(HPET),
1833 /* .cbInstanceCC = */ sizeof(HPETCC),
1834 /* .cbInstanceRC = */ sizeof(HPETRC),
1835 /* .cMaxPciDevices = */ 0,
1836 /* .cMaxMsixVectors = */ 0,
1837 /* .pszDescription = */ "High Precision Event Timer (HPET) Device",
1838#if defined(IN_RING3)
1839 /* .pszRCMod = */ "VBoxDDRC.rc",
1840 /* .pszR0Mod = */ "VBoxDDR0.r0",
1841 /* .pfnConstruct = */ hpetR3Construct,
1842 /* .pfnDestruct = */ NULL,
1843 /* .pfnRelocate = */ hpetR3Relocate,
1844 /* .pfnMemSetup = */ NULL,
1845 /* .pfnPowerOn = */ NULL,
1846 /* .pfnReset = */ hpetR3Reset,
1847 /* .pfnSuspend = */ NULL,
1848 /* .pfnResume = */ NULL,
1849 /* .pfnAttach = */ NULL,
1850 /* .pfnDetach = */ NULL,
1851 /* .pfnQueryInterface = */ NULL,
1852 /* .pfnInitComplete = */ NULL,
1853 /* .pfnPowerOff = */ NULL,
1854 /* .pfnSoftReset = */ NULL,
1855 /* .pfnReserved0 = */ NULL,
1856 /* .pfnReserved1 = */ NULL,
1857 /* .pfnReserved2 = */ NULL,
1858 /* .pfnReserved3 = */ NULL,
1859 /* .pfnReserved4 = */ NULL,
1860 /* .pfnReserved5 = */ NULL,
1861 /* .pfnReserved6 = */ NULL,
1862 /* .pfnReserved7 = */ NULL,
1863#elif defined(IN_RING0)
1864 /* .pfnEarlyConstruct = */ NULL,
1865 /* .pfnConstruct = */ hpetRZConstruct,
1866 /* .pfnDestruct = */ NULL,
1867 /* .pfnFinalDestruct = */ NULL,
1868 /* .pfnRequest = */ NULL,
1869 /* .pfnReserved0 = */ NULL,
1870 /* .pfnReserved1 = */ NULL,
1871 /* .pfnReserved2 = */ NULL,
1872 /* .pfnReserved3 = */ NULL,
1873 /* .pfnReserved4 = */ NULL,
1874 /* .pfnReserved5 = */ NULL,
1875 /* .pfnReserved6 = */ NULL,
1876 /* .pfnReserved7 = */ NULL,
1877#elif defined(IN_RC)
1878 /* .pfnConstruct = */ hpetRZConstruct,
1879 /* .pfnReserved0 = */ NULL,
1880 /* .pfnReserved1 = */ NULL,
1881 /* .pfnReserved2 = */ NULL,
1882 /* .pfnReserved3 = */ NULL,
1883 /* .pfnReserved4 = */ NULL,
1884 /* .pfnReserved5 = */ NULL,
1885 /* .pfnReserved6 = */ NULL,
1886 /* .pfnReserved7 = */ NULL,
1887#else
1888# error "Not in IN_RING3, IN_RING0 or IN_RC!"
1889#endif
1890 /* .u32VersionEnd = */ PDM_DEVREG_VERSION
1891};
1892
1893#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */
1894
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette