2008-05-21 03:40:12 +02:00
|
|
|
/** @file
|
2008-04-09 09:07:50 +02:00
|
|
|
UEFI Miscellaneous boot Services Stall service implementation
|
2007-07-04 12:51:54 +02:00
|
|
|
|
2011-12-06 06:24:27 +01:00
|
|
|
Copyright (c) 2006 - 2011, Intel Corporation. All rights reserved.<BR>
|
2010-04-24 11:49:11 +02:00
|
|
|
This program and the accompanying materials
|
2007-07-04 12:51:54 +02:00
|
|
|
are licensed and made available under the terms and conditions of the BSD License
|
|
|
|
which accompanies this distribution. The full text of the license may be found at
|
|
|
|
http://opensource.org/licenses/bsd-license.php
|
|
|
|
|
|
|
|
THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
|
|
|
|
|
2008-04-09 09:07:50 +02:00
|
|
|
**/
|
2007-07-04 12:51:54 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Include statements
|
|
|
|
//
|
|
|
|
|
2008-08-27 16:29:23 +02:00
|
|
|
#include "DxeMain.h"
|
2007-07-04 12:51:54 +02:00
|
|
|
|
2011-12-06 06:24:27 +01:00
|
|
|
/**
|
|
|
|
Internal worker function to call the Metronome Architectural Protocol for
|
|
|
|
the number of ticks specified by the UINT64 Counter value. WaitForTick()
|
|
|
|
service of the Metronome Architectural Protocol uses a UINT32 for the number
|
|
|
|
of ticks to wait, so this function loops when Counter is larger than 0xffffffff.
|
2007-07-04 12:51:54 +02:00
|
|
|
|
2011-12-06 06:24:27 +01:00
|
|
|
@param Counter Number of ticks to wait.
|
|
|
|
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
CoreInternalWaitForTick (
|
|
|
|
IN UINT64 Counter
|
|
|
|
)
|
|
|
|
{
|
2014-09-09 04:35:41 +02:00
|
|
|
while (RShiftU64 (Counter, 32) > 0) {
|
2011-12-06 06:24:27 +01:00
|
|
|
gMetronome->WaitForTick (gMetronome, 0xffffffff);
|
|
|
|
Counter -= 0xffffffff;
|
|
|
|
}
|
|
|
|
gMetronome->WaitForTick (gMetronome, (UINT32)Counter);
|
|
|
|
}
|
2007-07-04 12:51:54 +02:00
|
|
|
|
2008-05-09 09:08:30 +02:00
|
|
|
/**
|
2007-07-04 12:51:54 +02:00
|
|
|
Introduces a fine-grained stall.
|
|
|
|
|
2008-05-09 09:08:30 +02:00
|
|
|
@param Microseconds The number of microseconds to stall execution.
|
2007-07-04 12:51:54 +02:00
|
|
|
|
2008-07-24 04:54:45 +02:00
|
|
|
@retval EFI_SUCCESS Execution was stalled for at least the requested
|
|
|
|
amount of microseconds.
|
2008-05-09 09:08:30 +02:00
|
|
|
@retval EFI_NOT_AVAILABLE_YET gMetronome is not available yet
|
2007-07-04 12:51:54 +02:00
|
|
|
|
2008-05-09 09:08:30 +02:00
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
CoreStall (
|
|
|
|
IN UINTN Microseconds
|
|
|
|
)
|
2007-07-04 12:51:54 +02:00
|
|
|
{
|
2011-12-06 06:24:27 +01:00
|
|
|
UINT64 Counter;
|
2007-07-04 12:51:54 +02:00
|
|
|
UINT32 Remainder;
|
2011-12-06 06:24:27 +01:00
|
|
|
UINTN Index;
|
2007-07-04 12:51:54 +02:00
|
|
|
|
|
|
|
if (gMetronome == NULL) {
|
|
|
|
return EFI_NOT_AVAILABLE_YET;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
2011-12-06 06:24:27 +01:00
|
|
|
// Counter = Microseconds * 10 / gMetronome->TickPeriod
|
|
|
|
// 0x1999999999999999 = (2^64 - 1) / 10
|
2007-07-04 12:51:54 +02:00
|
|
|
//
|
2011-12-06 06:24:27 +01:00
|
|
|
if (Microseconds > 0x1999999999999999ULL) {
|
|
|
|
//
|
|
|
|
// Microseconds is too large to multiple by 10 first. Perform the divide
|
|
|
|
// operation first and loop 10 times to avoid 64-bit math overflow.
|
|
|
|
//
|
|
|
|
Counter = DivU64x32Remainder (
|
|
|
|
Microseconds,
|
|
|
|
gMetronome->TickPeriod,
|
|
|
|
&Remainder
|
|
|
|
);
|
|
|
|
for (Index = 0; Index < 10; Index++) {
|
|
|
|
CoreInternalWaitForTick (Counter);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Remainder != 0) {
|
|
|
|
//
|
|
|
|
// If Remainder was not zero, then normally, Counter would be rounded
|
|
|
|
// up by 1 tick. In this case, since a loop for 10 counts was used
|
|
|
|
// to emulate the multiply by 10 operation, Counter needs to be rounded
|
|
|
|
// up by 10 counts.
|
|
|
|
//
|
|
|
|
CoreInternalWaitForTick (10);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
//
|
|
|
|
// Calculate the number of ticks by dividing the number of microseconds by
|
|
|
|
// the TickPeriod. Calculation is based on 100ns unit.
|
|
|
|
//
|
|
|
|
Counter = DivU64x32Remainder (
|
|
|
|
MultU64x32 (Microseconds, 10),
|
|
|
|
gMetronome->TickPeriod,
|
|
|
|
&Remainder
|
|
|
|
);
|
|
|
|
if (Remainder != 0) {
|
|
|
|
//
|
|
|
|
// If Remainder is not zero, then round Counter up by one tick.
|
|
|
|
//
|
|
|
|
Counter++;
|
|
|
|
}
|
|
|
|
CoreInternalWaitForTick (Counter);
|
2007-07-04 12:51:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|