mirror of
https://github.com/reactos/reactos.git
synced 2024-12-28 01:55:19 +00:00
Implemented RtlExtendedMagicDivide(). Patch by Marc Schuetz.
svn path=/trunk/; revision=4814
This commit is contained in:
parent
8eb47000b7
commit
b036cea0eb
1 changed files with 70 additions and 10 deletions
|
@ -1,4 +1,4 @@
|
||||||
/* $Id: largeint.c,v 1.13 2002/12/08 16:23:32 robd Exp $
|
/* $Id: largeint.c,v 1.14 2003/06/01 18:13:23 ekohl Exp $
|
||||||
*
|
*
|
||||||
* COPYRIGHT: See COPYING in the top level directory
|
* COPYRIGHT: See COPYING in the top level directory
|
||||||
* PROJECT: ReactOS kernel
|
* PROJECT: ReactOS kernel
|
||||||
|
@ -119,18 +119,78 @@ RtlExtendedLargeIntegerDivide (
|
||||||
return RC;
|
return RC;
|
||||||
}
|
}
|
||||||
|
|
||||||
LARGE_INTEGER
|
|
||||||
STDCALL
|
|
||||||
RtlExtendedMagicDivide (LARGE_INTEGER Dividend,
|
|
||||||
LARGE_INTEGER MagicDivisor,
|
|
||||||
CCHAR ShiftCount)
|
|
||||||
{
|
|
||||||
LARGE_INTEGER Result;
|
|
||||||
|
|
||||||
Result.QuadPart = (Dividend.QuadPart * MagicDivisor.QuadPart) >> ShiftCount;
|
/******************************************************************************
|
||||||
return(Result);
|
* RtlExtendedMagicDivide
|
||||||
|
*
|
||||||
|
* Allows replacing a division by a longlong constant with a multiplication by
|
||||||
|
* the inverse constant.
|
||||||
|
*
|
||||||
|
* RETURNS
|
||||||
|
* (Dividend * MagicDivisor) >> (64 + ShiftCount)
|
||||||
|
*
|
||||||
|
* NOTES
|
||||||
|
* If the divisor of a division is constant, the constants MagicDivisor and
|
||||||
|
* shift must be chosen such that
|
||||||
|
* MagicDivisor = 2^(64 + ShiftCount) / Divisor.
|
||||||
|
*
|
||||||
|
* Then we have RtlExtendedMagicDivide(Dividend,MagicDivisor,ShiftCount) ==
|
||||||
|
* Dividend * MagicDivisor / 2^(64 + ShiftCount) == Dividend / Divisor.
|
||||||
|
*
|
||||||
|
* The Parameter MagicDivisor although defined as LONGLONG is used as
|
||||||
|
* ULONGLONG.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define LOWER_32(A) ((A) & 0xffffffff)
|
||||||
|
#define UPPER_32(A) ((A) >> 32)
|
||||||
|
|
||||||
|
LARGE_INTEGER STDCALL
|
||||||
|
RtlExtendedMagicDivide (LARGE_INTEGER Dividend,
|
||||||
|
LARGE_INTEGER MagicDivisor,
|
||||||
|
CCHAR ShiftCount)
|
||||||
|
{
|
||||||
|
ULONGLONG dividend_high;
|
||||||
|
ULONGLONG dividend_low;
|
||||||
|
ULONGLONG inverse_divisor_high;
|
||||||
|
ULONGLONG inverse_divisor_low;
|
||||||
|
ULONGLONG ah_bl;
|
||||||
|
ULONGLONG al_bh;
|
||||||
|
LARGE_INTEGER result;
|
||||||
|
BOOLEAN positive;
|
||||||
|
|
||||||
|
if (Dividend.QuadPart < 0)
|
||||||
|
{
|
||||||
|
dividend_high = UPPER_32((ULONGLONG) -Dividend.QuadPart);
|
||||||
|
dividend_low = LOWER_32((ULONGLONG) -Dividend.QuadPart);
|
||||||
|
positive = FALSE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
dividend_high = UPPER_32((ULONGLONG) Dividend.QuadPart);
|
||||||
|
dividend_low = LOWER_32((ULONGLONG) Dividend.QuadPart);
|
||||||
|
positive = TRUE;
|
||||||
|
}
|
||||||
|
inverse_divisor_high = UPPER_32((ULONGLONG) MagicDivisor.QuadPart);
|
||||||
|
inverse_divisor_low = LOWER_32((ULONGLONG) MagicDivisor.QuadPart);
|
||||||
|
|
||||||
|
ah_bl = dividend_high * inverse_divisor_low;
|
||||||
|
al_bh = dividend_low * inverse_divisor_high;
|
||||||
|
|
||||||
|
result.QuadPart =
|
||||||
|
(LONGLONG) ((dividend_high * inverse_divisor_high +
|
||||||
|
UPPER_32(ah_bl) +
|
||||||
|
UPPER_32(al_bh) +
|
||||||
|
UPPER_32(LOWER_32(ah_bl) + LOWER_32(al_bh) +
|
||||||
|
UPPER_32(dividend_low * inverse_divisor_low))) >> ShiftCount);
|
||||||
|
if (!positive)
|
||||||
|
{
|
||||||
|
result.QuadPart = -result.QuadPart;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
LARGE_INTEGER
|
LARGE_INTEGER
|
||||||
STDCALL
|
STDCALL
|
||||||
RtlLargeIntegerAdd (
|
RtlLargeIntegerAdd (
|
||||||
|
|
Loading…
Reference in a new issue