You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			280 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			280 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
|   | //===--------------------------- fp_test.h - ------------------------------===//
 | ||
|  | //
 | ||
|  | //                     The LLVM Compiler Infrastructure
 | ||
|  | //
 | ||
|  | // This file is dual licensed under the MIT and the University of Illinois Open
 | ||
|  | // Source Licenses. See LICENSE.TXT for details.
 | ||
|  | //
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | //
 | ||
|  | // This file defines shared functions for the test.
 | ||
|  | //
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | 
 | ||
|  | #include <stdlib.h>
 | ||
|  | #include <limits.h>
 | ||
|  | #include <string.h>
 | ||
|  | #include <stdint.h>
 | ||
|  | 
 | ||
|  | enum EXPECTED_RESULT { | ||
|  |     LESS_0, LESS_EQUAL_0, EQUAL_0, GREATER_0, GREATER_EQUAL_0, NEQUAL_0 | ||
|  | }; | ||
|  | 
 | ||
|  | static inline uint16_t fromRep16(uint16_t x) | ||
|  | { | ||
|  |     return x; | ||
|  | } | ||
|  | 
 | ||
|  | static inline float fromRep32(uint32_t x) | ||
|  | { | ||
|  |     float ret; | ||
|  |     memcpy(&ret, &x, 4); | ||
|  |     return ret; | ||
|  | } | ||
|  | 
 | ||
|  | static inline double fromRep64(uint64_t x) | ||
|  | { | ||
|  |     double ret; | ||
|  |     memcpy(&ret, &x, 8); | ||
|  |     return ret; | ||
|  | } | ||
|  | 
 | ||
|  | #if __LDBL_MANT_DIG__ == 113
 | ||
|  | static inline long double fromRep128(uint64_t hi, uint64_t lo) | ||
|  | { | ||
|  |     __uint128_t x = ((__uint128_t)hi << 64) + lo; | ||
|  |     long double ret; | ||
|  |     memcpy(&ret, &x, 16); | ||
|  |     return ret; | ||
|  | } | ||
|  | #endif
 | ||
|  | 
 | ||
|  | static inline uint16_t toRep16(uint16_t x) | ||
|  | { | ||
|  |     return x; | ||
|  | } | ||
|  | 
 | ||
|  | static inline uint32_t toRep32(float x) | ||
|  | { | ||
|  |     uint32_t ret; | ||
|  |     memcpy(&ret, &x, 4); | ||
|  |     return ret; | ||
|  | } | ||
|  | 
 | ||
|  | static inline uint64_t toRep64(double x) | ||
|  | { | ||
|  |     uint64_t ret; | ||
|  |     memcpy(&ret, &x, 8); | ||
|  |     return ret; | ||
|  | } | ||
|  | 
 | ||
|  | #if __LDBL_MANT_DIG__ == 113
 | ||
|  | static inline __uint128_t toRep128(long double x) | ||
|  | { | ||
|  |     __uint128_t ret; | ||
|  |     memcpy(&ret, &x, 16); | ||
|  |     return ret; | ||
|  | } | ||
|  | #endif
 | ||
|  | 
 | ||
|  | static inline int compareResultH(uint16_t result, | ||
|  |                                  uint16_t expected) | ||
|  | { | ||
|  |     uint16_t rep = toRep16(result); | ||
|  | 
 | ||
|  |     if (rep == expected){ | ||
|  |         return 0; | ||
|  |     } | ||
|  |     // test other possible NaN representation(signal NaN)
 | ||
|  |     else if (expected == 0x7e00U){ | ||
|  |         if ((rep & 0x7c00U) == 0x7c00U && | ||
|  |             (rep & 0x3ffU) > 0){ | ||
|  |             return 0; | ||
|  |         } | ||
|  |     } | ||
|  |     return 1; | ||
|  | } | ||
|  | 
 | ||
|  | static inline int compareResultF(float result, | ||
|  |                                  uint32_t expected) | ||
|  | { | ||
|  |     uint32_t rep = toRep32(result); | ||
|  | 
 | ||
|  |     if (rep == expected){ | ||
|  |         return 0; | ||
|  |     } | ||
|  |     // test other possible NaN representation(signal NaN)
 | ||
|  |     else if (expected == 0x7fc00000U){ | ||
|  |         if ((rep & 0x7f800000U) == 0x7f800000U && | ||
|  |             (rep & 0x7fffffU) > 0){ | ||
|  |             return 0; | ||
|  |         } | ||
|  |     } | ||
|  |     return 1; | ||
|  | } | ||
|  | 
 | ||
|  | static inline int compareResultD(double result, | ||
|  |                                  uint64_t expected) | ||
|  | { | ||
|  |     uint64_t rep = toRep64(result); | ||
|  | 
 | ||
|  |     if (rep == expected){ | ||
|  |         return 0; | ||
|  |     } | ||
|  |     // test other possible NaN representation(signal NaN)
 | ||
|  |     else if (expected == 0x7ff8000000000000UL){ | ||
|  |         if ((rep & 0x7ff0000000000000UL) == 0x7ff0000000000000UL && | ||
|  |             (rep & 0xfffffffffffffUL) > 0){ | ||
|  |             return 0; | ||
|  |         } | ||
|  |     } | ||
|  |     return 1; | ||
|  | } | ||
|  | 
 | ||
|  | #if __LDBL_MANT_DIG__ == 113
 | ||
|  | // return 0 if equal
 | ||
|  | // use two 64-bit integers intead of one 128-bit integer
 | ||
|  | // because 128-bit integer constant can't be assigned directly
 | ||
