1 #include <stdio.h>
2 #include "opcodes.h"
3 #include "dfp_utils.h"
4 #define __STDC_WANT_DEC_FP__ 1
5 #include <float.h>
6 
7 #define L2D(insn,  initial, target,round)                               \
8   ({                                                                    \
9     register unsigned long source asm("2") =  initial;                  \
10     register typeof(target) _t asm("f0");                               \
11     asm volatile(insn(round,0,0,2) :"=f" (_t):"d"(source));             \
12     _t;                                                                 \
13   })
14 
15 #define I2D(insn,  initial, target,round)                               \
16   ({                                                                    \
17     register int source asm("2") =  initial;                            \
18     register typeof(target) _t asm("f0");                               \
19     asm volatile(insn(round,0,0,2) :"=f" (_t):"d"(source));             \
20     _t;                                                                 \
21 })
22 
23 #define D2L(insn, initial, type, round, cc)                             \
24   ({                                                                    \
25     register type source asm("f0") =  initial;                          \
26     register unsigned long target asm ("2") = 0;                        \
27     asm volatile(insn(round,0,2,0)                                      \
28                  "ipm %1\n\t"                                           \
29                  "srl %1,28\n\t"                                        \
30                  :"=d" (target), "=d" (cc) :"f"(source):"cc");          \
31     target;                                                             \
32   })
33 
34 #define D2I(insn, initial, type, round, cc)                             \
35   ({                                                                    \
36     register type source asm("f0") =  initial;                          \
37     register int target asm ("2") = 0;                                  \
38     asm volatile(insn(round,0,2,0)                                      \
39                  "ipm %1\n\t"                                           \
40                  "srl %1,28\n\t"                                        \
41                  :"=d" (target), "=d" (cc) :"f"(source):"cc");          \
42     target;                                                             \
43 })
44 
45 
46 #define DO_PRINT_L2D(insn, l, d, round)                                 \
47   ({                                                                    \
48     printf(#insn " round=%d %lu -> ", 0x##round, l);                    \
49     d = L2D(insn, l, d, round);                                         \
50     DFP_VAL_PRINT(d, typeof(d));                                        \
51     printf("\n");                                                       \
52   })
53 
54 #define DO_INSN_L2D(insn, round, type)                                  \
55   ({                                                                    \
56     type d;                                                             \
57     DO_PRINT_L2D(insn, 0UL, d, round);                                  \
58     DO_PRINT_L2D(insn, 1UL, d, round);                                  \
59     DO_PRINT_L2D(insn, 0xffffffffUL, d, round);                         \
60     DO_PRINT_L2D(insn, 0x80000000UL, d, round);                         \
61     DO_PRINT_L2D(insn, 0x7fffffffUL, d, round);                         \
62     DO_PRINT_L2D(insn, 0x100000000UL, d, round);                        \
63     DO_PRINT_L2D(insn, 0xffffffffffffffffUL, d, round);                 \
64     DO_PRINT_L2D(insn, 0x8000000000000000UL, d, round);                 \
65     DO_PRINT_L2D(insn, 0x7fffffffffffffffUL, d, round);                 \
66   })
67 
68 #define DO_PRINT_I2D(insn, l, d, round)                                 \
69   ({                                                                    \
70     printf(#insn " round=%d %d -> ", 0x##round, l);                     \
71     d = I2D(insn, l, d, round);                                         \
72     DFP_VAL_PRINT(d, typeof(d));                                        \
73     printf("\n");                                                       \
74   })
75 
76 #define DO_INSN_I2D(insn, round, type)                                  \
77   ({                                                                    \
78     type d;                                                             \
79     DO_PRINT_I2D(insn, 0, d, round);                                    \
80     DO_PRINT_I2D(insn, 1, d, round);                                    \
81     DO_PRINT_I2D(insn, 0xffffffff, d, round);                           \
82     DO_PRINT_I2D(insn, 0x80000000, d, round);                           \
83     DO_PRINT_I2D(insn, 0x7fffffff, d, round);                           \
84    })
85 
86 #define DO_PRINT_D2L(insn, d, type, round, cc)                          \
87   ({                                                                    \
88     printf(#insn " round=%d ", 0x##round);                              \
89     DFP_VAL_PRINT(d, type);                                             \
90     printf(" -> %lu ", D2L(insn, d, type, round, cc));                  \
91     printf("cc=%d\n", cc);                                              \
92   })
93 
94 #define DO_INSN_D2L(insn, round, type)                                  \
95   ({                                                                    \
96     int cc;                                                             \
97     type d;                                                             \
98     d = -1.1DD;                                                         \
99     DO_PRINT_D2L(insn, d, type, round, cc);                             \
100     d = 0.DD;                                                           \
101     DO_PRINT_D2L(insn, d, type, round, cc);                             \
102     d = 1.DD;                                                           \
103     DO_PRINT_D2L(insn, d, type, round, cc);                             \
104     d = 1.4DD;                                                          \
105     DO_PRINT_D2L(insn, d, type, round, cc);                             \
106     d = 1.5DD;                                                          \
107     DO_PRINT_D2L(insn, d, type, round, cc);                             \
108     d = 1.6DD;                                                          \
109     DO_PRINT_D2L(insn, d, type, round, cc);                             \
110     d = 1.6E+4DD;                                                       \
111     DO_PRINT_D2L(insn, d, type, round, cc);                             \
112     d = 1.6E+8DD;                                                       \
113     DO_PRINT_D2L(insn, d, type, round, cc);                             \
114     d = 1.6E+4DD;                                                       \
115     DO_PRINT_D2L(insn, d, type, round, cc);                             \
116     d = 1.6E+12DD;                                                      \
117     DO_PRINT_D2L(insn, d, type, round, cc);                             \
118     d = 1.6E+20DD;                                                      \
119     DO_PRINT_D2L(insn, d, type, round, cc);                             \
120     d = 1.6E+200DD;                                                     \
121     DO_PRINT_D2L(insn, d, type, round, cc);                             \
122     d = 1.6E-4DD;                                                       \
123     DO_PRINT_D2L(insn, d, type, round, cc);                             \
124     d = DEC32_MIN;                                                      \
125     DO_PRINT_D2L(insn, d, type, round, cc);                             \
126     d = DEC32_MAX;                                                      \
127     DO_PRINT_D2L(insn, d, type, round, cc);                             \
128     d = DEC64_MIN;                                                      \
129     DO_PRINT_D2L(insn, d, type, round, cc);                             \
130     d = DEC64_MAX;                                                      \
131     DO_PRINT_D2L(insn, d, type, round, cc);                             \
132   })
133 
134 #define DO_PRINT_D2I(insn, d, type, round, cc)                          \
135   ({                                                                    \
136     printf(#insn " round=%d ", 0x##round);                              \
137     DFP_VAL_PRINT(d, type);                                             \
138     printf(" -> %d ", D2I(insn, d, type, round, cc));                   \
139     printf("cc=%d\n", cc);                                              \
140   })
141 
142 #define DO_INSN_D2I(insn, round, type)                                  \
143   ({                                                                    \
144     int cc;                                                             \
145     type d;                                                             \
146     d = -1.1DD;                                                         \
147     DO_PRINT_D2I(insn, d, type, round, cc);                             \
148     d = 0.DD;                                                           \
149     DO_PRINT_D2I(insn, d, type, round, cc);                             \
150     d = 1.DD;                                                           \
151     DO_PRINT_D2I(insn, d, type, round, cc);                             \
152     d = 1.4DD;                                                          \
153     DO_PRINT_D2I(insn, d, type, round, cc);                             \
154     d = 1.5DD;                                                          \
155     DO_PRINT_D2I(insn, d, type, round, cc);                             \
156     d = 1.6DD;                                                          \
157     DO_PRINT_D2I(insn, d, type, round, cc);                             \
158     d = 1.6E+4DD;                                                       \
159     DO_PRINT_D2I(insn, d, type, round, cc);                             \
160     d = 1.6E+8DD;                                                       \
161     DO_PRINT_D2I(insn, d, type, round, cc);                             \
162     d = 1.6E+4DD;                                                       \
163     DO_PRINT_D2I(insn, d, type, round, cc);                             \
164     d = 1.6E+12DD;                                                      \
165     DO_PRINT_D2I(insn, d, type, round, cc);                             \
166     d = 1.6E+20DD;                                                      \
167     DO_PRINT_D2I(insn, d, type, round, cc);                             \
168     d = 1.6E+200DD;                                                     \
169     DO_PRINT_D2I(insn, d, type, round, cc);                             \
170     d = 1.6E-4DD;                                                       \
171     DO_PRINT_D2I(insn, d, type, round, cc);                             \
172     d = DEC32_MIN;                                                      \
173     DO_PRINT_D2I(insn, d, type, round, cc);                             \
174     d = DEC32_MAX;                                                      \
175     DO_PRINT_D2I(insn, d, type, round, cc);                             \
176     d = DEC64_MIN;                                                      \
177     DO_PRINT_D2I(insn, d, type, round, cc);                             \
178     d = DEC64_MAX;                                                      \
179     DO_PRINT_D2I(insn, d, type, round, cc);                             \
180   })
181 
182 #define DO_D2L(round)                                                   \
183   ({                                                                    \
184     DO_INSN_D2L(CLFDTR, round, _Decimal64);                             \
185     DO_INSN_D2L(CLGDTR, round, _Decimal64);                             \
186     DO_INSN_D2I(CFDTR,  round, _Decimal64);                             \
187     DO_INSN_D2L(CLFXTR, round, _Decimal128);                            \
188     DO_INSN_D2L(CLGXTR, round, _Decimal128);                            \
189     DO_INSN_D2I(CFXTR,  round, _Decimal128);                            \
190   })
191 
192 
193 int main()
194 {
195   /* rounding mode is not used for the following insns */
196   DO_INSN_I2D(CDFTR,  0, _Decimal64);
197   DO_INSN_I2D(CXFTR,  0, _Decimal128);
198   DO_INSN_L2D(CDLFTR, 0, _Decimal64);
199   DO_INSN_L2D(CXLFTR, 0, _Decimal128);
200   DO_INSN_L2D(CXLGTR, 0, _Decimal128);
201 
202   /* Omit rounding mode value 0 and 2 as the current DFP rounding
203      mode is chosen for these values. */
204   DO_INSN_L2D(CDLGTR, 1, _Decimal64);
205   DO_D2L(1);
206 
207   DO_INSN_L2D(CDLGTR, 3, _Decimal64);
208   DO_D2L(3);
209 
210   DO_INSN_L2D(CDLGTR, 4, _Decimal64);
211   DO_D2L(4);
212 
213   DO_INSN_L2D(CDLGTR, 5, _Decimal64);
214   DO_D2L(5);
215 
216   DO_INSN_L2D(CDLGTR, 6, _Decimal64);
217   DO_D2L(6);
218 
219   DO_INSN_L2D(CDLGTR, 7, _Decimal64);
220   DO_D2L(7);
221 
222   DO_INSN_L2D(CDLGTR, 8, _Decimal64);
223   DO_D2L(8);
224 
225   DO_INSN_L2D(CDLGTR, 9, _Decimal64);
226   DO_D2L(9);
227 
228   DO_INSN_L2D(CDLGTR, a, _Decimal64);
229   DO_D2L(a);
230 
231   DO_INSN_L2D(CDLGTR, b, _Decimal64);
232   DO_D2L(b);
233 
234   DO_INSN_L2D(CDLGTR, c, _Decimal64);
235   DO_D2L(c);
236 
237   DO_INSN_L2D(CDLGTR, d, _Decimal64);
238   DO_D2L(d);
239 
240   DO_INSN_L2D(CDLGTR, e, _Decimal64);
241   DO_D2L(e);
242 
243   DO_INSN_L2D(CDLGTR, f, _Decimal64);
244   DO_D2L(f);
245 
246   return 0;
247 }
248