1 #include <stdio.h>
2 #include "opcodes.h"
3 
4 /* The FLOGR insn reads from register R2 and writes to register R1 and
5    R1 + 1. So we need to distinguish three cases:
6 
7    (1) All three registers R1, R1 + 1, and R2 are distinct
8    (2) R2 == R1
9    (3) R2 == R1 + 1
10 
11    These are tested by flogr1, flogr2, and flogr3, respectively. */
12 
13 /* Call FLOGR on INPUT. The results are returned through the parms. */
14 
15 /* R2 != R1 && R2 != R1 + 1 */
16 void
17 flogr1(unsigned long input, unsigned long *bitpos, unsigned long *modval,
18        unsigned int *cc)
19 {
20    unsigned int psw;
21    register unsigned long value asm("4") = input;
22 
23    asm volatile ( FLOGR(2,4)
24                   "ipm   %[psw]\n\t"
25                   "stg   2, %[bitpos]\n\t"
26                   "stg   3, %[modval]\n\t"
27                   : [bitpos]"=m"(*bitpos), [modval]"=m"(*modval),
28                     [psw]"=&d"(psw)
29                   : [val] "d"(value)
30                   : "2", "3", "cc");
31 
32    *cc = psw >> 28;
33 #if 0
34    printf("value = %lx,  bitpos = %lu,  modval = %lx,  cc = %d\n",
35           value, *bitpos, *modval, *cc);
36 #endif
37 }
38 
39 /* R2 == R1 */
40 void
41 flogr2(unsigned long input, unsigned long *bitpos, unsigned long *modval,
42        unsigned int *cc)
43 {
44    unsigned int psw;
45    register unsigned long value asm("2") = input;
46 
47    asm volatile ( FLOGR(2,2)
48                   "ipm   %[psw]\n\t"
49                   "stg   2, %[bitpos]\n\t"
50                   "stg   3, %[modval]\n\t"
51                   : [bitpos]"=m"(*bitpos), [modval]"=m"(*modval),
52                     [psw]"=&d"(psw), [val] "+d"(value)
53                   :
54                   : "3", "cc");
55 
56    *cc = psw >> 28;
57 #if 0
58    printf("value = %lx,  bitpos = %lu,  modval = %lx,  cc = %d\n",
59           value, *bitpos, *modval, *cc);
60 #endif
61 }
62 
63 /* R2 == R1 + 1 */
64 void
65 flogr3(unsigned long input, unsigned long *bitpos, unsigned long *modval,
66        unsigned int *cc)
67 {
68    unsigned int psw;
69    register unsigned long value asm("3") = input;
70 
71    asm volatile ( FLOGR(2,3)
72                   "ipm   %[psw]\n\t"
73                   "stg   2, %[bitpos]\n\t"
74                   "stg   3, %[modval]\n\t"
75                   : [bitpos]"=m"(*bitpos), [modval]"=m"(*modval),
76                     [psw]"=&d"(psw), [val] "+d"(value)
77                   :
78                   : "2", "cc");
79 
80    *cc = psw >> 28;
81 #if 0
82    printf("value = %lx,  bitpos = %lu,  modval = %lx,  cc = %d\n",
83           value, *bitpos, *modval, *cc);
84 #endif
85 }
86 
87 void
88 runtest(void (*func)(unsigned long, unsigned long *, unsigned long *,
89                      unsigned int *))
90 {
91    unsigned long bitpos, modval, value;
92    unsigned int cc;
93    int i;
94 
95    /* Value 0 is special */
96    value = 0;
97    func(value, &bitpos, &modval, &cc);
98    if (modval != 0)  fprintf(stderr, "modval is wrong for %lx\n", value);
99    if (bitpos != 64) fprintf(stderr, "bitpos is wrong for %lx\n", value);
100    if (cc != 0)      fprintf(stderr, "cc is wrong for %lx\n", value);
101 
102    /* Test with exactly 1 bit set */
103    for (i = 0; i < 64; ++i) {
104      value = 1ull << i;
105      func(value, &bitpos, &modval, &cc);
106      if (modval != 0) fprintf(stderr, "modval is wrong for %lx\n", value);
107      if (bitpos != 63 - i) fprintf(stderr, "bitpos is wrong for %lx\n", value);
108      if (cc != 2)          fprintf(stderr, "cc is wrong for %lx\n", value);
109    }
110 
111    /* Test with all bits 1 right from first 1 bit */
112    for (i = 1; i < 64; ++i) {
113      value = 1ull << i;
114      value = value | (value - 1);
115      func(value, &bitpos, &modval, &cc);
116      if (modval != (value >> 1)) fprintf(stderr, "modval is wrong for %lx\n", value);
117      if (bitpos != 63 - i) fprintf(stderr, "bitpos is wrong for %lx\n", value);
118      if (cc != 2)          fprintf(stderr, "cc is wrong for %lx\n", value);
119    }
120 }
121 
122 
123 int main()
124 {
125    runtest(flogr1);
126    runtest(flogr2);
127    runtest(flogr3);
128 
129    return 0;
130 }
131