1 //===-- lib/builtins/ppc/floattitf.c - Convert int128->long double -*-C -*-===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 // This file implements converting a signed 128 bit integer to a 128bit IBM /
10 // PowerPC long double (double-double) value.
12 //===----------------------------------------------------------------------===//
16 // Conversions from signed and unsigned 64-bit int to long double.
17 long double __floatditf(int64_t);
18 long double __floatunditf(uint64_t);
20 // Convert a signed 128-bit integer to long double.
21 // This uses the following property: Let hi and lo be 64-bits each,
22 // and let signed_val_k() and unsigned_val_k() be the value of the
23 // argument interpreted as a signed or unsigned k-bit integer. Then,
25 // signed_val_128(hi,lo) = signed_val_64(hi) * 2^64 + unsigned_val_64(lo)
26 // = (long double)hi * 2^64 + (long double)lo,
28 // where (long double)hi and (long double)lo are signed and
29 // unsigned 64-bit integer to long double conversions, respectively.
30 long double __floattitf(__int128_t arg
) {
31 // Split the int128 argument into 64-bit high and low int64 parts.
32 int64_t ArgHiPart
= (int64_t)(arg
>> 64);
33 uint64_t ArgLoPart
= (uint64_t)arg
;
35 // Convert each 64-bit part into long double. The high part
36 // must be a signed conversion and the low part an unsigned conversion
37 // to ensure the correct result.
38 long double ConvertedHiPart
= __floatditf(ArgHiPart
);
39 long double ConvertedLoPart
= __floatunditf(ArgLoPart
);
41 // The low bit of ArgHiPart corresponds to the 2^64 bit in arg.
42 // Multiply the high part by 2^64 to undo the right shift by 64-bits
43 // done in the splitting. Then, add to the low part to obtain the
45 return ((ConvertedHiPart
* 0x1.0p64
) + ConvertedLoPart
);