git-svn-id: https://edk2.svn.sourceforge.net/svnroot/edk2/trunk/edk2@10683 6f19259b-4bc3-4df7-8a09-765794883524
		
			
				
	
	
		
			369 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			369 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/** @file
 | 
						|
  Leaf math worker functions that require 64-bit arithmetic support from the
 | 
						|
  compiler.
 | 
						|
 | 
						|
  Copyright (c) 2006 - 2010, Intel Corporation. All rights reserved.<BR>
 | 
						|
  This program and the accompanying materials
 | 
						|
  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.
 | 
						|
 | 
						|
**/
 | 
						|
 | 
						|
#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.
 | 
						|
 | 
						|
  @return Operand << Count.
 | 
						|
 | 
						|
**/
 | 
						|
UINT64
 | 
						|
EFIAPI
 | 
						|
InternalMathLShiftU64 (
 | 
						|
  IN      UINT64                    Operand,
 | 
						|
  IN      UINTN                     Count
 | 
						|
  )
 | 
						|
{
 | 
						|
  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.
 | 
						|
 | 
						|
  @return Operand >> Count.
 | 
						|
 | 
						|
**/
 | 
						|
UINT64
 | 
						|
EFIAPI
 | 
						|
InternalMathRShiftU64 (
 | 
						|
  IN      UINT64                    Operand,
 | 
						|
  IN      UINTN                     Count
 | 
						|
  )
 | 
						|
{
 | 
						|
  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.
 | 
						|
 | 
						|
  @return Operand arithmetically shifted right by Count.
 | 
						|
 | 
						|
**/
 | 
						|
UINT64
 | 
						|
EFIAPI
 | 
						|
InternalMathARShiftU64 (
 | 
						|
  IN      UINT64                    Operand,
 | 
						|
  IN      UINTN                     Count
 | 
						|
  )
 | 
						|
{
 | 
						|
  INTN  TestValue;
 | 
						|
 | 
						|
  //
 | 
						|
  // Test if this compiler supports arithmetic shift
 | 
						|
  //
 | 
						|
  TestValue = (((-1) << (sizeof (-1) * 8 - 1)) >> (sizeof (-1) * 8 - 1));
 | 
						|
  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.
 | 
						|
 | 
						|
  @return Operand <<< Count.
 | 
						|
 | 
						|
**/
 | 
						|
UINT64
 | 
						|
EFIAPI
 | 
						|
InternalMathLRotU64 (
 | 
						|
  IN      UINT64                    Operand,
 | 
						|
  IN      UINTN                     Count
 | 
						|
  )
 | 
						|
{
 | 
						|
  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.
 | 
						|
 | 
						|
  @return Operand >>> Count.
 | 
						|
 | 
						|
**/
 | 
						|
UINT64
 | 
						|
EFIAPI
 | 
						|
InternalMathRRotU64 (
 | 
						|
  IN      UINT64                    Operand,
 | 
						|
  IN      UINTN                     Count
 | 
						|
  )
 | 
						|
{
 | 
						|
  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.
 | 
						|
 | 
						|
  @return The byte swapped Operand.
 | 
						|
 | 
						|
**/
 | 
						|
UINT64
 | 
						|
EFIAPI
 | 
						|
InternalMathSwapBytes64 (
 | 
						|
  IN      UINT64                    Operand
 | 
						|
  )
 | 
						|
{
 | 
						|
  UINT64  LowerBytes;
 | 
						|
  UINT64  HigherBytes;
 | 
						|
 | 
						|
  LowerBytes  = (UINT64) SwapBytes32 ((UINT32) Operand);
 | 
						|
  HigherBytes = (UINT64) SwapBytes32 ((UINT32) (Operand >> 32));
 | 
						|
 | 
						|
  return (LowerBytes << 32 | HigherBytes);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
  Multiplies a 64-bit unsigned integer by a 32-bit unsigned integer
 | 
						|
  and generates a 64-bit unsigned result.
 | 
						|
 | 
						|
  This function multiplies the 64-bit unsigned value Multiplicand by the 32-bit
 | 
						|
  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 (
 | 
						|
  IN      UINT64                    Multiplicand,
 | 
						|
  IN      UINT32                    Multiplier
 | 
						|
  )
 | 
						|
{
 | 
						|
  return Multiplicand * Multiplier;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/**
 | 
						|
  Multiplies a 64-bit unsigned integer by a 64-bit unsigned integer
 | 
						|
  and generates a 64-bit unsigned result.
 | 
						|
 | 
						|
  This function multiplies the 64-bit unsigned value Multiplicand by the 64-bit
 | 
						|
  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.
 | 
						|
 | 
						|
  @return Multiplicand * Multiplier.
 | 
						|
 | 
						|
**/
 | 
						|
UINT64
 | 
						|
EFIAPI
 | 
						|
InternalMathMultU64x64 (
 | 
						|
  IN      UINT64                    Multiplicand,
 | 
						|
  IN      UINT64                    Multiplier
 | 
						|
  )
 | 
						|
{
 | 
						|
  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.
 | 
						|
 | 
						|
  @param  Dividend  A 64-bit unsigned value.
 | 
						|
  @param  Divisor   A 32-bit unsigned value.
 | 
						|
 | 
						|
  @return Dividend / Divisor.
 | 
						|
 | 
						|
**/
 | 
						|
UINT64
 | 
						|
EFIAPI
 | 
						|
InternalMathDivU64x32 (
 | 
						|
  IN      UINT64                    Dividend,
 | 
						|
  IN      UINT32                    Divisor
 | 
						|
  )
 | 
						|
{
 | 
						|
  return Dividend / Divisor;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
  Divides a 64-bit unsigned integer by a 32-bit unsigned integer and
 | 
						|
  generates a 32-bit unsigned remainder.
 | 
						|
 | 
						|
  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.
 | 
						|
 | 
						|
  @return Dividend % Divisor.
 | 
						|
 | 
						|
**/
 | 
						|
UINT32
 | 
						|
EFIAPI
 | 
						|
InternalMathModU64x32 (
 | 
						|
  IN      UINT64                    Dividend,
 | 
						|
  IN      UINT32                    Divisor
 | 
						|
  )
 | 
						|
{
 | 
						|
  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.
 | 
						|
 | 
						|
  @return Dividend / Divisor.
 | 
						|
 | 
						|
**/
 | 
						|
UINT64
 | 
						|
EFIAPI
 | 
						|
InternalMathDivRemU64x32 (
 | 
						|
  IN      UINT64                    Dividend,
 | 
						|
  IN      UINT32                    Divisor,
 | 
						|
  OUT     UINT32                    *Remainder OPTIONAL
 | 
						|
  )
 | 
						|
{
 | 
						|
  if (Remainder != NULL) {
 | 
						|
    *Remainder = (UINT32)(Dividend % Divisor);
 | 
						|
  }
 | 
						|
  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 (
 | 
						|
  IN      UINT64                    Dividend,
 | 
						|
  IN      UINT64                    Divisor,
 | 
						|
  OUT     UINT64                    *Remainder OPTIONAL
 | 
						|
  )
 | 
						|
{
 | 
						|
  if (Remainder != NULL) {
 | 
						|
    *Remainder = Dividend % Divisor;
 | 
						|
  }
 | 
						|
  return Dividend / Divisor;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
  Divides a 64-bit signed integer by a 64-bit signed integer and
 | 
						|
  generates a 64-bit signed result and an optional 64-bit signed remainder.
 | 
						|
 | 
						|
  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.
 | 
						|
 | 
						|
  @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.
 | 
						|
 | 
						|
  @return Dividend / Divisor.
 | 
						|
 | 
						|
**/
 | 
						|
INT64
 | 
						|
EFIAPI
 | 
						|
InternalMathDivRemS64x64 (
 | 
						|
  IN      INT64                     Dividend,
 | 
						|
  IN      INT64                     Divisor,
 | 
						|
  OUT     INT64                     *Remainder  OPTIONAL
 | 
						|
  )
 | 
						|
{
 | 
						|
  if (Remainder != NULL) {
 | 
						|
    *Remainder = Dividend % Divisor;
 | 
						|
  }
 | 
						|
  return Dividend / Divisor;
 | 
						|
}
 |