1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25#include "qemu/osdep.h"
26#include "qemu/host-utils.h"
27#include "qemu/int128.h"
28
29#ifndef CONFIG_INT128
30
31
32
33
34
35
36
37
38
39static Int128 divrem128(Int128 u, Int128 v, Int128 *q)
40{
41 Int128 qq;
42 uint64_t hi, lo, tmp;
43 int s = clz64(v.hi);
44
45 if (s == 64) {
46
47 hi = u.hi;
48 lo = u.lo;
49 tmp = divu128(&lo, &hi, v.lo);
50 *q = int128_make128(lo, hi);
51 return int128_make128(tmp, 0);
52 } else {
53 hi = int128_gethi(int128_lshift(v, s));
54
55 if (hi > u.hi) {
56 lo = u.lo;
57 tmp = u.hi;
58 divu128(&lo, &tmp, hi);
59 lo = int128_gethi(int128_lshift(int128_make128(lo, 0), s));
60 } else {
61 lo = u.lo;
62 tmp = u.hi - hi;
63 divu128(&lo, &tmp, hi);
64 lo = int128_gethi(int128_lshift(int128_make128(lo, 1), s));
65 }
66
67 qq = int128_make64(lo);
68
69 tmp = lo * v.hi;
70 mulu64(&lo, &hi, lo, v.lo);
71 hi += tmp;
72
73 if (hi < tmp
74 || hi > u.hi
75 || (hi == u.hi && lo > u.lo)) {
76 qq.lo -= 1;
77 mulu64(&lo, &hi, qq.lo, v.lo);
78 hi += qq.lo * v.hi;
79 }
80
81 *q = qq;
82 u.hi -= hi + (u.lo < lo);
83 u.lo -= lo;
84 return u;
85 }
86}
87
88Int128 int128_divu(Int128 a, Int128 b)
89{
90 Int128 q;
91 divrem128(a, b, &q);
92 return q;
93}
94
95Int128 int128_remu(Int128 a, Int128 b)
96{
97 Int128 q;
98 return divrem128(a, b, &q);
99}
100
101Int128 int128_divs(Int128 a, Int128 b)
102{
103 Int128 q;
104 bool sgna = !int128_nonneg(a);
105 bool sgnb = !int128_nonneg(b);
106
107 if (sgna) {
108 a = int128_neg(a);
109 }
110
111 if (sgnb) {
112 b = int128_neg(b);
113 }
114
115 divrem128(a, b, &q);
116
117 if (sgna != sgnb) {
118 q = int128_neg(q);
119 }
120
121 return q;
122}
123
124Int128 int128_rems(Int128 a, Int128 b)
125{
126 Int128 q, r;
127 bool sgna = !int128_nonneg(a);
128 bool sgnb = !int128_nonneg(b);
129
130 if (sgna) {
131 a = int128_neg(a);
132 }
133
134 if (sgnb) {
135 b = int128_neg(b);
136 }
137
138 r = divrem128(a, b, &q);
139
140 if (sgna) {
141 r = int128_neg(r);
142 }
143
144 return r;
145}
146
147#endif
148