VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMR0/TMR0.cpp@ 93632

Last change on this file since 93632 was 93554, checked in by vboxsync, 3 years ago

VMM: Changed PAGE_SIZE -> GUEST_PAGE_SIZE / HOST_PAGE_SIZE, PAGE_SHIFT -> GUEST_PAGE_SHIFT / HOST_PAGE_SHIFT, and PAGE_OFFSET_MASK -> GUEST_PAGE_OFFSET_MASK / HOST_PAGE_OFFSET_MASK. Also removed most usage of ASMMemIsZeroPage and ASMMemZeroPage since the host and guest page size doesn't need to be the same any more. Some work left to do in the page pool code. bugref:9898

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 5.8 KB
Line 
1/* $Id: TMR0.cpp 93554 2022-02-02 22:57:02Z vboxsync $ */
2/** @file
3 * TM - Timeout Manager, host ring-0 context.
4 */
5
6/*
7 * Copyright (C) 2021-2022 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_TM
23#include <VBox/vmm/tm.h>
24#include "TMInternal.h"
25#include <VBox/vmm/gvm.h>
26
27#include <VBox/err.h>
28#include <VBox/log.h>
29#include <VBox/param.h>
30#include <iprt/assert.h>
31#include <iprt/mem.h>
32#include <iprt/memobj.h>
33#include <iprt/process.h>
34#include <iprt/string.h>
35
36
37
38/**
39 * Initializes the per-VM data for the TM.
40 *
41 * This is called from under the GVMM lock, so it should only initialize the
42 * data so TMR0CleanupVM and others will work smoothly.
43 *
44 * @param pGVM Pointer to the global VM structure.
45 */
46VMMR0_INT_DECL(void) TMR0InitPerVMData(PGVM pGVM)
47{
48 for (uint32_t idxQueue = 0; idxQueue < RT_ELEMENTS(pGVM->tmr0.s.aTimerQueues); idxQueue++)
49 {
50 pGVM->tmr0.s.aTimerQueues[idxQueue].hMemObj = NIL_RTR0MEMOBJ;
51 pGVM->tmr0.s.aTimerQueues[idxQueue].hMapObj = NIL_RTR0MEMOBJ;
52 }
53}
54
55
56/**
57 * Cleans up any loose ends before the GVM structure is destroyed.
58 */
59VMMR0_INT_DECL(void) TMR0CleanupVM(PGVM pGVM)
60{
61 for (uint32_t idxQueue = 0; idxQueue < RT_ELEMENTS(pGVM->tmr0.s.aTimerQueues); idxQueue++)
62 {
63 if (pGVM->tmr0.s.aTimerQueues[idxQueue].hMapObj == NIL_RTR0MEMOBJ)
64 {
65 RTR0MemObjFree(pGVM->tmr0.s.aTimerQueues[idxQueue].hMapObj, true /*fFreeMappings*/);
66 pGVM->tmr0.s.aTimerQueues[idxQueue].hMapObj = NIL_RTR0MEMOBJ;
67 }
68
69 if (pGVM->tmr0.s.aTimerQueues[idxQueue].hMemObj != NIL_RTR0MEMOBJ)
70 {
71 RTR0MemObjFree(pGVM->tmr0.s.aTimerQueues[idxQueue].hMemObj, true /*fFreeMappings*/);
72 pGVM->tmr0.s.aTimerQueues[idxQueue].hMemObj = NIL_RTR0MEMOBJ;
73 }
74 }
75}
76
77
78/**
79 * Grows the timer array for @a idxQueue to at least @a cMinTimers entries.
80 *
81 * @returns VBox status code.
82 * @param pGVM The ring-0 VM structure.
83 * @param idxQueue The index of the queue to grow.
84 * @param cMinTimers The minimum growth target.
85 * @thread EMT
86 * @note Caller must own the queue lock exclusively.
87 */
88VMMR0_INT_DECL(int) TMR0TimerQueueGrow(PGVM pGVM, uint32_t idxQueue, uint32_t cMinTimers)
89{
90 /*
91 * Validate input and state.
92 */
93 VM_ASSERT_EMT0_RETURN(pGVM, VERR_VM_THREAD_NOT_EMT);
94 VM_ASSERT_STATE_RETURN(pGVM, VMSTATE_CREATING, VERR_VM_INVALID_VM_STATE); /** @todo must do better than this! */
95 AssertReturn(idxQueue <= RT_ELEMENTS(pGVM->tmr0.s.aTimerQueues), VERR_TM_INVALID_TIMER_QUEUE);
96 AssertCompile(RT_ELEMENTS(pGVM->tmr0.s.aTimerQueues) == RT_ELEMENTS(pGVM->tm.s.aTimerQueues));
97 PTMTIMERQUEUER0 pQueueR0 = &pGVM->tmr0.s.aTimerQueues[idxQueue];
98 PTMTIMERQUEUE pQueueShared = &pGVM->tm.s.aTimerQueues[idxQueue];
99 AssertMsgReturn(PDMCritSectRwIsWriteOwner(pGVM, &pQueueShared->AllocLock),
100 ("queue=%s %.*Rhxs\n", pQueueShared->szName, sizeof(pQueueShared->AllocLock), &pQueueShared->AllocLock),
101 VERR_NOT_OWNER);
102
103 uint32_t cNewTimers = cMinTimers;
104 AssertReturn(cNewTimers <= _32K, VERR_TM_TOO_MANY_TIMERS);
105 uint32_t const cOldTimers = pQueueR0->cTimersAlloc;
106 ASMCompilerBarrier();
107 AssertReturn(cNewTimers >= cOldTimers, VERR_TM_IPE_1);
108 AssertReturn(cOldTimers == pQueueShared->cTimersAlloc, VERR_TM_IPE_2);
109
110 /*
111 * Round up the request to the nearest page and do the allocation.
112 */
113 size_t cbNew = sizeof(TMTIMER) * cNewTimers;
114 cbNew = RT_ALIGN_Z(cbNew, HOST_PAGE_SIZE);
115 cNewTimers = (uint32_t)(cbNew / sizeof(TMTIMER));
116
117 RTR0MEMOBJ hMemObj;
118 int rc = RTR0MemObjAllocPage(&hMemObj, cbNew, false /*fExecutable*/);
119 if (RT_SUCCESS(rc))
120 {
121 /*
122 * Zero and map it.
123 */
124 PTMTIMER paTimers = (PTMTIMER)RTR0MemObjAddress(hMemObj);
125 RT_BZERO(paTimers, cbNew);
126
127 RTR0MEMOBJ hMapObj;
128 rc = RTR0MemObjMapUser(&hMapObj, hMemObj, (RTR3PTR)-1, HOST_PAGE_SIZE,
129 RTMEM_PROT_READ | RTMEM_PROT_WRITE, RTR0ProcHandleSelf());
130 if (RT_SUCCESS(rc))
131 {
132 tmHCTimerQueueGrowInit(paTimers, pQueueR0->paTimers, cNewTimers, cOldTimers);
133
134 /*
135 * Switch the memory handles.
136 */
137 RTR0MEMOBJ hTmp = pQueueR0->hMapObj;
138 pQueueR0->hMapObj = hMapObj;
139 hMapObj = hTmp;
140
141 hTmp = pQueueR0->hMemObj;
142 pQueueR0->hMemObj = hMemObj;
143 hMemObj = hTmp;
144
145 /*
146 * Update the variables.
147 */
148 pQueueR0->paTimers = paTimers;
149 pQueueR0->cTimersAlloc = cNewTimers;
150 pQueueShared->paTimers = RTR0MemObjAddressR3(pQueueR0->hMapObj);
151 pQueueShared->cTimersAlloc = cNewTimers;
152 pQueueShared->cTimersFree += cNewTimers - (cOldTimers ? cOldTimers : 1);
153
154 /*
155 * Free the old allocation.
156 */
157 RTR0MemObjFree(hMapObj, true /*fFreeMappings*/);
158 }
159 RTR0MemObjFree(hMemObj, true /*fFreeMappings*/);
160 }
161
162 return rc;
163}
164
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