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