|  | static inline int compareResultLD(long double result, | ||
|  |                                   uint64_t expectedHi, | ||
|  |                                   uint64_t expectedLo) | ||
|  | { | ||
|  |     __uint128_t rep = toRep128(result); | ||
|  |     uint64_t hi = rep >> 64; | ||
|  |     uint64_t lo = rep; | ||
|  | 
 | ||
|  |     if (hi == expectedHi && lo == expectedLo){ | ||
|  |         return 0; | ||
|  |     } | ||
|  |     // test other possible NaN representation(signal NaN)
 | ||
|  |     else if (expectedHi == 0x7fff800000000000UL && expectedLo == 0x0UL){ | ||
|  |         if ((hi & 0x7fff000000000000UL) == 0x7fff000000000000UL && | ||
|  |             ((hi & 0xffffffffffffUL) > 0 || lo > 0)){ | ||
|  |             return 0; | ||
|  |         } | ||
|  |     } | ||
|  |     return 1; | ||
|  | } | ||
|  | #endif
 | ||
|  | 
 | ||
|  | static inline int compareResultCMP(int result, | ||
|  |                                    enum EXPECTED_RESULT expected) | ||
|  | { | ||
|  |     switch(expected){ | ||
|  |         case LESS_0: | ||
|  |             if (result < 0) | ||
|  |                 return 0; | ||
|  |             break; | ||
|  |         case LESS_EQUAL_0: | ||
|  |             if (result <= 0) | ||
|  |                 return 0; | ||
|  |             break; | ||
|  |         case EQUAL_0: | ||
|  |             if (result == 0) | ||
|  |                 return 0; | ||
|  |             break; | ||
|  |         case NEQUAL_0: | ||
|  |             if (result != 0) | ||
|  |                 return 0; | ||
|  |             break; | ||
|  |         case GREATER_EQUAL_0: | ||
|  |             if (result >= 0) | ||
|  |                 return 0; | ||
|  |             break; | ||
|  |         case GREATER_0: | ||
|  |             if (result > 0) | ||
|  |                 return 0; | ||
|  |             break; | ||
|  |         default: | ||
|  |             return 1; | ||
|  |     } | ||
|  |     return 1; | ||
|  | } | ||
|  | 
 | ||
|  | static inline char *expectedStr(enum EXPECTED_RESULT expected) | ||
|  | { | ||
|  |     switch(expected){ | ||
|  |         case LESS_0: | ||
|  |             return "<0"; | ||
|  |         case LESS_EQUAL_0: | ||
|  |             return "<=0"; | ||
|  |         case EQUAL_0: | ||
|  |             return "=0"; | ||
|  |         case NEQUAL_0: | ||
|  |             return "!=0"; | ||
|  |         case GREATER_EQUAL_0: | ||
|  |             return ">=0"; | ||
|  |         case GREATER_0: | ||
|  |             return ">0"; | ||
|  |         default: | ||
|  |             return ""; | ||
|  |     } | ||
|  |     return ""; | ||
|  | } | ||
|  | 
 | ||
|  | static inline uint16_t makeQNaN16() | ||
|  | { | ||
|  |     return fromRep16(0x7e00U); | ||
|  | } | ||
|  | 
 | ||
|  | static inline float makeQNaN32() | ||
|  | { | ||
|  |     return fromRep32(0x7fc00000U); | ||
|  | } | ||
|  | 
 | ||
|  | static inline double makeQNaN64() | ||
|  | { | ||
|  |     return fromRep64(0x7ff8000000000000UL); | ||
|  | } | ||
|  | 
 | ||
|  | #if __LDBL_MANT_DIG__ == 113
 | ||
|  | static inline long double makeQNaN128() | ||
|  | { | ||
|  |     return fromRep128(0x7fff800000000000UL, 0x0UL); | ||
|  | } | ||
|  | #endif
 | ||
|  | 
 | ||
|  | static inline uint16_t makeNaN16(uint16_t rand) | ||
|  | { | ||
|  |     return fromRep16(0x7c00U | (rand & 0x7fffU)); | ||
|  | } | ||
|  | 
 | ||
|  | static inline float makeNaN32(uint32_t rand) | ||
|  | { | ||
|  |     return fromRep32(0x7f800000U | (rand & 0x7fffffU)); | ||
|  | } | ||
|  | 
 | ||
|  | static inline double makeNaN64(uint64_t rand) | ||
|  | { | ||
|  |     return fromRep64(0x7ff0000000000000UL | (rand & 0xfffffffffffffUL)); | ||
|  | } | ||
|  | 
 | ||
|  | #if __LDBL_MANT_DIG__ == 113
 | ||
|  | static inline long double makeNaN128(uint64_t rand) | ||
|  | { | ||
|  |     return fromRep128(0x7fff000000000000UL | (rand & 0xffffffffffffUL), 0x0UL); | ||
|  | } | ||
|  | #endif
 | ||
|  | 
 | ||
|  | static inline uint16_t makeInf16() | ||
|  | { | ||
|  |     return fromRep16(0x7c00U); | ||
|  | } | ||
|  | 
 | ||
|  | static inline float makeInf32() | ||
|  | { | ||
|  |     return fromRep32(0x7f800000U); | ||
|  | } | ||
|  | 
 | ||
|  | static inline double makeInf64() | ||
|  | { | ||
|  |     return fromRep64(0x7ff0000000000000UL); | ||
|  | } | ||
|  | 
 | ||
|  | #if __LDBL_MANT_DIG__ == 113
 | ||
|  | static inline long double makeInf128() | ||
|  | { | ||
|  |     return fromRep128(0x7fff000000000000UL, 0x0UL); | ||
|  | } | ||
|  | #endif
 |