mirror of
https://review.haiku-os.org/buildtools
synced 2025-02-14 17:57:39 +01:00
Change-Id: I42dc2a783b3e0082a7583cc7ecaa6b043ba162f4 Reviewed-on: https://review.haiku-os.org/c/buildtools/+/3021 Reviewed-by: Fredrik Holmqvist <fredrik.holmqvist@gmail.com>
342 lines
9.0 KiB
C
342 lines
9.0 KiB
C
/* Test file for mpfr_custom_*
|
|
|
|
Copyright 2005-2020 Free Software Foundation, Inc.
|
|
Contributed by the AriC and Caramba projects, INRIA.
|
|
|
|
This file is part of the GNU MPFR Library.
|
|
|
|
The GNU MPFR 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 3 of the License, or (at your
|
|
option) any later version.
|
|
|
|
The GNU MPFR 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 the GNU MPFR Library; see the file COPYING.LESSER. If not, see
|
|
https://www.gnu.org/licenses/ or write to the Free Software Foundation, Inc.,
|
|
51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. */
|
|
|
|
#include "mpfr-test.h"
|
|
|
|
#define BUFFER_SIZE 250
|
|
#define PREC_TESTED 200
|
|
|
|
long Buffer[BUFFER_SIZE];
|
|
char *stack = (char *) Buffer;
|
|
long *org = (long *) Buffer;
|
|
mpfr_prec_t p = PREC_TESTED;
|
|
|
|
#define ALIGNED(s) (((s) + sizeof (long) - 1) / sizeof (long) * sizeof (long))
|
|
|
|
/* This code ensures alignment to "long". However, this might not be
|
|
sufficient on some platforms. GCC's -Wcast-align=strict option can
|
|
be useful, but this needs successive casts to help GCC, e.g.
|
|
|
|
newx = (mpfr_ptr) (long *) (void *) old_stack;
|
|
|
|
successively casts old_stack (of type char *) to
|
|
- void *: avoid a false positive for the following cast to long *
|
|
(as the code takes care of alignment to "long");
|
|
- long *: this corresponds to the alignment checked by MPFR; coming
|
|
from void *, it will not trigger a warning (even if incorrect);
|
|
- mpfr_ptr: -Wcast-align=strict will emit a warning if mpfr_ptr has
|
|
an alignment requirement stronger than long *. In such a case,
|
|
the code will have to be fixed.
|
|
*/
|
|
|
|
static void *
|
|
new_st (size_t s)
|
|
{
|
|
void *p = (void *) stack;
|
|
|
|
if (MPFR_UNLIKELY (s > (char *) &Buffer[BUFFER_SIZE] - stack))
|
|
{
|
|
printf ("[INTERNAL TEST ERROR] Stack overflow.\n");
|
|
exit (1);
|
|
}
|
|
stack += ALIGNED (s);
|
|
return p;
|
|
}
|
|
|
|
static void
|
|
reset_stack (void)
|
|
{
|
|
stack = (char *) Buffer;
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Alloc a new mpfr_t on the main stack */
|
|
static mpfr_ptr
|
|
new_mpfr (mpfr_prec_t p)
|
|
{
|
|
mpfr_ptr x = (mpfr_ptr) new_st (sizeof (mpfr_t));
|
|
void *mantissa = new_st (mpfr_custom_get_size (p));
|
|
mpfr_custom_init (mantissa, p);
|
|
mpfr_custom_init_set (x, 0, 0, p, mantissa);
|
|
return x;
|
|
}
|
|
|
|
/* Alloc a new mpfr_t on the main stack */
|
|
static mpfr_ptr
|
|
new_nan (mpfr_prec_t p)
|
|
{
|
|
mpfr_ptr x = (mpfr_ptr) new_st (sizeof (mpfr_t));
|
|
void *mantissa = new_st ((mpfr_custom_get_size) (p));
|
|
(mpfr_custom_init) (mantissa, p);
|
|
(mpfr_custom_init_set) (x, MPFR_NAN_KIND, 0, p, mantissa);
|
|
return x;
|
|
}
|
|
|
|
/* Alloc a new mpfr_t on the main stack */
|
|
static mpfr_ptr
|
|
new_inf (mpfr_prec_t p)
|
|
{
|
|
mpfr_ptr x = (mpfr_ptr) new_st (sizeof (mpfr_t));
|
|
void *mantissa = new_st ((mpfr_custom_get_size) (p));
|
|
(mpfr_custom_init) (mantissa, p);
|
|
(mpfr_custom_init_set) (x, -MPFR_INF_KIND, 0, p, mantissa);
|
|
return x;
|
|
}
|
|
|
|
/* Garbage the stack by keeping only x and save it in old_stack */
|
|
static mpfr_ptr
|
|
return_mpfr (mpfr_ptr x, char *old_stack)
|
|
{
|
|
void *mantissa = mpfr_custom_get_significand (x);
|
|
size_t size_mantissa = mpfr_custom_get_size (mpfr_get_prec (x));
|
|
mpfr_ptr newx;
|
|
long *newx2;
|
|
|
|
memmove (old_stack, x, sizeof (mpfr_t));
|
|
memmove (old_stack + ALIGNED (sizeof (mpfr_t)), mantissa, size_mantissa);
|
|
newx = (mpfr_ptr) (long *) (void *) old_stack;
|
|
newx2 = (long *) (void *) (old_stack + ALIGNED (sizeof (mpfr_t)));
|
|
mpfr_custom_move (newx, newx2);
|
|
stack = (char *) newx2 + ALIGNED (size_mantissa);
|
|
return newx;
|
|
}
|
|
|
|
/* Garbage the stack by keeping only x and save it in old_stack */
|
|
static mpfr_ptr
|
|
return_mpfr_func (mpfr_ptr x, char *old_stack)
|
|
{
|
|
void *mantissa = (mpfr_custom_get_significand) (x);
|
|
size_t size_mantissa = (mpfr_custom_get_size) (mpfr_get_prec (x));
|
|
mpfr_ptr newx;
|
|
|
|
memmove (old_stack, x, sizeof (mpfr_t));
|
|
memmove (old_stack + ALIGNED (sizeof (mpfr_t)), mantissa, size_mantissa);
|
|
newx = (mpfr_ptr) (long *) (void *) old_stack;
|
|
(mpfr_custom_move) (newx, old_stack + ALIGNED (sizeof (mpfr_t)));
|
|
stack = old_stack + ALIGNED (sizeof (mpfr_t)) + ALIGNED (size_mantissa);
|
|
return newx;
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
static void
|
|
test1 (void)
|
|
{
|
|
mpfr_ptr x, y;
|
|
|
|
reset_stack ();
|
|
org = (long *) (void *) stack;
|
|
|
|
x = new_mpfr (p);
|
|
y = new_mpfr (p);
|
|
mpfr_set_ui (x, 42, MPFR_RNDN);
|
|
mpfr_set_ui (y, 17, MPFR_RNDN);
|
|
mpfr_add (y, x, y, MPFR_RNDN);
|
|
y = return_mpfr (y, (char *) org);
|
|
if ((long *) y != org || mpfr_cmp_ui (y, 59) != 0)
|
|
{
|
|
printf ("Compact (1) failed!\n");
|
|
exit (1);
|
|
}
|
|
|
|
x = new_mpfr (p);
|
|
y = new_mpfr (p);
|
|
mpfr_set_ui (x, 4217, MPFR_RNDN);
|
|
mpfr_set_ui (y, 1742, MPFR_RNDN);
|
|
mpfr_add (y, x, y, MPFR_RNDN);
|
|
y = return_mpfr_func (y, (char *) org);
|
|
if ((long *) y != org || mpfr_cmp_ui (y, 5959) != 0)
|
|
{
|
|
printf ("Compact (5) failed!\n");
|
|
exit (1);
|
|
}
|
|
|
|
reset_stack ();
|
|
}
|
|
|
|
static void
|
|
test_nan_inf_zero (void)
|
|
{
|
|
mpfr_ptr val;
|
|
int sign;
|
|
int kind;
|
|
|
|
reset_stack ();
|
|
|
|
val = new_mpfr (MPFR_PREC_MIN);
|
|
mpfr_set_nan (val);
|
|
kind = (mpfr_custom_get_kind) (val);
|
|
if (kind != MPFR_NAN_KIND)
|
|
{
|
|
printf ("mpfr_custom_get_kind error: ");
|
|
mpfr_dump (val);
|
|
printf (" is kind %d instead of %d\n", kind, (int) MPFR_NAN_KIND);
|
|
exit (1);
|
|
}
|
|
|
|
val = new_nan (MPFR_PREC_MIN);
|
|
if (!mpfr_nan_p(val))
|
|
{
|
|
printf ("Error: mpfr_custom_init_set doesn't set NAN mpfr.\n");
|
|
exit (1);
|
|
}
|
|
|
|
val = new_inf (MPFR_PREC_MIN);
|
|
if (!mpfr_inf_p(val) || mpfr_sgn(val) >= 0)
|
|
{
|
|
printf ("Error: mpfr_custom_init_set doesn't set -INF mpfr.\n");
|
|
exit (1);
|
|
}
|
|
|
|
sign = 1;
|
|
mpfr_set_inf (val, sign);
|
|
kind = (mpfr_custom_get_kind) (val);
|
|
if ((ABS (kind) != MPFR_INF_KIND) || (VSIGN (kind) != VSIGN (sign)))
|
|
{
|
|
printf ("mpfr_custom_get_kind error: ");
|
|
mpfr_dump (val);
|
|
printf (" is kind %d instead of %d\n", kind, (int) MPFR_INF_KIND);
|
|
printf (" have sign %d instead of %d\n", VSIGN (kind), VSIGN (sign));
|
|
exit (1);
|
|
}
|
|
|
|
sign = -1;
|
|
mpfr_set_zero (val, sign);
|
|
kind = (mpfr_custom_get_kind) (val);
|
|
if ((ABS (kind) != MPFR_ZERO_KIND) || (VSIGN (kind) != VSIGN (sign)))
|
|
{
|
|
printf ("mpfr_custom_get_kind error: ");
|
|
mpfr_dump (val);
|
|
printf (" is kind %d instead of %d\n", kind, (int) MPFR_ZERO_KIND);
|
|
printf (" have sign %d instead of %d\n", VSIGN (kind), VSIGN (sign));
|
|
exit (1);
|
|
}
|
|
|
|
reset_stack ();
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* We build the MPFR variable each time it is needed */
|
|
/* a[0] is the kind, a[1] is the exponent, &a[2] is the mantissa */
|
|
static long *
|
|
dummy_new (void)
|
|
{
|
|
long *r;
|
|
|
|
r = (long *) new_st (ALIGNED (2 * sizeof (long)) +
|
|
ALIGNED (mpfr_custom_get_size (p)));
|
|
(mpfr_custom_init) (&r[2], p);
|
|
r[0] = (int) MPFR_NAN_KIND;
|
|
r[1] = 0;
|
|
return r;
|
|
}
|
|
|
|
static long *
|
|
dummy_set_si (long si)
|
|
{
|
|
mpfr_t x;
|
|
long * r = dummy_new ();
|
|
(mpfr_custom_init_set) (x, MPFR_REGULAR_KIND, 0, p, &r[2]);
|
|
mpfr_set_si (x, si, MPFR_RNDN);
|
|
r[0] = mpfr_custom_get_kind (x);
|
|
r[1] = mpfr_custom_get_exp (x);
|
|
return r;
|
|
}
|
|
|
|
static long *
|
|
dummy_add (long *a, long *b)
|
|
{
|
|
mpfr_t x, y, z;
|
|
long *r = dummy_new ();
|
|
mpfr_custom_init_set (x, 0 + MPFR_REGULAR_KIND, 0, p, &r[2]);
|
|
(mpfr_custom_init_set) (y, a[0], a[1], p, &a[2]);
|
|
mpfr_custom_init_set (z, 0 + b[0], b[1], p, &b[2]);
|
|
mpfr_add (x, y, z, MPFR_RNDN);
|
|
r[0] = (mpfr_custom_get_kind) (x);
|
|
r[1] = (mpfr_custom_get_exp) (x);
|
|
return r;
|
|
}
|
|
|
|
static long *
|
|
dummy_compact (long *r, long *org_stack)
|
|
{
|
|
memmove (org_stack, r,
|
|
ALIGNED (2*sizeof (long)) + ALIGNED ((mpfr_custom_get_size) (p)));
|
|
return org_stack;
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
static void
|
|
test2 (void)
|
|
{
|
|
mpfr_t x;
|
|
long *a, *b, *c;
|
|
|
|
reset_stack ();
|
|
org = (long *) (void *) stack;
|
|
|
|
a = dummy_set_si (42);
|
|
b = dummy_set_si (17);
|
|
c = dummy_add (a, b);
|
|
c = dummy_compact (c, org);
|
|
(mpfr_custom_init_set) (x, c[0], c[1], p, &c[2]);
|
|
if (c != org || mpfr_cmp_ui (x, 59) != 0)
|
|
{
|
|
printf ("Compact (2) failed! c=%p a=%p\n", (void *) c, (void *) a);
|
|
mpfr_dump (x);
|
|
exit (1);
|
|
}
|
|
|
|
a = dummy_set_si (42);
|
|
b = dummy_set_si (-17);
|
|
c = dummy_add (a, b);
|
|
c = dummy_compact (c, org);
|
|
(mpfr_custom_init_set) (x, c[0], c[1], p, &c[2]);
|
|
if (c != org || mpfr_cmp_ui (x, 25) != 0)
|
|
{
|
|
printf ("Compact (6) failed! c=%p a=%p\n", (void *) c, (void *) a);
|
|
mpfr_dump (x);
|
|
exit (1);
|
|
}
|
|
|
|
reset_stack ();
|
|
}
|
|
|
|
|
|
int
|
|
main (void)
|
|
{
|
|
tests_start_mpfr ();
|
|
/* We test iff long = mp_limb_t */
|
|
if (sizeof (long) == sizeof (mp_limb_t))
|
|
{
|
|
test1 ();
|
|
test2 ();
|
|
test_nan_inf_zero ();
|
|
}
|
|
tests_end_mpfr ();
|
|
return 0;
|
|
}
|