mirror of
				https://github.com/qemu/qemu.git
				synced 2025-10-24 19:01:24 +00:00 
			
		
		
		
	 603bd9c25e
			
		
	
	
		603bd9c25e
		
	
	
	
	
		
			
			We need to read the floating-point flags before printf may do other floating-point operations which may affect the flags. Hexagon reference files regenerated by Taylor Simpson. Signed-off-by: Taylor Simpson <tsimpson@quicinc.com> Signed-off-by: Richard Henderson <richard.henderson@linaro.org> Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org> Signed-off-by: Alex Bennée <alex.bennee@linaro.org> Reviewed-by: Taylor Simpson <tsimpson@quicinc.com> Message-Id: <1639510781-3790-1-git-send-email-tsimpson@quicinc.com> Message-Id: <20211224035541.2159966-2-richard.henderson@linaro.org> Message-Id: <20220105135009.1584676-29-alex.bennee@linaro.org>
		
			
				
	
	
		
			108 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			108 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Floating Point Convert Single to Various
 | |
|  *
 | |
|  * Copyright (c) 2019 Linaro
 | |
|  *
 | |
|  * SPDX-License-Identifier: GPL-3.0-or-later
 | |
|  */
 | |
| 
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <math.h>
 | |
| #include <float.h>
 | |
| #include <fenv.h>
 | |
| 
 | |
| 
 | |
| #include "float_helpers.h"
 | |
| 
 | |
| #define ARRAY_SIZE(x) (sizeof(x) / sizeof((x)[0]))
 | |
| 
 | |
| typedef struct {
 | |
|     int flag;
 | |
|     char *desc;
 | |
| } float_mapping;
 | |
| 
 | |
| float_mapping round_flags[] = {
 | |
|     { FE_TONEAREST, "to nearest" },
 | |
| #ifdef FE_UPWARD
 | |
|     { FE_UPWARD, "upwards" },
 | |
| #endif
 | |
| #ifdef FE_DOWNWARD
 | |
|     { FE_DOWNWARD, "downwards" },
 | |
| #endif
 | |
| #ifdef FE_TOWARDZERO
 | |
|     { FE_TOWARDZERO, "to zero" }
 | |
| #endif
 | |
| };
 | |
| 
 | |
| static void print_input(float input)
 | |
| {
 | |
|     char *in_fmt = fmt_f32(input);
 | |
|     printf("from single: %s\n", in_fmt);
 | |
|     free(in_fmt);
 | |
| }
 | |
| 
 | |
| static void convert_single_to_double(float input)
 | |
| {
 | |
|     double output;
 | |
|     char *out_fmt, *flag_fmt;
 | |
| 
 | |
|     feclearexcept(FE_ALL_EXCEPT);
 | |
| 
 | |
|     output = input;
 | |
| 
 | |
|     flag_fmt = fmt_flags();
 | |
|     out_fmt = fmt_f64(output);
 | |
|     printf("  to double: %s (%s)\n", out_fmt, flag_fmt);
 | |
|     free(out_fmt);
 | |
|     free(flag_fmt);
 | |
| }
 | |
| 
 | |
| #define xstr(a) str(a)
 | |
| #define str(a) #a
 | |
| 
 | |
| #define CONVERT_SINGLE_TO_INT(TYPE, FMT)                            \
 | |
|     static void convert_single_to_ ## TYPE(float input)             \
 | |
|     {                                                               \
 | |
|         TYPE ## _t output;                                          \
 | |
|         char *flag_fmt;                                             \
 | |
|         const char to[] = "to " xstr(TYPE);                         \
 | |
|         feclearexcept(FE_ALL_EXCEPT);                               \
 | |
|         output = input;                                             \
 | |
|         flag_fmt = fmt_flags();                                     \
 | |
|         printf("%11s: %" FMT " (%s)\n", to, output, flag_fmt);      \
 | |
|         free(flag_fmt);                                             \
 | |
|     }
 | |
| 
 | |
| CONVERT_SINGLE_TO_INT( int32, PRId32)
 | |
| CONVERT_SINGLE_TO_INT(uint32, PRId32)
 | |
| CONVERT_SINGLE_TO_INT( int64, PRId64)
 | |
| CONVERT_SINGLE_TO_INT(uint64, PRId64)
 | |
| 
 | |
| int main(int argc, char *argv[argc])
 | |
| {
 | |
|     int i, j, nums;
 | |
| 
 | |
|     nums = get_num_f32();
 | |
| 
 | |
|     for (i = 0; i < ARRAY_SIZE(round_flags); ++i) {
 | |
|         if (fesetround(round_flags[i].flag) != 0) {
 | |
|             printf("### Rounding %s skipped\n", round_flags[i].desc);
 | |
|             continue;
 | |
|         }
 | |
|         printf("### Rounding %s\n", round_flags[i].desc);
 | |
|         for (j = 0; j < nums; j++) {
 | |
|             float input = get_f32(j);
 | |
|             print_input(input);
 | |
|             /* convert_single_to_half(input); */
 | |
|             convert_single_to_double(input);
 | |
|             convert_single_to_int32(input);
 | |
|             convert_single_to_int64(input);
 | |
|             convert_single_to_uint32(input);
 | |
|             convert_single_to_uint64(input);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 |