mirror of
https://gcc.gnu.org/git/gcc.git
synced 2024-11-23 19:03:59 +08:00
45ab93d9af
I've tried to build stage3 with -Wleading-whitespace=blanks -Wtrailing-whitespace=blank -Wno-error=leading-whitespace=blanks -Wno-error=trailing-whitespace=blank added to STRICT_WARN and that expectably resulted in about 2744 unique trailing whitespace warnings and 124837 leading whitespace warnings when excluding *.md files (which obviously is in big part a generator issue). Others from that are generator related, I think those need to be solved later. The following patch just fixes up the easy case (trailing whitespace), which could be easily automated: for i in `find . -name \*.h -o -name \*.cc -o -name \*.c | xargs grep -l '[ ]$' | grep -v testsuite/`; do sed -i -e 's/[ ]*$//' $i; done I've excluded files which I knew are obviously generated or go FE. Is there anything else we'd want to avoid the changes? Due to patch size, I've split it between gcc/ part and rest (include/, libiberty/, libgcc/, libcpp/, libstdc++-v3/; this part). 2024-10-24 Jakub Jelinek <jakub@redhat.com> include/ * dyn-string.h: Remove trailing whitespace. * libiberty.h: Likewise. * xregex.h: Likewise. * splay-tree.h: Likewise. * partition.h: Likewise. * plugin-api.h: Likewise. * demangle.h: Likewise. * vtv-change-permission.h: Likewise. * fibheap.h: Likewise. * hsa_ext_image.h: Likewise. * hashtab.h: Likewise. * libcollector.h: Likewise. * sort.h: Likewise. * symcat.h: Likewise. * hsa_ext_amd.h: Likewise. libcpp/ * directives.cc: Remove trailing whitespace. * mkdeps.cc: Likewise. * line-map.cc: Likewise. * internal.h: Likewise. * files.cc: Likewise. * init.cc: Likewise. * makeucnid.cc: Likewise. * system.h: Likewise. * include/line-map.h: Likewise. * include/symtab.h: Likewise. * include/cpplib.h: Likewise. * expr.cc: Likewise. * charset.cc: Likewise. * macro.cc: Likewise. * errors.cc: Likewise. * lex.cc: Likewise. * traditional.cc: Likewise. libgcc/ * crtstuff.c: Remove trailing whitespace. * libgcov.h: Likewise. * config/alpha/crtfastmath.c: Likewise. * config/alpha/vms-gcc_shell_handler.c: Likewise. * config/alpha/vms-unwind.h: Likewise. * config/pa/linux-atomic.c: Likewise. * config/pa/linux-unwind.h: Likewise. * config/pa/quadlib.c: Likewise. * config/pa/fptr.c: Likewise. * config/s390/32/_fixsfdi.c: Likewise. * config/s390/32/_fixunssfdi.c: Likewise. * config/s390/32/_fixunsdfdi.c: Likewise. * config/c6x/pr-support.c: Likewise. * config/lm32/_udivsi3.c: Likewise. * config/lm32/libgcc_lm32.h: Likewise. * config/lm32/_udivmodsi4.c: Likewise. * config/lm32/_mulsi3.c: Likewise. * config/lm32/_modsi3.c: Likewise. * config/lm32/_umodsi3.c: Likewise. * config/lm32/_divsi3.c: Likewise. * config/darwin-crt3.c: Likewise. * config/msp430/mpy.c: Likewise. * config/ia64/tf-signs.c: Likewise. * config/ia64/fde-vms.c: Likewise. * config/ia64/unwind-ia64.c: Likewise. * config/ia64/vms-unwind.h: Likewise. * config/ia64/sfp-exceptions.c: Likewise. * config/ia64/quadlib.c: Likewise. * config/ia64/unwind-ia64.h: Likewise. * config/rl78/vregs.h: Likewise. * config/arm/bpabi.c: Likewise. * config/arm/unwind-arm.c: Likewise. * config/arm/pr-support.c: Likewise. * config/arm/linux-atomic.c: Likewise. * config/arm/bpabi-lib.h: Likewise. * config/frv/frvend.c: Likewise. * config/frv/cmovw.c: Likewise. * config/frv/frvbegin.c: Likewise. * config/frv/cmovd.c: Likewise. * config/frv/cmovh.c: Likewise. * config/aarch64/cpuinfo.c: Likewise. * config/i386/crtfastmath.c: Likewise. * config/i386/cygming-crtend.c: Likewise. * config/i386/32/tf-signs.c: Likewise. * config/i386/crtprec.c: Likewise. * config/i386/sfp-exceptions.c: Likewise. * config/i386/w32-unwind.h: Likewise. * config/m32r/initfini.c: Likewise. * config/sparc/crtfastmath.c: Likewise. * config/gcn/amdgcn_veclib.h: Likewise. * config/nios2/linux-atomic.c: Likewise. * config/nios2/linux-unwind.h: Likewise. * config/nios2/lib2-mul.c: Likewise. * config/nios2/lib2-nios2.h: Likewise. * config/xtensa/unwind-dw2-xtensa.c: Likewise. * config/rs6000/darwin-fallback.c: Likewise. * config/rs6000/ibm-ldouble.c: Likewise. * config/rs6000/sfp-machine.h: Likewise. * config/rs6000/darwin-asm.h: Likewise. * config/rs6000/darwin-crt2.c: Likewise. * config/rs6000/aix-unwind.h: Likewise. * config/rs6000/sfp-exceptions.c: Likewise. * config/gthr-vxworks.c: Likewise. * config/riscv/atomic.c: Likewise. * config/visium/memcpy.c: Likewise. * config/darwin-crt-tm.c: Likewise. * config/stormy16/lib2funcs.c: Likewise. * config/arc/ieee-754/divtab-arc-sf.c: Likewise. * config/arc/ieee-754/divtab-arc-df.c: Likewise. * config/arc/initfini.c: Likewise. * config/sol2/gmon.c: Likewise. * config/microblaze/divsi3_table.c: Likewise. * config/m68k/fpgnulib.c: Likewise. * libgcov-driver.c: Likewise. * unwind-dw2.c: Likewise. * fp-bit.c: Likewise. * dfp-bit.h: Likewise. * dfp-bit.c: Likewise. * libgcov-driver-system.c: Likewise. libgcc/config/libbid/ * _le_td.c: Remove trailing whitespace. * bid128_compare.c: Likewise. * bid_div_macros.h: Likewise. * bid64_to_bid128.c: Likewise. * bid64_to_uint32.c: Likewise. * bid128_to_uint64.c: Likewise. * bid64_div.c: Likewise. * bid128_round_integral.c: Likewise. * bid_binarydecimal.c: Likewise. * bid128_string.c: Likewise. * bid_flag_operations.c: Likewise. * bid128_to_int64.c: Likewise. * _mul_sd.c: Likewise. * bid64_mul.c: Likewise. * bid128_noncomp.c: Likewise. * _gt_dd.c: Likewise. * bid64_add.c: Likewise. * bid64_string.c: Likewise. * bid_from_int.c: Likewise. * bid128.c: Likewise. * _ge_dd.c: Likewise. * _ne_sd.c: Likewise. * _dd_to_td.c: Likewise. * _unord_sd.c: Likewise. * bid64_to_uint64.c: Likewise. * _gt_sd.c: Likewise. * _sd_to_td.c: Likewise. * _addsub_td.c: Likewise. * _ne_td.c: Likewise. * bid_dpd.c: Likewise. * bid128_add.c: Likewise. * bid128_next.c: Likewise. * _lt_sd.c: Likewise. * bid64_next.c: Likewise. * bid128_mul.c: Likewise. * _lt_dd.c: Likewise. * _ge_td.c: Likewise. * _unord_dd.c: Likewise. * bid64_sqrt.c: Likewise. * bid_sqrt_macros.h: Likewise. * bid64_fma.c: Likewise. * _sd_to_dd.c: Likewise. * bid_conf.h: Likewise. * bid64_noncomp.c: Likewise. * bid_gcc_intrinsics.h: Likewise. * _gt_td.c: Likewise. * _ge_sd.c: Likewise. * bid128_minmax.c: Likewise. * bid128_quantize.c: Likewise. * bid32_to_bid64.c: Likewise. * bid_round.c: Likewise. * _td_to_sd.c: Likewise. * bid_inline_add.h: Likewise. * bid128_fma.c: Likewise. * _eq_td.c: Likewise. * bid32_to_bid128.c: Likewise. * bid64_rem.c: Likewise. * bid128_2_str_tables.c: Likewise. * _mul_dd.c: Likewise. * _dd_to_sd.c: Likewise. * bid128_div.c: Likewise. * _lt_td.c: Likewise. * bid64_compare.c: Likewise. * bid64_to_int32.c: Likewise. * _unord_td.c: Likewise. * bid128_rem.c: Likewise. * bid_internal.h: Likewise. * bid64_to_int64.c: Likewise. * _eq_dd.c: Likewise. * _td_to_dd.c: Likewise. * bid128_to_int32.c: Likewise. * bid128_to_uint32.c: Likewise. * _ne_dd.c: Likewise. * bid64_quantize.c: Likewise. * _le_dd.c: Likewise. * bid64_round_integral.c: Likewise. * _le_sd.c: Likewise. * bid64_minmax.c: Likewise. libgcc/config/avr/libf7/ * f7-renames.h: Remove trailing whitespace. libstdc++-v3/ * include/debug/debug.h: Remove trailing whitespace. * include/parallel/base.h: Likewise. * include/parallel/types.h: Likewise. * include/parallel/settings.h: Likewise. * include/parallel/multiseq_selection.h: Likewise. * include/parallel/partition.h: Likewise. * include/parallel/random_number.h: Likewise. * include/parallel/find_selectors.h: Likewise. * include/parallel/partial_sum.h: Likewise. * include/parallel/list_partition.h: Likewise. * include/parallel/search.h: Likewise. * include/parallel/algorithmfwd.h: Likewise. * include/parallel/random_shuffle.h: Likewise. * include/parallel/multiway_mergesort.h: Likewise. * include/parallel/sort.h: Likewise. * include/parallel/algobase.h: Likewise. * include/parallel/numericfwd.h: Likewise. * include/parallel/multiway_merge.h: Likewise. * include/parallel/losertree.h: Likewise. * include/bits/basic_ios.h: Likewise. * include/bits/stringfwd.h: Likewise. * include/bits/ostream_insert.h: Likewise. * include/bits/stl_heap.h: Likewise. * include/bits/unordered_map.h: Likewise. * include/bits/hashtable_policy.h: Likewise. * include/bits/stl_iterator_base_funcs.h: Likewise. * include/bits/valarray_before.h: Likewise. * include/bits/regex.h: Likewise. * include/bits/postypes.h: Likewise. * include/bits/stl_iterator.h: Likewise. * include/bits/localefwd.h: Likewise. * include/bits/stl_algo.h: Likewise. * include/bits/ios_base.h: Likewise. * include/bits/stl_function.h: Likewise. * include/bits/basic_string.h: Likewise. * include/bits/hashtable.h: Likewise. * include/bits/valarray_after.h: Likewise. * include/bits/char_traits.h: Likewise. * include/bits/gslice.h: Likewise. * include/bits/locale_facets_nonio.h: Likewise. * include/bits/mask_array.h: Likewise. * include/bits/specfun.h: Likewise. * include/bits/random.h: Likewise. * include/bits/slice_array.h: Likewise. * include/bits/valarray_array.h: Likewise. * include/tr1/float.h: Likewise. * include/tr1/functional_hash.h: Likewise. * include/tr1/math.h: Likewise. * include/tr1/hashtable_policy.h: Likewise. * include/tr1/stdio.h: Likewise. * include/tr1/complex.h: Likewise. * include/tr1/stdbool.h: Likewise. * include/tr1/stdarg.h: Likewise. * include/tr1/inttypes.h: Likewise. * include/tr1/fenv.h: Likewise. * include/tr1/stdlib.h: Likewise. * include/tr1/wchar.h: Likewise. * include/tr1/tgmath.h: Likewise. * include/tr1/limits.h: Likewise. * include/tr1/wctype.h: Likewise. * include/tr1/stdint.h: Likewise. * include/tr1/ctype.h: Likewise. * include/tr1/random.h: Likewise. * include/tr1/shared_ptr.h: Likewise. * include/ext/mt_allocator.h: Likewise. * include/ext/sso_string_base.h: Likewise. * include/ext/debug_allocator.h: Likewise. * include/ext/vstring_fwd.h: Likewise. * include/ext/pointer.h: Likewise. * include/ext/pod_char_traits.h: Likewise. * include/ext/malloc_allocator.h: Likewise. * include/ext/vstring.h: Likewise. * include/ext/bitmap_allocator.h: Likewise. * include/ext/pool_allocator.h: Likewise. * include/ext/type_traits.h: Likewise. * include/ext/ropeimpl.h: Likewise. * include/ext/codecvt_specializations.h: Likewise. * include/ext/throw_allocator.h: Likewise. * include/ext/extptr_allocator.h: Likewise. * include/ext/atomicity.h: Likewise. * include/ext/concurrence.h: Likewise. * include/c_compatibility/wchar.h: Likewise. * include/c_compatibility/stdint.h: Likewise. * include/backward/hash_fun.h: Likewise. * include/backward/binders.h: Likewise. * include/backward/hashtable.h: Likewise. * include/backward/auto_ptr.h: Likewise. * libsupc++/eh_arm.cc: Likewise. * libsupc++/unwind-cxx.h: Likewise. * libsupc++/si_class_type_info.cc: Likewise. * libsupc++/vec.cc: Likewise. * libsupc++/class_type_info.cc: Likewise. * libsupc++/vmi_class_type_info.cc: Likewise. * libsupc++/guard_error.cc: Likewise. * libsupc++/bad_typeid.cc: Likewise. * libsupc++/eh_personality.cc: Likewise. * libsupc++/atexit_arm.cc: Likewise. * libsupc++/pmem_type_info.cc: Likewise. * libsupc++/vterminate.cc: Likewise. * libsupc++/eh_terminate.cc: Likewise. * libsupc++/bad_cast.cc: Likewise. * libsupc++/exception_ptr.h: Likewise. * libsupc++/eh_throw.cc: Likewise. * libsupc++/bad_alloc.cc: Likewise. * libsupc++/nested_exception.cc: Likewise. * libsupc++/pointer_type_info.cc: Likewise. * libsupc++/pbase_type_info.cc: Likewise. * libsupc++/bad_array_new.cc: Likewise. * libsupc++/pure.cc: Likewise. * libsupc++/eh_exception.cc: Likewise. * libsupc++/bad_array_length.cc: Likewise. * libsupc++/cxxabi.h: Likewise. * libsupc++/guard.cc: Likewise. * libsupc++/eh_catch.cc: Likewise. * libsupc++/cxxabi_forced.h: Likewise. * libsupc++/tinfo.h: Likewise.
690 lines
20 KiB
C
690 lines
20 KiB
C
/* This is a software decimal floating point library.
|
||
Copyright (C) 2005-2024 Free Software Foundation, Inc.
|
||
|
||
This file is part of GCC.
|
||
|
||
GCC is free software; you can redistribute it and/or modify it under
|
||
the terms of the GNU General Public License as published by the Free
|
||
Software Foundation; either version 3, or (at your option) any later
|
||
version.
|
||
|
||
GCC 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 General Public License
|
||
for more details.
|
||
|
||
Under Section 7 of GPL version 3, you are granted additional
|
||
permissions described in the GCC Runtime Library Exception, version
|
||
3.1, as published by the Free Software Foundation.
|
||
|
||
You should have received a copy of the GNU General Public License and
|
||
a copy of the GCC Runtime Library Exception along with this program;
|
||
see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
|
||
<http://www.gnu.org/licenses/>. */
|
||
|
||
/* This implements IEEE 754 decimal floating point arithmetic, but
|
||
does not provide a mechanism for setting the rounding mode, or for
|
||
generating or handling exceptions. Conversions between decimal
|
||
floating point types and other types depend on C library functions.
|
||
|
||
Contributed by Ben Elliston <bje@au.ibm.com>. */
|
||
|
||
#include <stdio.h>
|
||
#include <stdlib.h>
|
||
/* FIXME: compile with -std=gnu99 to get these from stdlib.h */
|
||
extern float strtof (const char *, char **);
|
||
extern long double strtold (const char *, char **);
|
||
#include <string.h>
|
||
#include <limits.h>
|
||
|
||
#include "dfp-bit.h"
|
||
|
||
/* Forward declarations. */
|
||
#if WIDTH == 32 || WIDTH_TO == 32
|
||
void __host_to_ieee_32 (_Decimal32 in, decimal32 *out);
|
||
void __ieee_to_host_32 (decimal32 in, _Decimal32 *out);
|
||
#endif
|
||
#if WIDTH == 64 || WIDTH_TO == 64
|
||
void __host_to_ieee_64 (_Decimal64 in, decimal64 *out);
|
||
void __ieee_to_host_64 (decimal64 in, _Decimal64 *out);
|
||
#endif
|
||
#if WIDTH == 128 || WIDTH_TO == 128
|
||
void __host_to_ieee_128 (_Decimal128 in, decimal128 *out);
|
||
void __ieee_to_host_128 (decimal128 in, _Decimal128 *out);
|
||
#endif
|
||
|
||
/* A pointer to a binary decFloat operation. */
|
||
typedef decFloat* (*dfp_binary_func)
|
||
(decFloat *, const decFloat *, const decFloat *, decContext *);
|
||
|
||
/* Binary operations. */
|
||
|
||
/* Use a decFloat (decDouble or decQuad) function to perform a DFP
|
||
binary operation. */
|
||
static inline decFloat
|
||
dfp_binary_op (dfp_binary_func op, decFloat arg_a, decFloat arg_b)
|
||
{
|
||
decFloat result;
|
||
decContext context;
|
||
|
||
decContextDefault (&context, CONTEXT_INIT);
|
||
DFP_INIT_ROUNDMODE (context.round);
|
||
|
||
/* Perform the operation. */
|
||
op (&result, &arg_a, &arg_b, &context);
|
||
|
||
if (DFP_EXCEPTIONS_ENABLED && context.status != 0)
|
||
{
|
||
/* decNumber exception flags we care about here. */
|
||
int ieee_flags;
|
||
int dec_flags = DEC_IEEE_854_Division_by_zero | DEC_IEEE_854_Inexact
|
||
| DEC_IEEE_854_Invalid_operation | DEC_IEEE_854_Overflow
|
||
| DEC_IEEE_854_Underflow;
|
||
dec_flags &= context.status;
|
||
ieee_flags = DFP_IEEE_FLAGS (dec_flags);
|
||
if (ieee_flags != 0)
|
||
DFP_HANDLE_EXCEPTIONS (ieee_flags);
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
#if WIDTH == 32
|
||
/* The decNumber package doesn't provide arithmetic for decSingle (32 bits);
|
||
convert to decDouble, use the operation for that, and convert back. */
|
||
static inline _Decimal32
|
||
d32_binary_op (dfp_binary_func op, _Decimal32 arg_a, _Decimal32 arg_b)
|
||
{
|
||
union { _Decimal32 c; decSingle f; } a32, b32, res32;
|
||
decDouble a, b, res;
|
||
decContext context;
|
||
|
||
/* Widen the operands and perform the operation. */
|
||
a32.c = arg_a;
|
||
b32.c = arg_b;
|
||
decSingleToWider (&a32.f, &a);
|
||
decSingleToWider (&b32.f, &b);
|
||
res = dfp_binary_op (op, a, b);
|
||
|
||
/* Narrow the result, which might result in an underflow or overflow. */
|
||
decContextDefault (&context, CONTEXT_INIT);
|
||
DFP_INIT_ROUNDMODE (context.round);
|
||
decSingleFromWider (&res32.f, &res, &context);
|
||
if (DFP_EXCEPTIONS_ENABLED && context.status != 0)
|
||
{
|
||
/* decNumber exception flags we care about here. */
|
||
int ieee_flags;
|
||
int dec_flags = DEC_IEEE_854_Inexact | DEC_IEEE_854_Overflow
|
||
| DEC_IEEE_854_Underflow;
|
||
dec_flags &= context.status;
|
||
ieee_flags = DFP_IEEE_FLAGS (dec_flags);
|
||
if (ieee_flags != 0)
|
||
DFP_HANDLE_EXCEPTIONS (ieee_flags);
|
||
}
|
||
|
||
return res32.c;
|
||
}
|
||
#else
|
||
/* decFloat operations are supported for decDouble (64 bits) and
|
||
decQuad (128 bits). The bit patterns for the types are the same. */
|
||
static inline DFP_C_TYPE
|
||
dnn_binary_op (dfp_binary_func op, DFP_C_TYPE arg_a, DFP_C_TYPE arg_b)
|
||
{
|
||
union { DFP_C_TYPE c; decFloat f; } a, b, result;
|
||
|
||
a.c = arg_a;
|
||
b.c = arg_b;
|
||
result.f = dfp_binary_op (op, a.f, b.f);
|
||
return result.c;
|
||
}
|
||
#endif
|
||
|
||
/* Comparison operations. */
|
||
|
||
/* Use a decFloat (decDouble or decQuad) function to perform a DFP
|
||
comparison. */
|
||
static inline CMPtype
|
||
dfp_compare_op (dfp_binary_func op, decFloat arg_a, decFloat arg_b)
|
||
{
|
||
decContext context;
|
||
decFloat res;
|
||
int result;
|
||
|
||
decContextDefault (&context, CONTEXT_INIT);
|
||
DFP_INIT_ROUNDMODE (context.round);
|
||
|
||
/* Perform the comparison. */
|
||
op (&res, &arg_a, &arg_b, &context);
|
||
|
||
if (DEC_FLOAT_IS_SIGNED (&res))
|
||
result = -1;
|
||
else if (DEC_FLOAT_IS_ZERO (&res))
|
||
result = 0;
|
||
else if (DEC_FLOAT_IS_NAN (&res))
|
||
result = -2;
|
||
else
|
||
result = 1;
|
||
|
||
return (CMPtype) result;
|
||
}
|
||
|
||
#if WIDTH == 32
|
||
/* The decNumber package doesn't provide comparisons for decSingle (32 bits);
|
||
convert to decDouble, use the operation for that, and convert back. */
|
||
static inline CMPtype
|
||
d32_compare_op (dfp_binary_func op, _Decimal32 arg_a, _Decimal32 arg_b)
|
||
{
|
||
union { _Decimal32 c; decSingle f; } a32, b32;
|
||
decDouble a, b;
|
||
|
||
a32.c = arg_a;
|
||
b32.c = arg_b;
|
||
decSingleToWider (&a32.f, &a);
|
||
decSingleToWider (&b32.f, &b);
|
||
return dfp_compare_op (op, a, b);
|
||
}
|
||
#else
|
||
/* decFloat comparisons are supported for decDouble (64 bits) and
|
||
decQuad (128 bits). The bit patterns for the types are the same. */
|
||
static inline CMPtype
|
||
dnn_compare_op (dfp_binary_func op, DFP_C_TYPE arg_a, DFP_C_TYPE arg_b)
|
||
{
|
||
union { DFP_C_TYPE c; decFloat f; } a, b;
|
||
|
||
a.c = arg_a;
|
||
b.c = arg_b;
|
||
return dfp_compare_op (op, a.f, b.f);
|
||
}
|
||
#endif
|
||
|
||
#if defined(L_conv_sd)
|
||
void
|
||
__host_to_ieee_32 (_Decimal32 in, decimal32 *out)
|
||
{
|
||
memcpy (out, &in, 4);
|
||
}
|
||
|
||
void
|
||
__ieee_to_host_32 (decimal32 in, _Decimal32 *out)
|
||
{
|
||
memcpy (out, &in, 4);
|
||
}
|
||
#endif /* L_conv_sd */
|
||
|
||
#if defined(L_conv_dd)
|
||
void
|
||
__host_to_ieee_64 (_Decimal64 in, decimal64 *out)
|
||
{
|
||
memcpy (out, &in, 8);
|
||
}
|
||
|
||
void
|
||
__ieee_to_host_64 (decimal64 in, _Decimal64 *out)
|
||
{
|
||
memcpy (out, &in, 8);
|
||
}
|
||
#endif /* L_conv_dd */
|
||
|
||
#if defined(L_conv_td)
|
||
void
|
||
__host_to_ieee_128 (_Decimal128 in, decimal128 *out)
|
||
{
|
||
memcpy (out, &in, 16);
|
||
}
|
||
|
||
void
|
||
__ieee_to_host_128 (decimal128 in, _Decimal128 *out)
|
||
{
|
||
memcpy (out, &in, 16);
|
||
}
|
||
#endif /* L_conv_td */
|
||
|
||
#if defined(L_addsub_sd) || defined(L_addsub_dd) || defined(L_addsub_td)
|
||
DFP_C_TYPE
|
||
DFP_ADD (DFP_C_TYPE arg_a, DFP_C_TYPE arg_b)
|
||
{
|
||
return DFP_BINARY_OP (DEC_FLOAT_ADD, arg_a, arg_b);
|
||
}
|
||
|
||
DFP_C_TYPE
|
||
DFP_SUB (DFP_C_TYPE arg_a, DFP_C_TYPE arg_b)
|
||
{
|
||
return DFP_BINARY_OP (DEC_FLOAT_SUBTRACT, arg_a, arg_b);
|
||
}
|
||
#endif /* L_addsub */
|
||
|
||
#if defined(L_mul_sd) || defined(L_mul_dd) || defined(L_mul_td)
|
||
DFP_C_TYPE
|
||
DFP_MULTIPLY (DFP_C_TYPE arg_a, DFP_C_TYPE arg_b)
|
||
{
|
||
return DFP_BINARY_OP (DEC_FLOAT_MULTIPLY, arg_a, arg_b);
|
||
}
|
||
#endif /* L_mul */
|
||
|
||
#if defined(L_div_sd) || defined(L_div_dd) || defined(L_div_td)
|
||
DFP_C_TYPE
|
||
DFP_DIVIDE (DFP_C_TYPE arg_a, DFP_C_TYPE arg_b)
|
||
{
|
||
return DFP_BINARY_OP (DEC_FLOAT_DIVIDE, arg_a, arg_b);
|
||
}
|
||
#endif /* L_div */
|
||
|
||
#if defined (L_eq_sd) || defined (L_eq_dd) || defined (L_eq_td)
|
||
CMPtype
|
||
DFP_EQ (DFP_C_TYPE arg_a, DFP_C_TYPE arg_b)
|
||
{
|
||
CMPtype stat;
|
||
stat = DFP_COMPARE_OP (DEC_FLOAT_COMPARE, arg_a, arg_b);
|
||
/* For EQ return zero for true, nonzero for false. */
|
||
return stat != 0;
|
||
}
|
||
#endif /* L_eq */
|
||
|
||
#if defined (L_ne_sd) || defined (L_ne_dd) || defined (L_ne_td)
|
||
CMPtype
|
||
DFP_NE (DFP_C_TYPE arg_a, DFP_C_TYPE arg_b)
|
||
{
|
||
int stat;
|
||
stat = DFP_COMPARE_OP (DEC_FLOAT_COMPARE, arg_a, arg_b);
|
||
/* For NE return zero for true, nonzero for false. */
|
||
if (__builtin_expect (stat == -2, 0)) /* An operand is NaN. */
|
||
return 1;
|
||
return stat != 0;
|
||
}
|
||
#endif /* L_ne */
|
||
|
||
#if defined (L_lt_sd) || defined (L_lt_dd) || defined (L_lt_td)
|
||
CMPtype
|
||
DFP_LT (DFP_C_TYPE arg_a, DFP_C_TYPE arg_b)
|
||
{
|
||
int stat;
|
||
stat = DFP_COMPARE_OP (DEC_FLOAT_COMPARE, arg_a, arg_b);
|
||
/* For LT return -1 (<0) for true, 1 for false. */
|
||
return (stat == -1) ? -1 : 1;
|
||
}
|
||
#endif /* L_lt */
|
||
|
||
#if defined (L_gt_sd) || defined (L_gt_dd) || defined (L_gt_td)
|
||
CMPtype
|
||
DFP_GT (DFP_C_TYPE arg_a, DFP_C_TYPE arg_b)
|
||
{
|
||
int stat;
|
||
stat = DFP_COMPARE_OP (DEC_FLOAT_COMPARE, arg_a, arg_b);
|
||
/* For GT return 1 (>0) for true, -1 for false. */
|
||
return (stat == 1) ? 1 : -1;
|
||
}
|
||
#endif
|
||
|
||
#if defined (L_le_sd) || defined (L_le_dd) || defined (L_le_td)
|
||
CMPtype
|
||
DFP_LE (DFP_C_TYPE arg_a, DFP_C_TYPE arg_b)
|
||
{
|
||
int stat;
|
||
stat = DFP_COMPARE_OP (DEC_FLOAT_COMPARE, arg_a, arg_b);
|
||
/* For LE return 0 (<= 0) for true, 1 for false. */
|
||
if (__builtin_expect (stat == -2, 0)) /* An operand is NaN. */
|
||
return 1;
|
||
return stat == 1;
|
||
}
|
||
#endif /* L_le */
|
||
|
||
#if defined (L_ge_sd) || defined (L_ge_dd) || defined (L_ge_td)
|
||
CMPtype
|
||
DFP_GE (DFP_C_TYPE arg_a, DFP_C_TYPE arg_b)
|
||
{
|
||
int stat;
|
||
stat = DFP_COMPARE_OP (DEC_FLOAT_COMPARE, arg_a, arg_b);
|
||
/* For GE return 1 (>=0) for true, -1 for false. */
|
||
if (__builtin_expect (stat == -2, 0)) /* An operand is NaN. */
|
||
return -1;
|
||
return (stat != -1) ? 1 : -1;
|
||
}
|
||
#endif /* L_ge */
|
||
|
||
#define BUFMAX 128
|
||
|
||
/* Check for floating point exceptions that are relevant for conversions
|
||
between decimal float values and handle them. */
|
||
static inline void
|
||
dfp_conversion_exceptions (const int status)
|
||
{
|
||
/* decNumber exception flags we care about here. */
|
||
int ieee_flags;
|
||
int dec_flags = DEC_IEEE_854_Inexact | DEC_IEEE_854_Invalid_operation
|
||
| DEC_IEEE_854_Overflow;
|
||
dec_flags &= status;
|
||
ieee_flags = DFP_IEEE_FLAGS (dec_flags);
|
||
if (ieee_flags != 0)
|
||
DFP_HANDLE_EXCEPTIONS (ieee_flags);
|
||
}
|
||
|
||
#if defined (L_sd_to_dd)
|
||
/* Use decNumber to convert directly from _Decimal32 to _Decimal64. */
|
||
_Decimal64
|
||
DFP_TO_DFP (_Decimal32 f_from)
|
||
{
|
||
union { _Decimal32 c; decSingle f; } from;
|
||
union { _Decimal64 c; decDouble f; } to;
|
||
|
||
from.c = f_from;
|
||
to.f = *decSingleToWider (&from.f, &to.f);
|
||
return to.c;
|
||
}
|
||
#endif
|
||
|
||
#if defined (L_sd_to_td)
|
||
/* Use decNumber to convert directly from _Decimal32 to _Decimal128. */
|
||
_Decimal128
|
||
DFP_TO_DFP (_Decimal32 f_from)
|
||
{
|
||
union { _Decimal32 c; decSingle f; } from;
|
||
union { _Decimal128 c; decQuad f; } to;
|
||
decDouble temp;
|
||
|
||
from.c = f_from;
|
||
temp = *decSingleToWider (&from.f, &temp);
|
||
to.f = *decDoubleToWider (&temp, &to.f);
|
||
return to.c;
|
||
}
|
||
#endif
|
||
|
||
#if defined (L_dd_to_td)
|
||
/* Use decNumber to convert directly from _Decimal64 to _Decimal128. */
|
||
_Decimal128
|
||
DFP_TO_DFP (_Decimal64 f_from)
|
||
{
|
||
union { _Decimal64 c; decDouble f; } from;
|
||
union { _Decimal128 c; decQuad f; } to;
|
||
|
||
from.c = f_from;
|
||
to.f = *decDoubleToWider (&from.f, &to.f);
|
||
return to.c;
|
||
}
|
||
#endif
|
||
|
||
#if defined (L_dd_to_sd)
|
||
/* Use decNumber to convert directly from _Decimal64 to _Decimal32. */
|
||
_Decimal32
|
||
DFP_TO_DFP (_Decimal64 f_from)
|
||
{
|
||
union { _Decimal32 c; decSingle f; } to;
|
||
union { _Decimal64 c; decDouble f; } from;
|
||
decContext context;
|
||
|
||
decContextDefault (&context, CONTEXT_INIT);
|
||
DFP_INIT_ROUNDMODE (context.round);
|
||
from.c = f_from;
|
||
to.f = *decSingleFromWider (&to.f, &from.f, &context);
|
||
if (DFP_EXCEPTIONS_ENABLED && context.status != 0)
|
||
dfp_conversion_exceptions (context.status);
|
||
return to.c;
|
||
}
|
||
#endif
|
||
|
||
#if defined (L_td_to_sd)
|
||
/* Use decNumber to convert directly from _Decimal128 to _Decimal32. */
|
||
_Decimal32
|
||
DFP_TO_DFP (_Decimal128 f_from)
|
||
{
|
||
union { _Decimal32 c; decSingle f; } to;
|
||
union { _Decimal128 c; decQuad f; } from;
|
||
decDouble temp;
|
||
decContext context;
|
||
|
||
decContextDefault (&context, CONTEXT_INIT);
|
||
DFP_INIT_ROUNDMODE (context.round);
|
||
from.c = f_from;
|
||
temp = *decDoubleFromWider (&temp, &from.f, &context);
|
||
to.f = *decSingleFromWider (&to.f, &temp, &context);
|
||
if (DFP_EXCEPTIONS_ENABLED && context.status != 0)
|
||
dfp_conversion_exceptions (context.status);
|
||
return to.c;
|
||
}
|
||
#endif
|
||
|
||
#if defined (L_td_to_dd)
|
||
/* Use decNumber to convert directly from _Decimal128 to _Decimal64. */
|
||
_Decimal64
|
||
DFP_TO_DFP (_Decimal128 f_from)
|
||
{
|
||
union { _Decimal64 c; decDouble f; } to;
|
||
union { _Decimal128 c; decQuad f; } from;
|
||
decContext context;
|
||
|
||
decContextDefault (&context, CONTEXT_INIT);
|
||
DFP_INIT_ROUNDMODE (context.round);
|
||
from.c = f_from;
|
||
to.f = *decDoubleFromWider (&to.f, &from.f, &context);
|
||
if (DFP_EXCEPTIONS_ENABLED && context.status != 0)
|
||
dfp_conversion_exceptions (context.status);
|
||
return to.c;
|
||
}
|
||
#endif
|
||
|
||
#if defined (L_dd_to_si) || defined (L_td_to_si) \
|
||
|| defined (L_dd_to_usi) || defined (L_td_to_usi)
|
||
/* Use decNumber to convert directly from decimal float to integer types. */
|
||
INT_TYPE
|
||
DFP_TO_INT (DFP_C_TYPE x)
|
||
{
|
||
union { DFP_C_TYPE c; decFloat f; } u;
|
||
decContext context;
|
||
INT_TYPE i;
|
||
|
||
decContextDefault (&context, DEC_INIT_DECIMAL128);
|
||
context.round = DEC_ROUND_DOWN;
|
||
u.c = x;
|
||
i = DEC_FLOAT_TO_INT (&u.f, &context, context.round);
|
||
if (DFP_EXCEPTIONS_ENABLED && context.status != 0)
|
||
dfp_conversion_exceptions (context.status);
|
||
return i;
|
||
}
|
||
#endif
|
||
|
||
#if defined (L_sd_to_si) || (L_sd_to_usi)
|
||
/* Use decNumber to convert directly from decimal float to integer types. */
|
||
INT_TYPE
|
||
DFP_TO_INT (_Decimal32 x)
|
||
{
|
||
union { _Decimal32 c; decSingle f; } u32;
|
||
decDouble f64;
|
||
decContext context;
|
||
INT_TYPE i;
|
||
|
||
decContextDefault (&context, DEC_INIT_DECIMAL128);
|
||
context.round = DEC_ROUND_DOWN;
|
||
u32.c = x;
|
||
f64 = *decSingleToWider (&u32.f, &f64);
|
||
i = DEC_FLOAT_TO_INT (&f64, &context, context.round);
|
||
if (DFP_EXCEPTIONS_ENABLED && context.status != 0)
|
||
dfp_conversion_exceptions (context.status);
|
||
return i;
|
||
}
|
||
#endif
|
||
|
||
#if defined (L_sd_to_di) || defined (L_dd_to_di) || defined (L_td_to_di) \
|
||
|| defined (L_sd_to_udi) || defined (L_dd_to_udi) || defined (L_td_to_udi)
|
||
/* decNumber doesn't provide support for conversions to 64-bit integer
|
||
types, so do it the hard way. */
|
||
INT_TYPE
|
||
DFP_TO_INT (DFP_C_TYPE x)
|
||
{
|
||
/* decNumber's decimal* types have the same format as C's _Decimal*
|
||
types, but they have different calling conventions. */
|
||
|
||
/* TODO: Decimal float to integer conversions should raise FE_INVALID
|
||
if the result value does not fit into the result type. */
|
||
|
||
IEEE_TYPE s;
|
||
char buf[BUFMAX];
|
||
char *pos;
|
||
decNumber qval, n1, n2;
|
||
decContext context;
|
||
|
||
/* Use a large context to avoid losing precision. */
|
||
decContextDefault (&context, DEC_INIT_DECIMAL128);
|
||
/* Need non-default rounding mode here. */
|
||
context.round = DEC_ROUND_DOWN;
|
||
|
||
HOST_TO_IEEE (x, &s);
|
||
TO_INTERNAL (&s, &n1);
|
||
/* Rescale if the exponent is less than zero. */
|
||
decNumberToIntegralValue (&n2, &n1, &context);
|
||
/* Get a value to use for the quantize call. */
|
||
decNumberFromString (&qval, "1.", &context);
|
||
/* Force the exponent to zero. */
|
||
decNumberQuantize (&n1, &n2, &qval, &context);
|
||
/* Get a string, which at this point will not include an exponent. */
|
||
decNumberToString (&n1, buf);
|
||
/* Ignore the fractional part. */
|
||
pos = strchr (buf, '.');
|
||
if (pos)
|
||
*pos = 0;
|
||
/* Use a C library function to convert to the integral type. */
|
||
return STR_TO_INT (buf, NULL, 10);
|
||
}
|
||
#endif
|
||
|
||
#if defined (L_si_to_dd) || defined (L_si_to_td) \
|
||
|| defined (L_usi_to_dd) || defined (L_usi_to_td)
|
||
/* Use decNumber to convert directly from integer to decimal float types. */
|
||
DFP_C_TYPE
|
||
INT_TO_DFP (INT_TYPE i)
|
||
{
|
||
union { DFP_C_TYPE c; decFloat f; } u;
|
||
|
||
u.f = *DEC_FLOAT_FROM_INT (&u.f, i);
|
||
return u.c;
|
||
}
|
||
#endif
|
||
|
||
#if defined (L_si_to_sd) || defined (L_usi_to_sd)
|
||
_Decimal32
|
||
/* Use decNumber to convert directly from integer to decimal float types. */
|
||
INT_TO_DFP (INT_TYPE i)
|
||
{
|
||
union { _Decimal32 c; decSingle f; } u32;
|
||
decDouble f64;
|
||
decContext context;
|
||
|
||
decContextDefault (&context, DEC_INIT_DECIMAL128);
|
||
f64 = *DEC_FLOAT_FROM_INT (&f64, i);
|
||
u32.f = *decSingleFromWider (&u32.f, &f64, &context);
|
||
if (DFP_EXCEPTIONS_ENABLED && context.status != 0)
|
||
dfp_conversion_exceptions (context.status);
|
||
return u32.c;
|
||
}
|
||
#endif
|
||
|
||
#if defined (L_di_to_sd) || defined (L_di_to_dd) || defined (L_di_to_td) \
|
||
|| defined (L_udi_to_sd) || defined (L_udi_to_dd) || defined (L_udi_to_td)
|
||
/* decNumber doesn't provide support for conversions from 64-bit integer
|
||
types, so do it the hard way. */
|
||
DFP_C_TYPE
|
||
INT_TO_DFP (INT_TYPE i)
|
||
{
|
||
DFP_C_TYPE f;
|
||
IEEE_TYPE s;
|
||
char buf[BUFMAX];
|
||
decContext context;
|
||
|
||
decContextDefault (&context, CONTEXT_INIT);
|
||
DFP_INIT_ROUNDMODE (context.round);
|
||
|
||
/* Use a C library function to get a floating point string. */
|
||
sprintf (buf, INT_FMT ".", CAST_FOR_FMT(i));
|
||
/* Convert from the floating point string to a decimal* type. */
|
||
FROM_STRING (&s, buf, &context);
|
||
IEEE_TO_HOST (s, &f);
|
||
|
||
if (DFP_EXCEPTIONS_ENABLED && context.status != 0)
|
||
dfp_conversion_exceptions (context.status);
|
||
|
||
return f;
|
||
}
|
||
#endif
|
||
|
||
#if defined (L_sd_to_sf) || defined (L_dd_to_sf) || defined (L_td_to_sf) \
|
||
|| defined (L_sd_to_df) || defined (L_dd_to_df) || defined (L_td_to_df) \
|
||
|| defined (L_sd_to_kf) || defined (L_dd_to_kf) || defined (L_td_to_kf) \
|
||
|| ((defined (L_sd_to_xf) || defined (L_dd_to_xf) || defined (L_td_to_xf)) \
|
||
&& LONG_DOUBLE_HAS_XF_MODE) \
|
||
|| ((defined (L_sd_to_tf) || defined (L_dd_to_tf) || defined (L_td_to_tf)) \
|
||
&& LONG_DOUBLE_HAS_TF_MODE)
|
||
BFP_TYPE
|
||
DFP_TO_BFP (DFP_C_TYPE f)
|
||
{
|
||
IEEE_TYPE s;
|
||
char buf[BUFMAX];
|
||
|
||
HOST_TO_IEEE (f, &s);
|
||
/* Write the value to a string. */
|
||
TO_STRING (&s, buf);
|
||
/* Read it as the binary floating point type and return that. */
|
||
return STR_TO_BFP (buf, NULL);
|
||
}
|
||
#endif
|
||
|
||
#if defined (L_sf_to_sd) || defined (L_sf_to_dd) || defined (L_sf_to_td) \
|
||
|| defined (L_df_to_sd) || defined (L_df_to_dd) || defined (L_df_to_td) \
|
||
|| defined (L_kf_to_sd) || defined (L_kf_to_dd) || defined (L_kf_to_td) \
|
||
|| ((defined (L_xf_to_sd) || defined (L_xf_to_dd) || defined (L_xf_to_td)) \
|
||
&& LONG_DOUBLE_HAS_XF_MODE) \
|
||
|| ((defined (L_tf_to_sd) || defined (L_tf_to_dd) || defined (L_tf_to_td)) \
|
||
&& LONG_DOUBLE_HAS_TF_MODE)
|
||
DFP_C_TYPE
|
||
BFP_TO_DFP (BFP_TYPE x)
|
||
{
|
||
DFP_C_TYPE f;
|
||
IEEE_TYPE s;
|
||
char buf[BUFMAX];
|
||
decContext context;
|
||
|
||
decContextDefault (&context, CONTEXT_INIT);
|
||
DFP_INIT_ROUNDMODE (context.round);
|
||
|
||
/* Use the sprintf library function to write the floating point value to a
|
||
string.
|
||
|
||
If we are handling the IEEE 128-bit floating point on PowerPC, use the
|
||
special function __sprintfkf instead of sprintf. This function allows us
|
||
to use __sprintfieee128 if we have a new enough GLIBC, and it can fall back
|
||
to using the traditional sprintf via conversion to IBM 128-bit if the glibc
|
||
is older. */
|
||
BFP_SPRINTF (buf, BFP_FMT, (BFP_VIA_TYPE) x);
|
||
|
||
/* Convert from the floating point string to a decimal* type. */
|
||
FROM_STRING (&s, buf, &context);
|
||
IEEE_TO_HOST (s, &f);
|
||
|
||
if (DFP_EXCEPTIONS_ENABLED && context.status != 0)
|
||
{
|
||
/* decNumber exception flags we care about here. */
|
||
int ieee_flags;
|
||
int dec_flags = DEC_IEEE_854_Inexact | DEC_IEEE_854_Invalid_operation
|
||
| DEC_IEEE_854_Overflow | DEC_IEEE_854_Underflow;
|
||
dec_flags &= context.status;
|
||
ieee_flags = DFP_IEEE_FLAGS (dec_flags);
|
||
if (ieee_flags != 0)
|
||
DFP_HANDLE_EXCEPTIONS (ieee_flags);
|
||
}
|
||
|
||
return f;
|
||
}
|
||
#endif
|
||
|
||
#if defined (L_unord_sd) || defined (L_unord_dd) || defined (L_unord_td)
|
||
CMPtype
|
||
DFP_UNORD (DFP_C_TYPE arg_a, DFP_C_TYPE arg_b)
|
||
{
|
||
decNumber arg1, arg2;
|
||
IEEE_TYPE a, b;
|
||
|
||
HOST_TO_IEEE (arg_a, &a);
|
||
HOST_TO_IEEE (arg_b, &b);
|
||
TO_INTERNAL (&a, &arg1);
|
||
TO_INTERNAL (&b, &arg2);
|
||
return (decNumberIsNaN (&arg1) || decNumberIsNaN (&arg2));
|
||
}
|
||
#endif /* L_unord_sd || L_unord_dd || L_unord_td */
|