xref: /llvm-project/compiler-rt/lib/builtins/multi3.c (revision 0ba22f51d128bee9d69756c56c4678097270e10b)
1*0ba22f51SPetr Hosek //===-- multi3.c - Implement __multi3 -------------------------------------===//
2*0ba22f51SPetr Hosek //
3*0ba22f51SPetr Hosek // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*0ba22f51SPetr Hosek // See https://llvm.org/LICENSE.txt for license information.
5*0ba22f51SPetr Hosek // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*0ba22f51SPetr Hosek //
7*0ba22f51SPetr Hosek //===----------------------------------------------------------------------===//
8*0ba22f51SPetr Hosek //
9*0ba22f51SPetr Hosek // This file implements __multi3 for the compiler_rt library.
10*0ba22f51SPetr Hosek //
11*0ba22f51SPetr Hosek //===----------------------------------------------------------------------===//
12a6b264b5SAlexey Samsonov 
13a6b264b5SAlexey Samsonov #include "int_lib.h"
14a6b264b5SAlexey Samsonov 
15938b0df7SJoerg Sonnenberger #ifdef CRT_HAS_128BIT
16a6b264b5SAlexey Samsonov 
17*0ba22f51SPetr Hosek // Returns: a * b
18a6b264b5SAlexey Samsonov 
__mulddi3(du_int a,du_int b)19082b89b2SPetr Hosek static ti_int __mulddi3(du_int a, du_int b) {
20a6b264b5SAlexey Samsonov   twords r;
21a6b264b5SAlexey Samsonov   const int bits_in_dword_2 = (int)(sizeof(di_int) * CHAR_BIT) / 2;
22a6b264b5SAlexey Samsonov   const du_int lower_mask = (du_int)~0 >> bits_in_dword_2;
23a6b264b5SAlexey Samsonov   r.s.low = (a & lower_mask) * (b & lower_mask);
24a6b264b5SAlexey Samsonov   du_int t = r.s.low >> bits_in_dword_2;
25a6b264b5SAlexey Samsonov   r.s.low &= lower_mask;
26a6b264b5SAlexey Samsonov   t += (a >> bits_in_dword_2) * (b & lower_mask);
27a6b264b5SAlexey Samsonov   r.s.low += (t & lower_mask) << bits_in_dword_2;
28a6b264b5SAlexey Samsonov   r.s.high = t >> bits_in_dword_2;
29a6b264b5SAlexey Samsonov   t = r.s.low >> bits_in_dword_2;
30a6b264b5SAlexey Samsonov   r.s.low &= lower_mask;
31a6b264b5SAlexey Samsonov   t += (b >> bits_in_dword_2) * (a & lower_mask);
32a6b264b5SAlexey Samsonov   r.s.low += (t & lower_mask) << bits_in_dword_2;
33a6b264b5SAlexey Samsonov   r.s.high += t >> bits_in_dword_2;
34a6b264b5SAlexey Samsonov   r.s.high += (a >> bits_in_dword_2) * (b >> bits_in_dword_2);
35a6b264b5SAlexey Samsonov   return r.all;
36a6b264b5SAlexey Samsonov }
37a6b264b5SAlexey Samsonov 
38*0ba22f51SPetr Hosek // Returns: a * b
39a6b264b5SAlexey Samsonov 
__multi3(ti_int a,ti_int b)40082b89b2SPetr Hosek COMPILER_RT_ABI ti_int __multi3(ti_int a, ti_int b) {
41a6b264b5SAlexey Samsonov   twords x;
42a6b264b5SAlexey Samsonov   x.all = a;
43a6b264b5SAlexey Samsonov   twords y;
44a6b264b5SAlexey Samsonov   y.all = b;
45a6b264b5SAlexey Samsonov   twords r;
46a6b264b5SAlexey Samsonov   r.all = __mulddi3(x.s.low, y.s.low);
47a6b264b5SAlexey Samsonov   r.s.high += x.s.high * y.s.low + x.s.low * y.s.high;
48a6b264b5SAlexey Samsonov   return r.all;
49a6b264b5SAlexey Samsonov }
50a6b264b5SAlexey Samsonov 
51*0ba22f51SPetr Hosek #endif // CRT_HAS_128BIT
52