OpenCores
URL https://opencores.org/ocsvn/openrisc/openrisc/trunk

Subversion Repositories openrisc

[/] [openrisc/] [trunk/] [orpsocv2/] [sw/] [apps/] [testfloat/] [testfloat.c] - Rev 438

Go to most recent revision | Compare with Previous | Blame | View Log

 
/*
===============================================================================
 
This C source file is part of TestFloat, Release 2a, a package of programs
for testing the correctness of floating-point arithmetic complying to the
IEC/IEEE Standard for Floating-Point.
 
Written by John R. Hauser.  More information is available through the Web
page `http://HTTP.CS.Berkeley.EDU/~jhauser/arithmetic/TestFloat.html'.
 
THIS SOFTWARE IS DISTRIBUTED AS IS, FOR FREE.  Although reasonable effort
has been made to avoid it, THIS SOFTWARE MAY CONTAIN FAULTS THAT WILL AT
TIMES RESULT IN INCORRECT BEHAVIOR.  USE OF THIS SOFTWARE IS RESTRICTED TO
PERSONS AND ORGANIZATIONS WHO CAN AND WILL TAKE FULL RESPONSIBILITY FOR ANY
AND ALL LOSSES, COSTS, OR OTHER PROBLEMS ARISING FROM ITS USE.
 
Derivative works are acceptable, even for commercial purposes, so long as
(1) they include prominent notice that the work is derivative, and (2) they
include prominent notice akin to these four paragraphs for those parts of
this code that are retained.
 
Modified for use with or1ksim's testsuite.
 
Contributor Julius Baxter <julius.baxter@orsoc.se>
 
 
===============================================================================
*/
								 /*
#include <stdlib.h>
#include <signal.h>
#include <string.h>
								 */
#include "cpu-utils.h" // OR1k support C library
#include "milieu.h"
#include "printf.h"
#include "spr-defs.h"
#include "int.h"
#include "fail.h"
#include "softfloat.h"
#include "testCases.h"
#include "testLoops.h"
#include "systflags.h" // defines fpcsr_flags global
#include "testFunction.h"
								 /*
static void catchSIGINT( int signalCode )
{
 
    if ( stop ) exit( EXIT_FAILURE );
    stop = TRUE;
 
}
*/
 
// Clear flags in floating point control/status reg (FPCSR), and enable
// exception
void inline or1k_enable_fpee(void)
{
  // Read the FPCSR
  unsigned int spr = SPR_FPCSR;
  unsigned int value;
  // Read the SPR
  asm("l.mfspr\t\t%0,%1,0" : "=r" (value) : "r" (spr));
 
  // Clear flags in FPCSR
  value &= ~(SPR_FPCSR_ALLF);
 
  // Re-enable floating point exceptions
  value |= SPR_FPCSR_FPEE;
 
  // Write value back to FPCSR
  asm("l.mtspr\t\t%0,%1,0": : "r" (spr), "r" (value));
 
}
  // Interrupt handler for floating point exceptions
  // Just copy out the flags and go back to work...
 
void float_except_handler(void)
{
 
  // Read the FPCSR
  unsigned int spr = SPR_FPCSR;
  unsigned int value;
 
  // Clear the global we'll use 
  fpcsr_flags = 0;
  // Read the SPR
  asm("l.mfspr\t\t%0,%1,0" : "=r" (value) : "r" (spr));
 
  // Extract the flags from OR1K's FPCSR, put into testfloat's flags format  
  if (value & SPR_FPCSR_IXF)
    fpcsr_flags |= float_flag_inexact;
 
  if (value & SPR_FPCSR_UNF)
    fpcsr_flags |= float_flag_underflow;
 
  if (value & SPR_FPCSR_OVF)
    fpcsr_flags |= float_flag_overflow;
 
  if (value & SPR_FPCSR_DZF)
    fpcsr_flags |= float_flag_divbyzero;
 
  if (value & SPR_FPCSR_IVF)
    fpcsr_flags |= float_flag_invalid;
 
  //  printf("testfloat: getting flags, FPCSR: 0x%x, softfloatflags: 0x%x\n", 
  //         value & SPR_FPCSR_ALLF, flags);
 
  // This clears flags and re-enables FPEE bit
  or1k_enable_fpee();
 
}
 
// Running this bare metal standalone for OR1K - hard set the configuration
int
main( int argc, char **argv )
{
  //    char *argPtr; // Unused variable
    flag functionArgument;
    uint8 functionCode;
    int8 operands, roundingPrecision, roundingMode;
 
    // Add exception handler for floating point exception
    add_handler(0xd, float_except_handler);
 
    // Enable floating point exceptions in FPCSR
    or1k_enable_fpee();
 
    // If we have UART init it:
#ifdef _UART_H_
    uart_init(DEFAULT_UART);
#endif
 
    printf("testfloat\n");
 
    fail_programName = "testfloat";
    //if ( argc <= 1 ) goto writeHelpMessage;
    testCases_setLevel( 1 );
    trueName = "soft";
    testName = "syst";
    errorStop = FALSE;
    forever = FALSE;
    maxErrorCount = 0;
    trueFlagsPtr = &float_exception_flags;
    testFlagsFunctionPtr = syst_float_flags_clear;
    tininessModeName = 0;
    operands = 0;
    roundingPrecision = 0;
    roundingMode = 0;// ROUND_DOWN// - for only round down tests ; 
                     //0 - for do all rounding modes 
    // "all" setting:
    functionArgument = TRUE;
    functionCode = 0; // See testFunction.c for list. 
    // 0 = all possible functions
    // 9 = float32_to_int32
    // 10 = float32_to_int32_round_to_zero
    // 17 = float32_add
    // 18 = float32_sub
    // 19 = float32_mul
    // 20 = float32_div
    // 23 = float32_eq
    // 24 = float32_le
    // 25 = float32_lt
 
    operands = 0;
 
    if ( ! functionArgument ) fail( "Function argument required" );
    //    (void) signal( SIGINT, catchSIGINT );
    //    (void) signal( SIGTERM, catchSIGINT );
    if ( functionCode ) {
        if ( forever ) {
            if ( ! roundingPrecision ) roundingPrecision = 80;
            if ( ! roundingMode ) roundingMode = ROUND_NEAREST_EVEN;
        }
        testFunction( functionCode, roundingPrecision, roundingMode );
    }
    else {
 
      for ( functionCode = 1;
                  functionCode < NUM_FUNCTIONS;
                  ++functionCode
                ) {
	//printf("trying function %d\n",functionCode);
                if ( functionExists[ functionCode ] ) {
                    testFunction(
                        functionCode, roundingPrecision, roundingMode );
                }
            }
 
    }
    exitWithStatus();
 
    // Should never reach here
    return 1;
 
}
 
 

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.