blob: bc65c60098be9f5e3d902f3c5287b80dfc1e6d21 [file] [log] [blame]
metzman40132972019-01-09 21:46:09 +00001//===- FuzzerBuiltinsMSVC.h - Internal header for builtins ------*- C++ -* ===//
2//
chandlerc40284492019-01-19 08:50:56 +00003// 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
metzman40132972019-01-09 21:46:09 +00006//
7//===----------------------------------------------------------------------===//
8// Wrapper functions and marcos that use intrinsics instead of builtin functions
9// which cannot be compiled by MSVC.
10//===----------------------------------------------------------------------===//
11
12#ifndef LLVM_FUZZER_BUILTINS_MSVC_H
13#define LLVM_FUZZER_BUILTINS_MSVC_H
14
15#include "FuzzerDefs.h"
16
17#if LIBFUZZER_MSVC
metzman40132972019-01-09 21:46:09 +000018#include <intrin.h>
19#include <cstdint>
20#include <cstdlib>
21
22// __builtin_return_address() cannot be compiled with MSVC. Use the equivalent
23// from <intrin.h>
metzman1c6a8592019-01-22 18:59:25 +000024#define GET_CALLER_PC() _ReturnAddress()
metzman40132972019-01-09 21:46:09 +000025
26namespace fuzzer {
27
28inline uint8_t Bswap(uint8_t x) { return x; }
29// Use alternatives to __builtin functions from <stdlib.h> and <intrin.h> on
30// Windows since the builtins are not supported by MSVC.
31inline uint16_t Bswap(uint16_t x) { return _byteswap_ushort(x); }
32inline uint32_t Bswap(uint32_t x) { return _byteswap_ulong(x); }
33inline uint64_t Bswap(uint64_t x) { return _byteswap_uint64(x); }
34
35// The functions below were mostly copied from
36// compiler-rt/lib/builtins/int_lib.h which defines the __builtin functions used
37// outside of Windows.
38inline uint32_t Clzll(uint64_t X) {
39 unsigned long LeadZeroIdx = 0;
mcgov6f7c7952019-08-22 20:44:34 +000040
41#if !defined(_M_ARM) && !defined(_M_X64)
42 // Scan the high 32 bits.
43 if (_BitScanReverse(&LeadZeroIdx, static_cast<unsigned long>(X >> 32)))
44 return static_cast<int>(63 - (LeadZeroIdx + 32)); // Create a bit offset from the MSB.
45 // Scan the low 32 bits.
46 if (_BitScanReverse(&LeadZeroIdx, static_cast<unsigned long>(X)))
47 return static_cast<int>(63 - LeadZeroIdx);
48
49#else
metzman40132972019-01-09 21:46:09 +000050 if (_BitScanReverse64(&LeadZeroIdx, X)) return 63 - LeadZeroIdx;
mcgov6f7c7952019-08-22 20:44:34 +000051#endif
metzman40132972019-01-09 21:46:09 +000052 return 64;
53}
54
55inline uint32_t Clz(uint32_t X) {
56 unsigned long LeadZeroIdx = 0;
57 if (_BitScanReverse(&LeadZeroIdx, X)) return 31 - LeadZeroIdx;
58 return 32;
59}
60
mcgov6f7c7952019-08-22 20:44:34 +000061inline int Popcountll(unsigned long long X) {
62#if !defined(_M_ARM) && !defined(_M_X64)
63 return __popcnt(X) + __popcnt(X >> 32);
64#else
65 return __popcnt64(X);
66#endif
67}
metzman40132972019-01-09 21:46:09 +000068
69} // namespace fuzzer
70
71#endif // LIBFUZER_MSVC
72#endif // LLVM_FUZZER_BUILTINS_MSVC_H