2007-06-22 05:21:45 +02:00
|
|
|
/** @file
|
|
|
|
Leaf math worker functions that require 64-bit arithmetic support from the
|
|
|
|
compiler.
|
|
|
|
|
2010-06-24 02:20:35 +02:00
|
|
|
Copyright (c) 2006 - 2010, Intel Corporation. All rights reserved.<BR>
|
2019-04-04 01:06:00 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2007-06-22 05:21:45 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include "BaseLibInternals.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
Shifts a 64-bit integer left between 0 and 63 bits. The low bits
|
|
|
|
are filled with zeros. The shifted value is returned.
|
|
|
|
|
|
|
|
This function shifts the 64-bit value Operand to the left by Count bits. The
|
|
|
|
low Count bits are set to zero. The shifted value is returned.
|
|
|
|
|
|
|
|
@param Operand The 64-bit operand to shift left.
|
|
|
|
@param Count The number of bits to shift left.
|
|
|
|
|
2010-06-24 02:20:35 +02:00
|
|
|
@return Operand << Count.
|
2007-06-22 05:21:45 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
UINT64
|
|
|
|
EFIAPI
|
|
|
|
InternalMathLShiftU64 (
|
2021-12-05 23:54:05 +01:00
|
|
|
IN UINT64 Operand,
|
|
|
|
IN UINTN Count
|
2007-06-22 05:21:45 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
return Operand << Count;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Shifts a 64-bit integer right between 0 and 63 bits. This high bits
|
|
|
|
are filled with zeros. The shifted value is returned.
|
|
|
|
|
|
|
|
This function shifts the 64-bit value Operand to the right by Count bits. The
|
|
|
|
high Count bits are set to zero. The shifted value is returned.
|
|
|
|
|
|
|
|
@param Operand The 64-bit operand to shift right.
|
|
|
|
@param Count The number of bits to shift right.
|
|
|
|
|
2010-06-24 02:20:35 +02:00
|
|
|
@return Operand >> Count.
|
2007-06-22 05:21:45 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
UINT64
|
|
|
|
EFIAPI
|
|
|
|
InternalMathRShiftU64 (
|
2021-12-05 23:54:05 +01:00
|
|
|
IN UINT64 Operand,
|
|
|
|
IN UINTN Count
|
2007-06-22 05:21:45 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
return Operand >> Count;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Shifts a 64-bit integer right between 0 and 63 bits. The high bits
|
|
|
|
are filled with original integer's bit 63. The shifted value is returned.
|
|
|
|
|
|
|
|
This function shifts the 64-bit value Operand to the right by Count bits. The
|
|
|
|
high Count bits are set to bit 63 of Operand. The shifted value is returned.
|
|
|
|
|
|
|
|
@param Operand The 64-bit operand to shift right.
|
|
|
|
@param Count The number of bits to shift right.
|
|
|
|
|
2010-06-24 02:20:35 +02:00
|
|
|
@return Operand arithmetically shifted right by Count.
|
2007-06-22 05:21:45 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
UINT64
|
|
|
|
EFIAPI
|
|
|
|
InternalMathARShiftU64 (
|
2021-12-05 23:54:05 +01:00
|
|
|
IN UINT64 Operand,
|
|
|
|
IN UINTN Count
|
2007-06-22 05:21:45 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
INTN TestValue;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Test if this compiler supports arithmetic shift
|
|
|
|
//
|
2015-12-31 11:47:22 +01:00
|
|
|
TestValue = (INTN)((INT64)(1ULL << 63) >> 63);
|
2007-06-22 05:21:45 +02:00
|
|
|
if (TestValue == -1) {
|
|
|
|
//
|
|
|
|
// Arithmetic shift is supported
|
|
|
|
//
|
|
|
|
return (UINT64)((INT64)Operand >> Count);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Arithmetic is not supported
|
|
|
|
//
|
|
|
|
return (Operand >> Count) |
|
|
|
|
((INTN)Operand < 0 ? ~((UINTN)-1 >> Count) : 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Rotates a 64-bit integer left between 0 and 63 bits, filling
|
|
|
|
the low bits with the high bits that were rotated.
|
|
|
|
|
|
|
|
This function rotates the 64-bit value Operand to the left by Count bits. The
|
|
|
|
low Count bits are fill with the high Count bits of Operand. The rotated
|
|
|
|
value is returned.
|
|
|
|
|
|
|
|
@param Operand The 64-bit operand to rotate left.
|
|
|
|
@param Count The number of bits to rotate left.
|
|
|
|
|
2010-06-24 02:20:35 +02:00
|
|
|
@return Operand <<< Count.
|
2007-06-22 05:21:45 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
UINT64
|
|
|
|
EFIAPI
|
|
|
|
InternalMathLRotU64 (
|
2021-12-05 23:54:05 +01:00
|
|
|
IN UINT64 Operand,
|
|
|
|
IN UINTN Count
|
2007-06-22 05:21:45 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
return (Operand << Count) | (Operand >> (64 - Count));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Rotates a 64-bit integer right between 0 and 63 bits, filling
|
|
|
|
the high bits with the high low bits that were rotated.
|
|
|
|
|
|
|
|
This function rotates the 64-bit value Operand to the right by Count bits.
|
|
|
|
The high Count bits are fill with the low Count bits of Operand. The rotated
|
|
|
|
value is returned.
|
|
|
|
|
|
|
|
@param Operand The 64-bit operand to rotate right.
|
|
|
|
@param Count The number of bits to rotate right.
|
|
|
|
|
2010-06-24 02:20:35 +02:00
|
|
|
@return Operand >>> Count.
|
2007-06-22 05:21:45 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
UINT64
|
|
|
|
EFIAPI
|
|
|
|
InternalMathRRotU64 (
|
2021-12-05 23:54:05 +01:00
|
|
|
IN UINT64 Operand,
|
|
|
|
IN UINTN Count
|
2007-06-22 05:21:45 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
return (Operand >> Count) | (Operand << (64 - Count));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Switches the endianess of a 64-bit integer.
|
|
|
|
|
|
|
|
This function swaps the bytes in a 64-bit unsigned value to switch the value
|
|
|
|
from little endian to big endian or vice versa. The byte swapped value is
|
|
|
|
returned.
|
|
|
|
|
|
|
|
@param Operand A 64-bit unsigned value.
|
|
|
|
|
2008-12-11 03:59:41 +01:00
|
|
|
@return The byte swapped Operand.
|
2007-06-22 05:21:45 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
UINT64
|
|
|
|
EFIAPI
|
|
|
|
InternalMathSwapBytes64 (
|
2021-12-05 23:54:05 +01:00
|
|
|
IN UINT64 Operand
|
2007-06-22 05:21:45 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT64 LowerBytes;
|
|
|
|
UINT64 HigherBytes;
|
|
|
|
|
2021-12-05 23:54:05 +01:00
|
|
|
LowerBytes = (UINT64)SwapBytes32 ((UINT32)Operand);
|
|
|
|
HigherBytes = (UINT64)SwapBytes32 ((UINT32)(Operand >> 32));
|
2007-06-22 05:21:45 +02:00
|
|
|
|
|
|
|
return (LowerBytes << 32 | HigherBytes);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2010-06-24 02:20:35 +02:00
|
|
|
Multiplies a 64-bit unsigned integer by a 32-bit unsigned integer
|
2007-06-22 05:21:45 +02:00
|
|
|
and generates a 64-bit unsigned result.
|
|
|
|
|
2010-06-24 02:20:35 +02:00
|
|
|
This function multiplies the 64-bit unsigned value Multiplicand by the 32-bit
|
2007-06-22 05:21:45 +02:00
|
|
|
unsigned value Multiplier and generates a 64-bit unsigned result. This 64-
|
|
|
|
bit unsigned result is returned.
|
|
|
|
|
|
|
|
@param Multiplicand A 64-bit unsigned value.
|
|
|
|
@param Multiplier A 32-bit unsigned value.
|
|
|
|
|
|
|
|
@return Multiplicand * Multiplier
|
|
|
|
|
|
|
|
**/
|
|
|
|
UINT64
|
|
|
|
EFIAPI
|
|
|
|
InternalMathMultU64x32 (
|
2021-12-05 23:54:05 +01:00
|
|
|
IN UINT64 Multiplicand,
|
|
|
|
IN UINT32 Multiplier
|
2007-06-22 05:21:45 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
return Multiplicand * Multiplier;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2010-06-24 02:20:35 +02:00
|
|
|
Multiplies a 64-bit unsigned integer by a 64-bit unsigned integer
|
2007-06-22 05:21:45 +02:00
|
|
|
and generates a 64-bit unsigned result.
|
|
|
|
|
2010-06-24 02:20:35 +02:00
|
|
|
This function multiplies the 64-bit unsigned value Multiplicand by the 64-bit
|
2007-06-22 05:21:45 +02:00
|
|
|
unsigned value Multiplier and generates a 64-bit unsigned result. This 64-
|
|
|
|
bit unsigned result is returned.
|
|
|
|
|
|
|
|
@param Multiplicand A 64-bit unsigned value.
|
|
|
|
@param Multiplier A 64-bit unsigned value.
|
|
|
|
|
2010-06-24 02:20:35 +02:00
|
|
|
@return Multiplicand * Multiplier.
|
2007-06-22 05:21:45 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
UINT64
|
|
|
|
EFIAPI
|
|
|
|
InternalMathMultU64x64 (
|
2021-12-05 23:54:05 +01:00
|
|
|
IN UINT64 Multiplicand,
|
|
|
|
IN UINT64 Multiplier
|
2007-06-22 05:21:45 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
return Multiplicand * Multiplier;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Divides a 64-bit unsigned integer by a 32-bit unsigned integer and
|
|
|
|
generates a 64-bit unsigned result.
|
|
|
|
|
|
|
|
This function divides the 64-bit unsigned value Dividend by the 32-bit
|
|
|
|
unsigned value Divisor and generates a 64-bit unsigned quotient. This
|
|
|
|
function returns the 64-bit unsigned quotient.
|
|
|
|
|
2008-12-11 03:59:41 +01:00
|
|
|
@param Dividend A 64-bit unsigned value.
|
2007-06-22 05:21:45 +02:00
|
|
|
@param Divisor A 32-bit unsigned value.
|
|
|
|
|
2010-06-24 02:20:35 +02:00
|
|
|
@return Dividend / Divisor.
|
2007-06-22 05:21:45 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
UINT64
|
|
|
|
EFIAPI
|
|
|
|
InternalMathDivU64x32 (
|
2021-12-05 23:54:05 +01:00
|
|
|
IN UINT64 Dividend,
|
|
|
|
IN UINT32 Divisor
|
2007-06-22 05:21:45 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
return Dividend / Divisor;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2008-12-11 03:59:41 +01:00
|
|
|
Divides a 64-bit unsigned integer by a 32-bit unsigned integer and
|
|
|
|
generates a 32-bit unsigned remainder.
|
2007-06-22 05:21:45 +02:00
|
|
|
|
|
|
|
This function divides the 64-bit unsigned value Dividend by the 32-bit
|
|
|
|
unsigned value Divisor and generates a 32-bit remainder. This function
|
|
|
|
returns the 32-bit unsigned remainder.
|
|
|
|
|
|
|
|
@param Dividend A 64-bit unsigned value.
|
|
|
|
@param Divisor A 32-bit unsigned value.
|
|
|
|
|
2010-06-24 02:20:35 +02:00
|
|
|
@return Dividend % Divisor.
|
2007-06-22 05:21:45 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
UINT32
|
|
|
|
EFIAPI
|
|
|
|
InternalMathModU64x32 (
|
2021-12-05 23:54:05 +01:00
|
|
|
IN UINT64 Dividend,
|
|
|
|
IN UINT32 Divisor
|
2007-06-22 05:21:45 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
return (UINT32)(Dividend % Divisor);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Divides a 64-bit unsigned integer by a 32-bit unsigned integer and
|
|
|
|
generates a 64-bit unsigned result and an optional 32-bit unsigned remainder.
|
|
|
|
|
|
|
|
This function divides the 64-bit unsigned value Dividend by the 32-bit
|
|
|
|
unsigned value Divisor and generates a 64-bit unsigned quotient. If Remainder
|
|
|
|
is not NULL, then the 32-bit unsigned remainder is returned in Remainder.
|
|
|
|
This function returns the 64-bit unsigned quotient.
|
|
|
|
|
|
|
|
@param Dividend A 64-bit unsigned value.
|
|
|
|
@param Divisor A 32-bit unsigned value.
|
|
|
|
@param Remainder A pointer to a 32-bit unsigned value. This parameter is
|
|
|
|
optional and may be NULL.
|
|
|
|
|
2010-06-24 02:20:35 +02:00
|
|
|
@return Dividend / Divisor.
|
2007-06-22 05:21:45 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
UINT64
|
|
|
|
EFIAPI
|
|
|
|
InternalMathDivRemU64x32 (
|
2021-12-05 23:54:05 +01:00
|
|
|
IN UINT64 Dividend,
|
|
|
|
IN UINT32 Divisor,
|
|
|
|
OUT UINT32 *Remainder OPTIONAL
|
2007-06-22 05:21:45 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
if (Remainder != NULL) {
|
|
|
|
*Remainder = (UINT32)(Dividend % Divisor);
|
|
|
|
}
|
2021-12-05 23:54:05 +01:00
|
|
|
|
2007-06-22 05:21:45 +02:00
|
|
|
return Dividend / Divisor;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Divides a 64-bit unsigned integer by a 64-bit unsigned integer and
|
|
|
|
generates a 64-bit unsigned result and an optional 64-bit unsigned remainder.
|
|
|
|
|
|
|
|
This function divides the 64-bit unsigned value Dividend by the 64-bit
|
|
|
|
unsigned value Divisor and generates a 64-bit unsigned quotient. If Remainder
|
|
|
|
is not NULL, then the 64-bit unsigned remainder is returned in Remainder.
|
|
|
|
This function returns the 64-bit unsigned quotient.
|
|
|
|
|
|
|
|
@param Dividend A 64-bit unsigned value.
|
|
|
|
@param Divisor A 64-bit unsigned value.
|
|
|
|
@param Remainder A pointer to a 64-bit unsigned value. This parameter is
|
|
|
|
optional and may be NULL.
|
|
|
|
|
|
|
|
@return Dividend / Divisor
|
|
|
|
|
|
|
|
**/
|
|
|
|
UINT64
|
|
|
|
EFIAPI
|
|
|
|
InternalMathDivRemU64x64 (
|
2021-12-05 23:54:05 +01:00
|
|
|
IN UINT64 Dividend,
|
|
|
|
IN UINT64 Divisor,
|
|
|
|
OUT UINT64 *Remainder OPTIONAL
|
2007-06-22 05:21:45 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
if (Remainder != NULL) {
|
|
|
|
*Remainder = Dividend % Divisor;
|
|
|
|
}
|
2021-12-05 23:54:05 +01:00
|
|
|
|
2007-06-22 05:21:45 +02:00
|
|
|
return Dividend / Divisor;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Divides a 64-bit signed integer by a 64-bit signed integer and
|
2008-12-11 03:59:41 +01:00
|
|
|
generates a 64-bit signed result and an optional 64-bit signed remainder.
|
2007-06-22 05:21:45 +02:00
|
|
|
|
2008-09-24 06:17:16 +02:00
|
|
|
This function divides the 64-bit signed value Dividend by the 64-bit
|
|
|
|
signed value Divisor and generates a 64-bit signed quotient. If Remainder
|
|
|
|
is not NULL, then the 64-bit signed remainder is returned in Remainder.
|
|
|
|
This function returns the 64-bit signed quotient.
|
2007-06-22 05:21:45 +02:00
|
|
|
|
|
|
|
@param Dividend A 64-bit signed value.
|
|
|
|
@param Divisor A 64-bit signed value.
|
|
|
|
@param Remainder A pointer to a 64-bit signed value. This parameter is
|
|
|
|
optional and may be NULL.
|
|
|
|
|
2010-06-24 02:20:35 +02:00
|
|
|
@return Dividend / Divisor.
|
2007-06-22 05:21:45 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
INT64
|
2008-07-08 11:38:43 +02:00
|
|
|
EFIAPI
|
2007-06-22 05:21:45 +02:00
|
|
|
InternalMathDivRemS64x64 (
|
2021-12-05 23:54:05 +01:00
|
|
|
IN INT64 Dividend,
|
|
|
|
IN INT64 Divisor,
|
|
|
|
OUT INT64 *Remainder OPTIONAL
|
2007-06-22 05:21:45 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
if (Remainder != NULL) {
|
|
|
|
*Remainder = Dividend % Divisor;
|
|
|
|
}
|
2021-12-05 23:54:05 +01:00
|
|
|
|
2007-06-22 05:21:45 +02:00
|
|
|
return Dividend / Divisor;
|
|
|
|
}
|