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 Hosekstatic 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 HosekCOMPILER_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