xref: /freebsd-src/contrib/llvm-project/compiler-rt/lib/builtins/clzdi2.c (revision e25152834cdf3b353892835a4f3b157e066a8ed4)
10b57cec5SDimitry Andric //===-- clzdi2.c - Implement __clzdi2 -------------------------------------===//
20b57cec5SDimitry Andric //
30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
60b57cec5SDimitry Andric //
70b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
80b57cec5SDimitry Andric //
90b57cec5SDimitry Andric // This file implements __clzdi2 for the compiler_rt library.
100b57cec5SDimitry Andric //
110b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
120b57cec5SDimitry Andric 
130b57cec5SDimitry Andric #include "int_lib.h"
140b57cec5SDimitry Andric 
150b57cec5SDimitry Andric // Returns: the number of leading 0-bits
160b57cec5SDimitry Andric 
170b57cec5SDimitry Andric #if !defined(__clang__) &&                                                     \
180b57cec5SDimitry Andric     ((defined(__sparc__) && defined(__arch64__)) || defined(__mips64) ||       \
190b57cec5SDimitry Andric      (defined(__riscv) && __SIZEOF_POINTER__ >= 8))
200b57cec5SDimitry Andric // On 64-bit architectures with neither a native clz instruction nor a native
210b57cec5SDimitry Andric // ctz instruction, gcc resolves __builtin_clz to __clzdi2 rather than
220b57cec5SDimitry Andric // __clzsi2, leading to infinite recursion.
230b57cec5SDimitry Andric #define __builtin_clz(a) __clzsi2(a)
24*5ffd83dbSDimitry Andric extern int __clzsi2(si_int);
250b57cec5SDimitry Andric #endif
260b57cec5SDimitry Andric 
270b57cec5SDimitry Andric // Precondition: a != 0
280b57cec5SDimitry Andric 
__clzdi2(di_int a)29*5ffd83dbSDimitry Andric COMPILER_RT_ABI int __clzdi2(di_int a) {
300b57cec5SDimitry Andric   dwords x;
310b57cec5SDimitry Andric   x.all = a;
320b57cec5SDimitry Andric   const si_int f = -(x.s.high == 0);
33*5ffd83dbSDimitry Andric   return clzsi((x.s.high & ~f) | (x.s.low & f)) +
340b57cec5SDimitry Andric          (f & ((si_int)(sizeof(si_int) * CHAR_BIT)));
350b57cec5SDimitry Andric }
36