2 * Copyright © 2013 Keith Packard <keithp@keithp.com>
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation; version 2 of the License.
8 * This program is distributed in the hope that it will be useful, but
9 * WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
11 * General Public License for more details.
13 * You should have received a copy of the GNU General Public License along
14 * with this program; if not, write to the Free Software Foundation, Inc.,
15 * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
20 __pdata ao_int64_t *__data ao_64r, *__data ao_64a, *__data ao_64b;
22 void ao_plus64(__pdata ao_int64_t *r, __pdata ao_int64_t *a, __pdata ao_int64_t *b) __FATTR {
25 r->high = a->high + b->high;
32 void ao_minus64(__pdata ao_int64_t *r, __pdata ao_int64_t *a, __pdata ao_int64_t *b) __FATTR {
35 r->high = a->high - b->high;
42 void ao_rshift64(__pdata ao_int64_t *r, __pdata ao_int64_t *a, uint8_t d) __FATTR {
46 r->low |= a->high << (32 - d);
47 r->high = (int32_t) a->high >> d;
50 r->low = (int32_t) a->high >> d;
55 void ao_lshift64(__pdata ao_int64_t *r, __pdata ao_int64_t *a, uint8_t d) __FATTR {
57 r->high = a->high << d;
59 r->high |= a->low >> (32 - d);
63 r->high = a->low << d;
68 static void ao_umul64_32_32(__ARG ao_int64_t *r, uint32_t a, uint32_t b) __reentrant {
71 r->low = (uint32_t) (uint16_t) a * (uint16_t) b;
72 r->high = (uint32_t) (uint16_t) (a >> 16) * (uint16_t) (b >> 16);
74 s = (uint32_t) (uint16_t) (a >> 16) * (uint16_t) b;
80 s = (uint32_t) (uint16_t) a * (uint16_t) (b >> 16);
87 void ao_neg64(__pdata ao_int64_t *r, __pdata ao_int64_t *a) __FATTR {
89 if (!(r->low = ~a->low + 1))
93 void ao_mul64_32_32(__ARG ao_int64_t *r, int32_t a, int32_t b) __FATTR {
104 ao_umul64_32_32(r, a, b);
109 static void ao_umul64(__ARG ao_int64_t *r, __ARG ao_int64_t *a, __ARG ao_int64_t *b) __reentrant {
110 __LOCAL ao_int64_t r2, r3;
112 ao_umul64_32_32(&r2, a->high, b->low);
113 ao_umul64_32_32(&r3, a->low, b->high);
114 ao_umul64_32_32(r, a->low, b->low);
116 r->high += r2.low + r3.low;
119 static __ARG ao_int64_t ap, bp;
121 void ao_mul64(__ARG ao_int64_t *r, __ARG ao_int64_t *a, __ARG ao_int64_t *b) __FATTR {
122 uint8_t negative = 0;
124 if (ao_int64_negativep(a)) {
129 if (ao_int64_negativep(b)) {
139 static void ao_umul64_64_16(__ARG ao_int64_t *r, __ARG ao_int64_t *a, uint16_t b) __reentrant {
143 ao_umul64_32_32(r, a->low, b);
147 void ao_mul64_64_16(__ARG ao_int64_t *r, __ARG ao_int64_t *a, __ARG uint16_t b) __FATTR {
148 uint8_t negative = 0;
150 if ((int32_t) a->high < 0) {
155 ao_umul64_64_16(r, a, b);