9#ifndef FORTRAN_DECIMAL_BINARY_FLOATING_POINT_H_
10#define FORTRAN_DECIMAL_BINARY_FLOATING_POINT_H_
15#include "flang/Common/api-attrs.h"
16#include "flang/Common/real.h"
17#include "flang/Common/uint128.h"
23namespace Fortran::decimal {
36 static constexpr int binaryPrecision{BINARY_PRECISION};
37 static constexpr int bits{realChars.bits};
38 static constexpr int isImplicitMSB{realChars.isImplicitMSB};
39 static constexpr int significandBits{realChars.significandBits};
40 static constexpr int exponentBits{realChars.exponentBits};
41 static constexpr int exponentBias{realChars.exponentBias};
42 static constexpr int maxExponent{realChars.maxExponent};
43 static constexpr int decimalPrecision{realChars.decimalPrecision};
44 static constexpr int decimalRange{realChars.decimalRange};
45 static constexpr int maxDecimalConversionDigits{
46 realChars.maxDecimalConversionDigits};
48 using RawType = common::HostUnsignedIntType<bits>;
49 static_assert(CHAR_BIT *
sizeof(RawType) >= bits);
50 RT_OFFLOAD_VAR_GROUP_BEGIN
51 static constexpr RawType significandMask{(RawType{1} << significandBits) - 1};
54 RT_OFFLOAD_VAR_GROUP_END
66 RT_API_ATTRS RawType raw()
const {
return raw_; }
70 static_assert(
sizeof raw_ <=
sizeof x);
71 std::memcpy(
reinterpret_cast<void *
>(&raw_),
72 reinterpret_cast<const void *
>(&x),
sizeof raw_);
75 constexpr RT_API_ATTRS
int BiasedExponent()
const {
76 return static_cast<int>(
77 (raw_ >> significandBits) & ((1 << exponentBits) - 1));
79 constexpr RT_API_ATTRS
int UnbiasedExponent()
const {
80 int biased{BiasedExponent()};
81 return biased - exponentBias + (biased == 0);
83 constexpr RT_API_ATTRS RawType Significand()
const {
84 return raw_ & significandMask;
86 constexpr RT_API_ATTRS RawType Fraction()
const {
87 RawType sig{Significand()};
88 if (isImplicitMSB && BiasedExponent() > 0) {
89 sig |= RawType{1} << significandBits;
94 constexpr RT_API_ATTRS
bool IsZero()
const {
95 return (raw_ & ((RawType{1} << (bits - 1)) - 1)) == 0;
97 constexpr RT_API_ATTRS
bool IsNaN()
const {
98 auto expo{BiasedExponent()};
99 auto sig{Significand()};
100 if constexpr (bits == 80) {
101 if (expo == maxExponent) {
102 return sig != (significandMask >> 1) + 1;
104 return expo != 0 && !(sig & (RawType{1} << (significandBits - 1)));
108 return expo == maxExponent && sig != 0;
111 constexpr RT_API_ATTRS
bool IsInfinite()
const {
112 if constexpr (bits == 80) {
113 return BiasedExponent() == maxExponent &&
114 Significand() == ((significandMask >> 1) + 1);
116 return BiasedExponent() == maxExponent && Significand() == 0;
119 constexpr RT_API_ATTRS
bool IsMaximalFiniteMagnitude()
const {
120 return BiasedExponent() == maxExponent - 1 &&
121 Significand() == significandMask;
123 constexpr RT_API_ATTRS
bool IsNegative()
const {
124 return ((raw_ >> (bits - 1)) & 1) != 0;
127 constexpr RT_API_ATTRS
void Negate() { raw_ ^= RawType{1} << (bits - 1); }
130 constexpr RT_API_ATTRS
void Previous() {
135 constexpr RT_API_ATTRS
void Next() {
143 RawType result{RawType{maxExponent} << significandBits};
145 result |= RawType{1} << (bits - 1);
151 constexpr RT_API_ATTRS
bool RoundToBits(
152 int keepBits,
enum FortranRounding mode) {
153 if (IsNaN() || IsInfinite() || keepBits >= binaryPrecision) {
156 int lostBits{keepBits < binaryPrecision ? binaryPrecision - keepBits : 0};
157 RawType lostMask{
static_cast<RawType
>((RawType{1} << lostBits) - 1)};
158 if (RawType lost{
static_cast<RawType
>(raw_ & lostMask)}; lost != 0) {
159 bool increase{
false};
162 if (lost >> (lostBits - 1) != 0) {
163 if ((lost & (lostMask >> 1)) != 0) {
166 increase = ((raw_ >> lostBits) & 1) != 0;
171 increase = !IsNegative();
174 increase = IsNegative();
178 case RoundCompatible:
179 increase = lost >> (lostBits - 1) != 0;
193 constexpr RT_API_ATTRS
void RemoveExplicitMSB() {
194 if constexpr (!isImplicitMSB) {
195 raw_ = (raw_ & (significandMask >> 1)) | ((raw_ & ~significandMask) >> 1);
198 constexpr RT_API_ATTRS
void InsertExplicitMSB() {
199 if constexpr (!isImplicitMSB) {
200 constexpr RawType mask{significandMask >> 1};
201 raw_ = (raw_ & mask) | ((raw_ & ~mask) << 1);
202 if (BiasedExponent() > 0) {
203 raw_ |= RawType{1} << (significandBits - 1);
Definition: binary-floating-point.h:33