/* * Large integer functions * * Copyright 2000 Alexandre Julliard * Copyright 2003 Thomas Mertes * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include "config.h" #include "wine/port.h" #include #include "ntstatus.h" #define WIN32_NO_STATUS #include "windef.h" #include "winternl.h" #include "wine/asm.h" #ifndef _WIN64 /* * Note: we use LONGLONG instead of LARGE_INTEGER, because * the latter is a structure and the calling convention for * returning a structure would not be binary-compatible. * * FIXME: for platforms that don't have a native LONGLONG type, * we should define LONGLONG as a structure similar to LARGE_INTEGER * and do everything by hand. You are welcome to do it... */ /****************************************************************************** * RtlLargeIntegerAdd (NTDLL.@) * * Add two 64 bit integers. * * PARAMS * a [I] Initial number. * b [I] Number to add to a. * * RETURNS * The sum of a and b. */ LONGLONG WINAPI RtlLargeIntegerAdd( LONGLONG a, LONGLONG b ) { return a + b; } /****************************************************************************** * RtlLargeIntegerSubtract (NTDLL.@) * * Subtract two 64 bit integers. * * PARAMS * a [I] Initial number. * b [I] Number to subtract from a. * * RETURNS * The difference of a and b. */ LONGLONG WINAPI RtlLargeIntegerSubtract( LONGLONG a, LONGLONG b ) { return a - b; } /****************************************************************************** * RtlLargeIntegerNegate (NTDLL.@) * * Negate a 64 bit integer. * * PARAMS * a [I] Initial number. * * RETURNS * The value of a negated. */ LONGLONG WINAPI RtlLargeIntegerNegate( LONGLONG a ) { return -a; } /****************************************************************************** * RtlLargeIntegerShiftLeft (NTDLL.@) * * Perform a shift left on a 64 bit integer. * * PARAMS * a [I] Initial number. * count [I] Number of bits to shift by * * RETURNS * The value of a following the shift. */ LONGLONG WINAPI RtlLargeIntegerShiftLeft( LONGLONG a, INT count ) { return a << count; } /****************************************************************************** * RtlLargeIntegerShiftRight (NTDLL.@) * * Perform a shift right on a 64 bit integer. * * PARAMS * a [I] Initial number. * count [I] Number of bits to shift by * * RETURNS * The value of a following the shift. */ LONGLONG WINAPI RtlLargeIntegerShiftRight( LONGLONG a, INT count ) { return (ULONGLONG)a >> count; } /****************************************************************************** * RtlLargeIntegerArithmeticShift (NTDLL.@) * * Perform an arithmetic shift right on a 64 bit integer. * * PARAMS * a [I] Initial number. * count [I] Number of bits to shift by * * RETURNS * The value of a following the shift. */ LONGLONG WINAPI RtlLargeIntegerArithmeticShift( LONGLONG a, INT count ) { /* FIXME: gcc does arithmetic shift here, but it may not be true on all platforms */ return a >> count; } /****************************************************************************** * RtlLargeIntegerDivide (NTDLL.@) * * Divide one 64 bit unsigned integer by another, with remainder. * * PARAMS * a [I] Initial number. * b [I] Number to divide a by * rem [O] Destination for remainder * * RETURNS * The dividend of a and b. If rem is non-NULL it is set to the remainder. * * FIXME * Should it be signed division instead? */ ULONGLONG WINAPI RtlLargeIntegerDivide( ULONGLONG a, ULONGLONG b, ULONGLONG *rem ) { ULONGLONG ret = a / b; if (rem) *rem = a - ret * b; return ret; } /****************************************************************************** * RtlConvertLongToLargeInteger (NTDLL.@) * * Convert a 32 bit integer into 64 bits. * * PARAMS * a [I] Number to convert * * RETURNS * a. */ LONGLONG WINAPI RtlConvertLongToLargeInteger( LONG a ) { return a; } /****************************************************************************** * RtlConvertUlongToLargeInteger (NTDLL.@) * * Convert a 32 bit unsigned integer into 64 bits. * * PARAMS * a [I] Number to convert * * RETURNS * a. */ ULONGLONG WINAPI RtlConvertUlongToLargeInteger( ULONG a ) { return a; } /****************************************************************************** * RtlEnlargedIntegerMultiply (NTDLL.@) * * Multiply two integers giving a 64 bit integer result. * * PARAMS * a [I] Initial number. * b [I] Number to multiply a by. * * RETURNS * The product of a and b. */ LONGLONG WINAPI RtlEnlargedIntegerMultiply( INT a, INT b ) { return (LONGLONG)a * b; } /****************************************************************************** * RtlEnlargedUnsignedMultiply (NTDLL.@) * * Multiply two unsigned integers giving a 64 bit unsigned integer result. * * PARAMS * a [I] Initial number. * b [I] Number to multiply a by. * * RETURNS * The product of a and b. */ ULONGLONG WINAPI RtlEnlargedUnsignedMultiply( UINT a, UINT b ) { return (ULONGLONG)a * b; } /****************************************************************************** * RtlEnlargedUnsignedDivide (NTDLL.@) * * Divide one 64 bit unsigned integer by a 32 bit unsigned integer, with remainder. * * PARAMS * a [I] Initial number. * b [I] Number to divide a by * remptr [O] Destination for remainder * * RETURNS * The dividend of a and b. If remptr is non-NULL it is set to the remainder. */ UINT WINAPI RtlEnlargedUnsignedDivide( ULONGLONG a, UINT b, UINT *remptr ) { #if defined(__i386__) && defined(__GNUC__) UINT ret, rem; __asm__("divl %4" : "=a" (ret), "=d" (rem) : "0" ((UINT)a), "1" ((UINT)(a >> 32)), "g" (b) ); if (remptr) *remptr = rem; return ret; #else UINT ret = a / b; if (remptr) *remptr = a % b; return ret; #endif } /****************************************************************************** * RtlExtendedLargeIntegerDivide (NTDLL.@) * * Divide one 64 bit integer by a 32 bit integer, with remainder. * * PARAMS * a [I] Initial number. * b [I] Number to divide a by * rem [O] Destination for remainder * * RETURNS * The dividend of a and b. If rem is non-NULL it is set to the remainder. */ LONGLONG WINAPI RtlExtendedLargeIntegerDivide( LONGLONG a, INT b, INT *rem ) { LONGLONG ret = a / b; if (rem) *rem = a - b * ret; return ret; } /****************************************************************************** * RtlExtendedIntegerMultiply (NTDLL.@) * * Multiply one 64 bit integer by another 32 bit integer. * * PARAMS * a [I] Initial number. * b [I] Number to multiply a by. * * RETURNS * The product of a and b. */ LONGLONG WINAPI RtlExtendedIntegerMultiply( LONGLONG a, INT b ) { return a * b; } /****************************************************************************** * RtlExtendedMagicDivide (NTDLL.@) * * Allows replacing a division by a longlong constant with a multiplication by * the inverse constant. * * RETURNS * (dividend * inverse_divisor) >> (64 + shift) * * NOTES * If the divisor of a division is constant, the constants inverse_divisor and * shift must be chosen such that inverse_divisor = 2^(64 + shift) / divisor. * Then we have RtlExtendedMagicDivide(dividend,inverse_divisor,shift) == * dividend * inverse_divisor / 2^(64 + shift) == dividend / divisor. * * The Parameter inverse_divisor although defined as LONGLONG is used as * ULONGLONG. */ #define LOWER_32(A) ((A) & 0xffffffff) #define UPPER_32(A) ((A) >> 32) LONGLONG WINAPI RtlExtendedMagicDivide( LONGLONG dividend, /* [I] Dividend to be divided by the constant divisor */ LONGLONG inverse_divisor, /* [I] Constant computed manually as 2^(64+shift) / divisor */ INT shift) /* [I] Constant shift chosen to make inverse_divisor as big as possible for 64 bits */ { ULONGLONG dividend_high; ULONGLONG dividend_low; ULONGLONG inverse_divisor_high; ULONGLONG inverse_divisor_low; ULONGLONG ah_bl; ULONGLONG al_bh; LONGLONG result; int positive; if (dividend < 0) { dividend_high = UPPER_32((ULONGLONG) -dividend); dividend_low = LOWER_32((ULONGLONG) -dividend); positive = 0; } else { dividend_high = UPPER_32((ULONGLONG) dividend); dividend_low = LOWER_32((ULONGLONG) dividend); positive = 1; } /* if */ inverse_divisor_high = UPPER_32((ULONGLONG) inverse_divisor); inverse_divisor_low = LOWER_32((ULONGLONG) inverse_divisor); ah_bl = dividend_high * inverse_divisor_low; al_bh = dividend_low * inverse_divisor_high; result = (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))) >> shift); if (positive) { return result; } else { return -result; } /* if */ } /************************************************************************* * RtlInterlockedCompareExchange64 (NTDLL.@) */ #ifdef __GCC_HAVE_SYNC_COMPARE_AND_SWAP_8 LONGLONG WINAPI RtlInterlockedCompareExchange64( LONGLONG *dest, LONGLONG xchg, LONGLONG compare ) { return __sync_val_compare_and_swap( dest, compare, xchg ); } #else __ASM_STDCALL_FUNC(RtlInterlockedCompareExchange64, 20, "push %ebx\n\t" __ASM_CFI(".cfi_adjust_cfa_offset 4\n\t") __ASM_CFI(".cfi_rel_offset %ebx,0\n\t") "push %esi\n\t" __ASM_CFI(".cfi_adjust_cfa_offset 4\n\t") __ASM_CFI(".cfi_rel_offset %esi,0\n\t") "movl 12(%esp),%esi\n\t" "movl 16(%esp),%ebx\n\t" "movl 20(%esp),%ecx\n\t" "movl 24(%esp),%eax\n\t" "movl 28(%esp),%edx\n\t" "lock; cmpxchg8b (%esi)\n\t" "pop %esi\n\t" __ASM_CFI(".cfi_same_value %esi\n\t") __ASM_CFI(".cfi_adjust_cfa_offset -4\n\t") "pop %ebx\n\t" __ASM_CFI(".cfi_same_value %ebx\n\t") __ASM_CFI(".cfi_adjust_cfa_offset -4\n\t") "ret $20") #endif #endif /* _WIN64 */ /****************************************************************************** * RtlLargeIntegerToChar [NTDLL.@] * * Convert an unsigned large integer to a character string. * * RETURNS * Success: STATUS_SUCCESS. str contains the converted number * Failure: STATUS_INVALID_PARAMETER, if base is not 0, 2, 8, 10 or 16. * STATUS_BUFFER_OVERFLOW, if str would be larger than length. * STATUS_ACCESS_VIOLATION, if str is NULL. * * NOTES * Instead of base 0 it uses 10 as base. * Writes at most length characters to the string str. * Str is '\0' terminated when length allows it. * When str fits exactly in length characters the '\0' is omitted. * If value_ptr is NULL it crashes, as the native function does. * * DIFFERENCES * - Accept base 0 as 10 instead of crashing as native function does. * - The native function does produce garbage or STATUS_BUFFER_OVERFLOW for * base 2, 8 and 16 when the value is larger than 0xFFFFFFFF. */ NTSTATUS WINAPI RtlLargeIntegerToChar( const ULONGLONG *value_ptr, /* [I] Pointer to the value to be converted */ ULONG base, /* [I] Number base for conversion (allowed 0, 2, 8, 10 or 16) */ ULONG length, /* [I] Length of the str buffer in bytes */ PCHAR str) /* [O] Destination for the converted value */ { ULONGLONG value = *value_ptr; CHAR buffer[65]; PCHAR pos; CHAR digit; ULONG len; if (base == 0) { base = 10; } else if (base != 2 && base != 8 && base != 10 && base != 16) { return STATUS_INVALID_PARAMETER; } /* if */ pos = &buffer[64]; *pos = '\0'; do { pos--; digit = value % base; value = value / base; if (digit < 10) { *pos = '0' + digit; } else { *pos = 'A' + digit - 10; } /* if */ } while (value != 0L); len = &buffer[64] - pos; if (len > length) { return STATUS_BUFFER_OVERFLOW; } else if (str == NULL) { return STATUS_ACCESS_VIOLATION; } else if (len == length) { memcpy(str, pos, len); } else { memcpy(str, pos, len + 1); } /* if */ return STATUS_SUCCESS; } /************************************************************************** * RtlInt64ToUnicodeString (NTDLL.@) * * Convert a large unsigned integer to a '\0' terminated unicode string. * * RETURNS * Success: STATUS_SUCCESS. str contains the converted number * Failure: STATUS_INVALID_PARAMETER, if base is not 0, 2, 8, 10 or 16. * STATUS_BUFFER_OVERFLOW, if str is too small to hold the string * (with the '\0' termination). In this case str->Length * is set to the length, the string would have (which can * be larger than the MaximumLength). * * NOTES * Instead of base 0 it uses 10 as base. * If str is NULL it crashes, as the native function does. * * DIFFERENCES * - Accept base 0 as 10 instead of crashing as native function does. * - Do not return STATUS_BUFFER_OVERFLOW when the string is long enough. * The native function does this when the string would be longer than 31 * characters even when the string parameter is long enough. * - The native function does produce garbage or STATUS_BUFFER_OVERFLOW for * base 2, 8 and 16 when the value is larger than 0xFFFFFFFF. */ NTSTATUS WINAPI RtlInt64ToUnicodeString( ULONGLONG value, /* [I] Value to be converted */ ULONG base, /* [I] Number base for conversion (allowed 0, 2, 8, 10 or 16) */ UNICODE_STRING *str) /* [O] Destination for the converted value */ { WCHAR buffer[65]; PWCHAR pos; WCHAR digit; if (base == 0) { base = 10; } else if (base != 2 && base != 8 && base != 10 && base != 16) { return STATUS_INVALID_PARAMETER; } /* if */ pos = &buffer[64]; *pos = '\0'; do { pos--; digit = value % base; value = value / base; if (digit < 10) { *pos = '0' + digit; } else { *pos = 'A' + digit - 10; } /* if */ } while (value != 0L); str->Length = (&buffer[64] - pos) * sizeof(WCHAR); if (str->Length >= str->MaximumLength) { return STATUS_BUFFER_OVERFLOW; } else { memcpy(str->Buffer, pos, str->Length + sizeof(WCHAR)); } /* if */ return STATUS_SUCCESS; } #ifdef __i386__ /* those builtin functions use stdcall calling convention, but compilers reference them without stdcall declarations */ #if defined(__MINGW32__) || defined(_MSC_VER) LONGLONG WINAPI _alldiv( LONGLONG a, LONGLONG b ) asm("_alldiv"); LONGLONG WINAPI _allmul( LONGLONG a, LONGLONG b ) asm("_allmul"); LONGLONG WINAPI _allrem( LONGLONG a, LONGLONG b ) asm("_allrem"); ULONGLONG WINAPI _aulldiv( ULONGLONG a, ULONGLONG b ) asm("_aulldiv"); ULONGLONG WINAPI _aullrem( ULONGLONG a, ULONGLONG b ) asm("_aullrem"); #endif static ULONGLONG udivmod(ULONGLONG a, ULONGLONG b, ULONGLONG *rem) { const ULARGE_INTEGER n = { .QuadPart = a }; const ULARGE_INTEGER d = { .QuadPart = b }; DWORD sr, carry, index; ULARGE_INTEGER q, r; const unsigned n_uword_bits = 32; const unsigned n_udword_bits = 64; /* special cases, X is unknown, K != 0 */ if (n.u.HighPart == 0) { if (d.u.HighPart == 0) { /* 0 X / 0 X */ if (rem) *rem = n.u.LowPart % d.u.LowPart; return n.u.LowPart / d.u.LowPart; } /* 0 X / K X */ if (rem) *rem = n.u.LowPart; return 0; } /* n.u.HighPart != 0 */ if (d.u.LowPart == 0) { if (d.u.HighPart == 0) { /* K X / 0 0 */ if (rem) *rem = n.u.HighPart % d.u.LowPart; return n.u.HighPart / d.u.LowPart; } /* d.u.HighPart != 0 */ if (n.u.LowPart == 0) { /* K 0 / K 0 */ if (rem) { r.u.HighPart = n.u.HighPart % d.u.HighPart; r.u.LowPart = 0; *rem = r.QuadPart; } return n.u.HighPart / d.u.HighPart; } /* K K / K 0 */ if ((d.u.HighPart & (d.u.HighPart - 1)) == 0) /* if d is a power of 2 */ { if (rem) { r.u.LowPart = n.u.LowPart; r.u.HighPart = n.u.HighPart & (d.u.HighPart - 1); *rem = r.QuadPart; } BitScanForward(&index, d.u.HighPart); return n.u.HighPart >> index; } /* K K / K 0 */ BitScanReverse(&index, d.u.HighPart); BitScanReverse(&sr, n.u.HighPart); sr -= index; /* 0 <= sr <= n_uword_bits - 2 or sr large */ if (sr > n_uword_bits - 2) { if (rem) *rem = n.QuadPart; return 0; } ++sr; /* 1 <= sr <= n_uword_bits - 1 */ /* q.QuadPart = n.QuadPart << (n_udword_bits - sr); */ q.u.LowPart = 0; q.u.HighPart = n.u.LowPart << (n_uword_bits - sr); /* r.QuadPart = n.QuadPart >> sr; */ r.u.HighPart = n.u.HighPart >> sr; r.u.LowPart = (n.u.HighPart << (n_uword_bits - sr)) | (n.u.LowPart >> sr); } else /* d.u.LowPart != 0 */ { if (d.u.HighPart == 0) { /* K X / 0 K */ if ((d.u.LowPart & (d.u.LowPart - 1)) == 0) /* if d is a power of 2 */ { if (rem) *rem = n.u.LowPart & (d.u.LowPart - 1); if (d.u.LowPart == 1) return n.QuadPart; BitScanForward(&sr, d.u.LowPart); q.u.HighPart = n.u.HighPart >> sr; q.u.LowPart = (n.u.HighPart << (n_uword_bits - sr)) | (n.u.LowPart >> sr); return q.QuadPart; } BitScanReverse(&index, d.u.LowPart); BitScanReverse(&sr, n.u.HighPart); sr = 1 + n_uword_bits + sr - index; /* 2 <= sr <= n_udword_bits - 1 * q.QuadPart = n.QuadPart << (n_udword_bits - sr); * r.QuadPart = n.QuadPart >> sr; */ if (sr == n_uword_bits) { q.u.LowPart = 0; q.u.HighPart = n.u.LowPart; r.u.HighPart = 0; r.u.LowPart = n.u.HighPart; } else if (sr < n_uword_bits) /* 2 <= sr <= n_uword_bits - 1 */ { q.u.LowPart = 0; q.u.HighPart = n.u.LowPart << (n_uword_bits - sr); r.u.HighPart = n.u.HighPart >> sr; r.u.LowPart = (n.u.HighPart << (n_uword_bits - sr)) | (n.u.LowPart >> sr); } else /* n_uword_bits + 1 <= sr <= n_udword_bits - 1 */ { q.u.LowPart = n.u.LowPart << (n_udword_bits - sr); q.u.HighPart = (n.u.HighPart << (n_udword_bits - sr)) | (n.u.LowPart >> (sr - n_uword_bits)); r.u.HighPart = 0; r.u.LowPart = n.u.HighPart >> (sr - n_uword_bits); } } else { /* K X / K K */ BitScanReverse(&index, d.u.HighPart); BitScanReverse(&sr, n.u.HighPart); sr -= index; /* 0 <= sr <= n_uword_bits - 1 or sr large */ if (sr > n_uword_bits - 1) { if (rem) *rem = n.QuadPart; return 0; } ++sr; /* 1 <= sr <= n_uword_bits * q.QuadPart = n.QuadPart << (n_udword_bits - sr); */ q.u.LowPart = 0; if (sr == n_uword_bits) { q.u.HighPart = n.u.LowPart; r.u.HighPart = 0; r.u.LowPart = n.u.HighPart; } else { q.u.HighPart = n.u.LowPart << (n_uword_bits - sr); r.u.HighPart = n.u.HighPart >> sr; r.u.LowPart = (n.u.HighPart << (n_uword_bits - sr)) | (n.u.LowPart >> sr); } } } /* Not a special case * q and r are initialized with: * q.QuadPart = n.QuadPart << (n_udword_bits - sr); * r.QuadPart = n.QuadPart >> sr; * 1 <= sr <= n_udword_bits - 1 */ carry = 0; for (; sr > 0; --sr) { LONGLONG s; /* r:q = ((r:q) << 1) | carry */ r.u.HighPart = (r.u.HighPart << 1) | (r.u.LowPart >> (n_uword_bits - 1)); r.u.LowPart = (r.u.LowPart << 1) | (q.u.HighPart >> (n_uword_bits - 1)); q.u.HighPart = (q.u.HighPart << 1) | (q.u.LowPart >> (n_uword_bits - 1)); q.u.LowPart = (q.u.LowPart << 1) | carry; /* if (r.QuadPart >= d.QuadPart) * { * r.QuadPart -= d.QuadPart; * carry = 1; * } */ s = (LONGLONG)(d.QuadPart - r.QuadPart - 1) >> (n_udword_bits - 1); carry = s & 1; r.QuadPart -= d.QuadPart & s; } q.QuadPart = (q.QuadPart << 1) | carry; if (rem) *rem = r.QuadPart; return q.QuadPart; } /****************************************************************************** * _alldiv (NTDLL.@) * * Divide two 64 bit unsigned integers. * * PARAMS * a [I] Initial number. * b [I] Number to divide a by. * * RETURNS * The dividend of a and b. */ LONGLONG WINAPI _alldiv( LONGLONG a, LONGLONG b ) { LONGLONG s_a = a >> 63; /* s_a = a < 0 ? -1 : 0 */ LONGLONG s_b = b >> 63; /* s_b = b < 0 ? -1 : 0 */ a = (a ^ s_a) - s_a; /* negate if s_a == -1 */ b = (b ^ s_b) - s_b; /* negate if s_b == -1 */ s_a ^= s_b; /* sign of quotient */ return (udivmod(a, b, NULL) ^ s_a) - s_a; /* negate if s_a == -1 */ } /****************************************************************************** * _allmul (NTDLL.@) * * Multiply two 64 bit integers. * * PARAMS * a [I] Initial number. * b [I] Number to multiply a by. * * RETURNS * The product of a and b. */ LONGLONG WINAPI _allmul( LONGLONG a, LONGLONG b ) { LARGE_INTEGER x = { .QuadPart = a }; LARGE_INTEGER y = { .QuadPart = b }; LARGE_INTEGER r; unsigned int t; const int bits_in_word_2 = 16; const unsigned int lower_mask = ~0u >> bits_in_word_2; r.u.LowPart = (x.u.LowPart & lower_mask) * (y.u.LowPart & lower_mask); t = r.u.LowPart >> bits_in_word_2; r.u.LowPart &= lower_mask; t += (x.u.LowPart >> bits_in_word_2) * (y.u.LowPart & lower_mask); r.u.LowPart += (t & lower_mask) << bits_in_word_2; r.u.HighPart = t >> bits_in_word_2; t = r.u.LowPart >> bits_in_word_2; r.u.LowPart &= lower_mask; t += (y.u.LowPart >> bits_in_word_2) * (x.u.LowPart & lower_mask); r.u.LowPart += (t & lower_mask) << bits_in_word_2; r.u.HighPart += t >> bits_in_word_2; r.u.HighPart += (x.u.LowPart >> bits_in_word_2) * (y.u.LowPart >> bits_in_word_2); r.u.HighPart += x.u.HighPart * y.u.LowPart + x.u.LowPart * y.u.HighPart; return r.QuadPart; } /****************************************************************************** * _allrem (NTDLL.@) * * Calculate the remainder after dividing two 64 bit integers. * * PARAMS * a [I] Initial number. * b [I] Number to divide a by. * * RETURNS * The remainder of a divided by b. */ LONGLONG WINAPI _allrem( LONGLONG a, LONGLONG b ) { LONGLONG s = b >> 63; /* s = b < 0 ? -1 : 0 */ ULONGLONG r; b = (b ^ s) - s; /* negate if s == -1 */ s = a >> 63; /* s = a < 0 ? -1 : 0 */ a = (a ^ s) - s; /* negate if s == -1 */ udivmod(a, b, &r); return ((LONGLONG)r ^ s) - s; /* negate if s == -1 */ } /****************************************************************************** * _aulldiv (NTDLL.@) * * Divide two 64 bit unsigned integers. * * PARAMS * a [I] Initial number. * b [I] Number to divide a by. * * RETURNS * The dividend of a and b. */ ULONGLONG WINAPI _aulldiv( ULONGLONG a, ULONGLONG b ) { return udivmod(a, b, NULL); } /****************************************************************************** * _allshl (NTDLL.@) * * Shift a 64 bit integer to the left. * * PARAMS * a [I] Initial number. * b [I] Number to shift a by to the left. * * RETURNS * The left-shifted value. */ LONGLONG WINAPI _allshl( LONGLONG a, LONG b ) { return a << b; } /****************************************************************************** * _allshr (NTDLL.@) * * Shift a 64 bit integer to the right. * * PARAMS * a [I] Initial number. * b [I] Number to shift a by to the right. * * RETURNS * The right-shifted value. */ LONGLONG WINAPI _allshr( LONGLONG a, LONG b ) { return a >> b; } /****************************************************************************** * _alldvrm (NTDLL.@) * * Divide two 64 bit integers. * * PARAMS * a [I] Initial number. * b [I] Number to divide a by. * * RETURNS * Returns the quotient of a and b in edx:eax. * Returns the remainder of a and b in ebx:ecx. */ __ASM_GLOBAL_FUNC( _alldvrm, "pushl %ebp\n\t" __ASM_CFI(".cfi_adjust_cfa_offset 4\n\t") __ASM_CFI(".cfi_rel_offset %ebp,0\n\t") "movl %esp,%ebp\n\t" __ASM_CFI(".cfi_def_cfa_register %ebp\n\t") "pushl 20(%ebp)\n\t" "pushl 16(%ebp)\n\t" "pushl 12(%ebp)\n\t" "pushl 8(%ebp)\n\t" "call " __ASM_NAME("_allrem") "\n\t" "movl %edx,%ebx\n\t" "pushl %eax\n\t" "pushl 20(%ebp)\n\t" "pushl 16(%ebp)\n\t" "pushl 12(%ebp)\n\t" "pushl 8(%ebp)\n\t" "call " __ASM_NAME("_alldiv") "\n\t" "popl %ecx\n\t" "leave\n\t" __ASM_CFI(".cfi_def_cfa %esp,4\n\t") __ASM_CFI(".cfi_same_value %ebp\n\t") "ret $16" ) /****************************************************************************** * _aullrem (NTDLL.@) * * Calculate the remainder after dividing two 64 bit unsigned integers. * * PARAMS * a [I] Initial number. * b [I] Number to divide a by. * * RETURNS * The remainder of a divided by b. */ ULONGLONG WINAPI _aullrem( ULONGLONG a, ULONGLONG b ) { ULONGLONG r; udivmod(a, b, &r); return r; } /****************************************************************************** * _aullshr (NTDLL.@) * * Shift a 64 bit unsigned integer to the right. * * PARAMS * a [I] Initial number. * b [I] Number to shift a by to the right. * * RETURNS * The right-shifted value. */ ULONGLONG WINAPI _aullshr( ULONGLONG a, LONG b ) { return a >> b; } /****************************************************************************** * _aulldvrm (NTDLL.@) * * Divide two 64 bit unsigned integers. * * PARAMS * a [I] Initial number. * b [I] Number to divide a by. * * RETURNS * Returns the quotient of a and b in edx:eax. * Returns the remainder of a and b in ebx:ecx. */ __ASM_GLOBAL_FUNC( _aulldvrm, "pushl %ebp\n\t" __ASM_CFI(".cfi_adjust_cfa_offset 4\n\t") __ASM_CFI(".cfi_rel_offset %ebp,0\n\t") "movl %esp,%ebp\n\t" __ASM_CFI(".cfi_def_cfa_register %ebp\n\t") "pushl 20(%ebp)\n\t" "pushl 16(%ebp)\n\t" "pushl 12(%ebp)\n\t" "pushl 8(%ebp)\n\t" "call " __ASM_NAME("_aullrem") "\n\t" "movl %edx,%ebx\n\t" "pushl %eax\n\t" "pushl 20(%ebp)\n\t" "pushl 16(%ebp)\n\t" "pushl 12(%ebp)\n\t" "pushl 8(%ebp)\n\t" "call " __ASM_NAME("_aulldiv") "\n\t" "popl %ecx\n\t" "leave\n\t" __ASM_CFI(".cfi_def_cfa %esp,4\n\t") __ASM_CFI(".cfi_same_value %ebp\n\t") "ret $16" ) #endif /* __i386__ */