Quellcodebibliothek Statistik Leitseite products/Sources/formale Sprachen/C/Linux/arch/parisc/math-emu/   (Open Source Betriebssystem Version 6.17.9©)  Datei vom 24.10.2025 mit Größe 8 kB image not shown  

Quelle  dfrem.c   Sprache: C

 
// SPDX-License-Identifier: GPL-2.0-or-later
/*
 * Linux/PA-RISC Project (http://www.parisc-linux.org/)
 *
 * Floating-point emulation code
 *  Copyright (C) 2001 Hewlett-Packard (Paul Bame) <bame@debian.org>
 */

/*
 * BEGIN_DESC
 *
 *  File:
 * @(#) pa/spmath/dfrem.c $Revision: 1.1 $
 *
 *  Purpose:
 * Double Precision Floating-point Remainder
 *
 *  External Interfaces:
 * dbl_frem(srcptr1,srcptr2,dstptr,status)
 *
 *  Internal Interfaces:
 *
 *  Theory:
 * <<please update with a overview of the operation of this file>>
 *
 * END_DESC
*/




#include "float.h"
#include "dbl_float.h"

/*
 *  Double Precision Floating-point Remainder
 */


int
dbl_frem (dbl_floating_point * srcptr1, dbl_floating_point * srcptr2,
   dbl_floating_point * dstptr, unsigned int *status)
{
 register unsigned int opnd1p1, opnd1p2, opnd2p1, opnd2p2;
 register unsigned int resultp1, resultp2;
 register int opnd1_exponent, opnd2_exponent, dest_exponent, stepcount;
 register boolean roundup = FALSE;

 Dbl_copyfromptr(srcptr1,opnd1p1,opnd1p2);
 Dbl_copyfromptr(srcptr2,opnd2p1,opnd2p2);
 /*
 * check first operand for NaN's or infinity
 */

 if ((opnd1_exponent = Dbl_exponent(opnd1p1)) == DBL_INFINITY_EXPONENT) {
  if (Dbl_iszero_mantissa(opnd1p1,opnd1p2)) {
   if (Dbl_isnotnan(opnd2p1,opnd2p2)) {
    /* invalid since first operand is infinity */
    if (Is_invalidtrap_enabled()) 
                                 return(INVALIDEXCEPTION);
                                Set_invalidflag();
                                Dbl_makequietnan(resultp1,resultp2);
    Dbl_copytoptr(resultp1,resultp2,dstptr);
    return(NOEXCEPTION);
   }
  }
  else {
                 /*
                   * is NaN; signaling or quiet?
                   */

                 if (Dbl_isone_signaling(opnd1p1)) {
                         /* trap if INVALIDTRAP enabled */
                         if (Is_invalidtrap_enabled()) 
                              return(INVALIDEXCEPTION);
                         /* make NaN quiet */
                         Set_invalidflag();
                         Dbl_set_quiet(opnd1p1);
                 }
   /* 
 * is second operand a signaling NaN? 
 */

   else if (Dbl_is_signalingnan(opnd2p1)) {
                         /* trap if INVALIDTRAP enabled */
                         if (Is_invalidtrap_enabled()) 
                              return(INVALIDEXCEPTION);
                         /* make NaN quiet */
                         Set_invalidflag();
                         Dbl_set_quiet(opnd2p1);
    Dbl_copytoptr(opnd2p1,opnd2p2,dstptr);
                  return(NOEXCEPTION);
   }
                 /*
                   * return quiet NaN
                   */

   Dbl_copytoptr(opnd1p1,opnd1p2,dstptr);
                 return(NOEXCEPTION);
  }
 } 
 /*
 * check second operand for NaN's or infinity
 */

 if ((opnd2_exponent = Dbl_exponent(opnd2p1)) == DBL_INFINITY_EXPONENT) {
  if (Dbl_iszero_mantissa(opnd2p1,opnd2p2)) {
   /*
 * return first operand
 */

   Dbl_copytoptr(opnd1p1,opnd1p2,dstptr);
   return(NOEXCEPTION);
  }
                /*
                 * is NaN; signaling or quiet?
                 */

                if (Dbl_isone_signaling(opnd2p1)) {
                        /* trap if INVALIDTRAP enabled */
                        if (Is_invalidtrap_enabled()) return(INVALIDEXCEPTION);
                        /* make NaN quiet */
                        Set_invalidflag();
                        Dbl_set_quiet(opnd2p1);
                }
                /*
                 * return quiet NaN
                 */

  Dbl_copytoptr(opnd2p1,opnd2p2,dstptr);
                return(NOEXCEPTION);
 }
 /*
 * check second operand for zero
 */

 if (Dbl_iszero_exponentmantissa(opnd2p1,opnd2p2)) {
  /* invalid since second operand is zero */
  if (Is_invalidtrap_enabled()) return(INVALIDEXCEPTION);
                Set_invalidflag();
                Dbl_makequietnan(resultp1,resultp2);
  Dbl_copytoptr(resultp1,resultp2,dstptr);
  return(NOEXCEPTION);
 }

 /* 
 * get sign of result
 */

 resultp1 = opnd1p1;  

 /* 
 * check for denormalized operands
 */

 if (opnd1_exponent == 0) {
  /* check for zero */
  if (Dbl_iszero_mantissa(opnd1p1,opnd1p2)) {
   Dbl_copytoptr(opnd1p1,opnd1p2,dstptr);
   return(NOEXCEPTION);
  }
  /* normalize, then continue */
  opnd1_exponent = 1;
  Dbl_normalize(opnd1p1,opnd1p2,opnd1_exponent);
 }
 else {
  Dbl_clear_signexponent_set_hidden(opnd1p1);
 }
 if (opnd2_exponent == 0) {
  /* normalize, then continue */
  opnd2_exponent = 1;
  Dbl_normalize(opnd2p1,opnd2p2,opnd2_exponent);
 }
 else {
  Dbl_clear_signexponent_set_hidden(opnd2p1);
 }

 /* find result exponent and divide step loop count */
 dest_exponent = opnd2_exponent - 1;
 stepcount = opnd1_exponent - opnd2_exponent;

 /*
 * check for opnd1/opnd2 < 1
 */

 if (stepcount < 0) {
  /*
 * check for opnd1/opnd2 > 1/2
 *
 * In this case n will round to 1, so 
 *    r = opnd1 - opnd2 
 */

  if (stepcount == -1 && 
      Dbl_isgreaterthan(opnd1p1,opnd1p2,opnd2p1,opnd2p2)) {
   /* set sign */
   Dbl_allp1(resultp1) = ~Dbl_allp1(resultp1);
   /* align opnd2 with opnd1 */
   Dbl_leftshiftby1(opnd2p1,opnd2p2); 
   Dbl_subtract(opnd2p1,opnd2p2,opnd1p1,opnd1p2,
    opnd2p1,opnd2p2);
   /* now normalize */
                 while (Dbl_iszero_hidden(opnd2p1)) {
                         Dbl_leftshiftby1(opnd2p1,opnd2p2);
                         dest_exponent--;
   }
   Dbl_set_exponentmantissa(resultp1,resultp2,opnd2p1,opnd2p2);
   goto testforunderflow;
  }
  /*
 * opnd1/opnd2 <= 1/2
 *
 * In this case n will round to zero, so 
 *    r = opnd1
 */

  Dbl_set_exponentmantissa(resultp1,resultp2,opnd1p1,opnd1p2);
  dest_exponent = opnd1_exponent;
  goto testforunderflow;
 }

 /*
 * Generate result
 *
 * Do iterative subtract until remainder is less than operand 2.
 */

 while (stepcount-- > 0 && (Dbl_allp1(opnd1p1) || Dbl_allp2(opnd1p2))) {
  if (Dbl_isnotlessthan(opnd1p1,opnd1p2,opnd2p1,opnd2p2)) {
   Dbl_subtract(opnd1p1,opnd1p2,opnd2p1,opnd2p2,opnd1p1,opnd1p2);
  }
  Dbl_leftshiftby1(opnd1p1,opnd1p2);
 }
 /*
 * Do last subtract, then determine which way to round if remainder 
 * is exactly 1/2 of opnd2 
 */

 if (Dbl_isnotlessthan(opnd1p1,opnd1p2,opnd2p1,opnd2p2)) {
  Dbl_subtract(opnd1p1,opnd1p2,opnd2p1,opnd2p2,opnd1p1,opnd1p2);
  roundup = TRUE;
 }
 if (stepcount > 0 || Dbl_iszero(opnd1p1,opnd1p2)) {
  /* division is exact, remainder is zero */
  Dbl_setzero_exponentmantissa(resultp1,resultp2);
  Dbl_copytoptr(resultp1,resultp2,dstptr);
  return(NOEXCEPTION);
 }

 /* 
 * Check for cases where opnd1/opnd2 < n 
 *
 * In this case the result's sign will be opposite that of
 * opnd1.  The mantissa also needs some correction.
 */

 Dbl_leftshiftby1(opnd1p1,opnd1p2);
 if (Dbl_isgreaterthan(opnd1p1,opnd1p2,opnd2p1,opnd2p2)) {
  Dbl_invert_sign(resultp1);
  Dbl_leftshiftby1(opnd2p1,opnd2p2);
  Dbl_subtract(opnd2p1,opnd2p2,opnd1p1,opnd1p2,opnd1p1,opnd1p2);
 }
 /* check for remainder being exactly 1/2 of opnd2 */
 else if (Dbl_isequal(opnd1p1,opnd1p2,opnd2p1,opnd2p2) && roundup) { 
  Dbl_invert_sign(resultp1);
 }

 /* normalize result's mantissa */
        while (Dbl_iszero_hidden(opnd1p1)) {
                dest_exponent--;
                Dbl_leftshiftby1(opnd1p1,opnd1p2);
        }
 Dbl_set_exponentmantissa(resultp1,resultp2,opnd1p1,opnd1p2);

        /* 
         * Test for underflow
         */

    testforunderflow:
 if (dest_exponent <= 0) {
                /* trap if UNDERFLOWTRAP enabled */
                if (Is_underflowtrap_enabled()) {
                        /*
                         * Adjust bias of result
                         */

                        Dbl_setwrapped_exponent(resultp1,dest_exponent,unfl);
   /* frem is always exact */
   Dbl_copytoptr(resultp1,resultp2,dstptr);
   return(UNDERFLOWEXCEPTION);
                }
                /*
                 * denormalize result or set to signed zero
                 */

                if (dest_exponent >= (1 - DBL_P)) {
   Dbl_rightshift_exponentmantissa(resultp1,resultp2,
    1-dest_exponent);
                }
                else {
   Dbl_setzero_exponentmantissa(resultp1,resultp2);
  }
 }
 else Dbl_set_exponent(resultp1,dest_exponent);
 Dbl_copytoptr(resultp1,resultp2,dstptr);
 return(NOEXCEPTION);
}

Messung V0.5
C=86 H=86 G=85

¤ Dauer der Verarbeitung: 0.1 Sekunden  (vorverarbeitet)  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

Die Informationen auf dieser Webseite wurden nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit, noch Qualität der bereit gestellten Informationen zugesichert.

Bemerkung:

Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.