mirror of
https://github.com/reactos/reactos.git
synced 2024-11-02 04:37:32 +00:00
72 lines
1.7 KiB
C
72 lines
1.7 KiB
C
|
/*
|
||
|
* PROJECT: ReactOS CRT library
|
||
|
* LICENSE: MIT (https://spdx.org/licenses/MIT)
|
||
|
* PURPOSE: Implementation of floor
|
||
|
* COPYRIGHT: Copyright 2021 Timo Kreuzer <timo.kreuzer@reactos.org>
|
||
|
*/
|
||
|
|
||
|
#define _USE_MATH_DEFINES
|
||
|
#include <math.h>
|
||
|
#include <limits.h>
|
||
|
|
||
|
#ifdef _MSC_VER
|
||
|
#pragma function(floor)
|
||
|
#endif
|
||
|
|
||
|
double
|
||
|
__cdecl
|
||
|
floor(double x)
|
||
|
{
|
||
|
/* Load the value as uint64 */
|
||
|
unsigned long long u64 = *(unsigned long long*)&x;
|
||
|
|
||
|
/* Check for NAN */
|
||
|
if ((u64 & ~(1ULL << 63)) > 0x7FF0000000000000ull)
|
||
|
{
|
||
|
/* Set error bit */
|
||
|
u64 |= 0x0008000000000000ull;
|
||
|
return *(double*)&u64;
|
||
|
}
|
||
|
|
||
|
/* Check if x is positive */
|
||
|
if ((u64 & (1ULL << 63)) == 0)
|
||
|
{
|
||
|
/* Check if it fits into an int64 */
|
||
|
if (x < (double)_I64_MAX)
|
||
|
{
|
||
|
/* Just cast to int64, which will truncate towards 0,
|
||
|
which is what we want here.*/
|
||
|
return (double)(long long)x;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
/* The exponent is larger than the fraction bits.
|
||
|
This means the number is already an integer. */
|
||
|
return x;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
/* Check if it fits into an int64 */
|
||
|
if (x > (double)_I64_MIN)
|
||
|
{
|
||
|
/* Check if it is -0 */
|
||
|
if (x == -0.)
|
||
|
{
|
||
|
return -0.;
|
||
|
}
|
||
|
|
||
|
/* Cast to int64 to truncate towards 0. If this matches the
|
||
|
input, return it as is, otherwise subtract 1 */
|
||
|
double y = (double)(long long)x;
|
||
|
return (x == y) ? y : y - 1;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
/* The exponent is larger than the fraction bits.
|
||
|
This means the number is already an integer. */
|
||
|
return x;
|
||
|
}
|
||
|
}
|
||
|
}
|