2016-06-07 22:24:20 +02:00
|
|
|
#pragma once
|
2018-01-29 22:31:38 +01:00
|
|
|
|
|
|
|
|
#ifdef LLVM_AVAILABLE
|
|
|
|
|
|
|
|
|
|
#include "restore_new.h"
|
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
|
#pragma warning(push, 0)
|
|
|
|
|
#endif
|
|
|
|
|
#include "llvm/IR/LLVMContext.h"
|
|
|
|
|
#include "llvm/IR/IRBuilder.h"
|
|
|
|
|
#include "llvm/IR/Module.h"
|
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
|
#pragma warning(pop)
|
|
|
|
|
#endif
|
|
|
|
|
#include "define_new_memleakdetect.h"
|
|
|
|
|
|
|
|
|
|
#include "../Utilities/types.h"
|
|
|
|
|
#include "../Utilities/StrFmt.h"
|
|
|
|
|
#include "../Utilities/BEType.h"
|
|
|
|
|
#include "../Utilities/BitField.h"
|
|
|
|
|
|
|
|
|
|
#include <unordered_map>
|
|
|
|
|
#include <map>
|
|
|
|
|
#include <unordered_set>
|
|
|
|
|
#include <set>
|
|
|
|
|
#include <array>
|
|
|
|
|
#include <vector>
|
|
|
|
|
|
2018-06-25 12:58:39 +02:00
|
|
|
enum class i2 : char
|
|
|
|
|
{
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
enum class i4 : char
|
|
|
|
|
{
|
|
|
|
|
};
|
|
|
|
|
|
2018-01-29 22:31:38 +01:00
|
|
|
template <typename T = void>
|
|
|
|
|
struct llvm_value_t
|
|
|
|
|
{
|
|
|
|
|
static_assert(std::is_same<T, void>::value, "llvm_value_t<> error: unknown type");
|
|
|
|
|
|
|
|
|
|
using type = void;
|
2018-02-12 17:35:04 +01:00
|
|
|
using base = llvm_value_t;
|
2018-01-29 22:31:38 +01:00
|
|
|
static constexpr uint esize = 0;
|
|
|
|
|
static constexpr bool is_int = false;
|
|
|
|
|
static constexpr bool is_sint = false;
|
|
|
|
|
static constexpr bool is_uint = false;
|
|
|
|
|
static constexpr bool is_float = false;
|
|
|
|
|
static constexpr uint is_vector = false;
|
|
|
|
|
static constexpr uint is_pointer = false;
|
|
|
|
|
|
|
|
|
|
static llvm::Type* get_type(llvm::LLVMContext& context)
|
|
|
|
|
{
|
|
|
|
|
return llvm::Type::getVoidTy(context);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
llvm::Value* eval(llvm::IRBuilder<>* ir) const
|
|
|
|
|
{
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
llvm::Value* value;
|
|
|
|
|
|
|
|
|
|
// llvm_value_t() = default;
|
|
|
|
|
|
|
|
|
|
// llvm_value_t(llvm::Value* value)
|
|
|
|
|
// : value(value)
|
|
|
|
|
// {
|
|
|
|
|
// }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
struct llvm_value_t<bool> : llvm_value_t<void>
|
|
|
|
|
{
|
|
|
|
|
using type = bool;
|
|
|
|
|
using base = llvm_value_t<void>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr uint esize = 1;
|
2018-09-09 17:05:56 +02:00
|
|
|
static constexpr bool is_int = true;
|
2018-01-29 22:31:38 +01:00
|
|
|
|
|
|
|
|
static llvm::Type* get_type(llvm::LLVMContext& context)
|
|
|
|
|
{
|
|
|
|
|
return llvm::Type::getInt1Ty(context);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2018-06-25 12:58:39 +02:00
|
|
|
template <>
|
|
|
|
|
struct llvm_value_t<i2> : llvm_value_t<void>
|
|
|
|
|
{
|
|
|
|
|
using type = i2;
|
|
|
|
|
using base = llvm_value_t<void>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr uint esize = 2;
|
2018-09-09 17:05:56 +02:00
|
|
|
static constexpr bool is_int = true;
|
2018-06-25 12:58:39 +02:00
|
|
|
|
|
|
|
|
static llvm::Type* get_type(llvm::LLVMContext& context)
|
|
|
|
|
{
|
|
|
|
|
return llvm::Type::getIntNTy(context, 2);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
struct llvm_value_t<i4> : llvm_value_t<void>
|
|
|
|
|
{
|
|
|
|
|
using type = i4;
|
|
|
|
|
using base = llvm_value_t<void>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr uint esize = 4;
|
2018-09-09 17:05:56 +02:00
|
|
|
static constexpr bool is_int = true;
|
2018-06-25 12:58:39 +02:00
|
|
|
|
|
|
|
|
static llvm::Type* get_type(llvm::LLVMContext& context)
|
|
|
|
|
{
|
|
|
|
|
return llvm::Type::getIntNTy(context, 4);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2018-01-29 22:31:38 +01:00
|
|
|
template <>
|
|
|
|
|
struct llvm_value_t<char> : llvm_value_t<void>
|
|
|
|
|
{
|
|
|
|
|
using type = char;
|
|
|
|
|
using base = llvm_value_t<void>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr uint esize = 8;
|
|
|
|
|
static constexpr bool is_int = true;
|
|
|
|
|
|
|
|
|
|
static llvm::Type* get_type(llvm::LLVMContext& context)
|
|
|
|
|
{
|
|
|
|
|
return llvm::Type::getInt8Ty(context);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
struct llvm_value_t<s8> : llvm_value_t<char>
|
|
|
|
|
{
|
|
|
|
|
using type = s8;
|
|
|
|
|
using base = llvm_value_t<char>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr bool is_sint = true;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
struct llvm_value_t<u8> : llvm_value_t<char>
|
|
|
|
|
{
|
|
|
|
|
using type = u8;
|
|
|
|
|
using base = llvm_value_t<char>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr bool is_uint = true;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
struct llvm_value_t<s16> : llvm_value_t<s8>
|
|
|
|
|
{
|
|
|
|
|
using type = s16;
|
|
|
|
|
using base = llvm_value_t<s8>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr uint esize = 16;
|
|
|
|
|
|
|
|
|
|
static llvm::Type* get_type(llvm::LLVMContext& context)
|
|
|
|
|
{
|
|
|
|
|
return llvm::Type::getInt16Ty(context);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
struct llvm_value_t<u16> : llvm_value_t<s16>
|
|
|
|
|
{
|
|
|
|
|
using type = u16;
|
|
|
|
|
using base = llvm_value_t<s16>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr bool is_sint = false;
|
|
|
|
|
static constexpr bool is_uint = true;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
struct llvm_value_t<s32> : llvm_value_t<s8>
|
|
|
|
|
{
|
|
|
|
|
using type = s32;
|
|
|
|
|
using base = llvm_value_t<s8>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr uint esize = 32;
|
|
|
|
|
|
|
|
|
|
static llvm::Type* get_type(llvm::LLVMContext& context)
|
|
|
|
|
{
|
|
|
|
|
return llvm::Type::getInt32Ty(context);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
struct llvm_value_t<u32> : llvm_value_t<s32>
|
|
|
|
|
{
|
|
|
|
|
using type = u32;
|
|
|
|
|
using base = llvm_value_t<s32>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr bool is_sint = false;
|
|
|
|
|
static constexpr bool is_uint = true;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
struct llvm_value_t<s64> : llvm_value_t<s8>
|
|
|
|
|
{
|
|
|
|
|
using type = s64;
|
|
|
|
|
using base = llvm_value_t<s8>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr uint esize = 64;
|
|
|
|
|
|
|
|
|
|
static llvm::Type* get_type(llvm::LLVMContext& context)
|
|
|
|
|
{
|
|
|
|
|
return llvm::Type::getInt64Ty(context);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
struct llvm_value_t<u64> : llvm_value_t<s64>
|
|
|
|
|
{
|
|
|
|
|
using type = u64;
|
|
|
|
|
using base = llvm_value_t<s64>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr bool is_sint = false;
|
|
|
|
|
static constexpr bool is_uint = true;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
struct llvm_value_t<s128> : llvm_value_t<s8>
|
|
|
|
|
{
|
|
|
|
|
using type = s128;
|
|
|
|
|
using base = llvm_value_t<s8>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr uint esize = 128;
|
|
|
|
|
|
|
|
|
|
static llvm::Type* get_type(llvm::LLVMContext& context)
|
|
|
|
|
{
|
|
|
|
|
return llvm::Type::getIntNTy(context, 128);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
struct llvm_value_t<u128> : llvm_value_t<s128>
|
|
|
|
|
{
|
|
|
|
|
using type = u128;
|
|
|
|
|
using base = llvm_value_t<s128>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr bool is_sint = false;
|
|
|
|
|
static constexpr bool is_uint = true;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
struct llvm_value_t<f32> : llvm_value_t<void>
|
|
|
|
|
{
|
|
|
|
|
using type = f32;
|
|
|
|
|
using base = llvm_value_t<void>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr uint esize = 32;
|
|
|
|
|
static constexpr bool is_float = true;
|
|
|
|
|
|
|
|
|
|
static llvm::Type* get_type(llvm::LLVMContext& context)
|
|
|
|
|
{
|
|
|
|
|
return llvm::Type::getFloatTy(context);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
|
struct llvm_value_t<f64> : llvm_value_t<void>
|
|
|
|
|
{
|
|
|
|
|
using type = f64;
|
|
|
|
|
using base = llvm_value_t<void>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr uint esize = 64;
|
|
|
|
|
static constexpr bool is_float = true;
|
|
|
|
|
|
|
|
|
|
static llvm::Type* get_type(llvm::LLVMContext& context)
|
|
|
|
|
{
|
|
|
|
|
return llvm::Type::getDoubleTy(context);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
struct llvm_value_t<T*> : llvm_value_t<T>
|
|
|
|
|
{
|
|
|
|
|
static_assert(!std::is_void<T>::value, "llvm_value_t<> error: invalid pointer to void type");
|
|
|
|
|
|
|
|
|
|
using type = T*;
|
|
|
|
|
using base = llvm_value_t<T>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
2019-04-17 19:58:04 +02:00
|
|
|
static constexpr uint esize = 64;
|
|
|
|
|
static constexpr bool is_int = false;
|
|
|
|
|
static constexpr bool is_sint = false;
|
|
|
|
|
static constexpr bool is_uint = false;
|
|
|
|
|
static constexpr bool is_float = false;
|
|
|
|
|
static constexpr uint is_vector = false;
|
2018-01-29 22:31:38 +01:00
|
|
|
static constexpr uint is_pointer = llvm_value_t<T>::is_pointer + 1;
|
|
|
|
|
|
|
|
|
|
static llvm::Type* get_type(llvm::LLVMContext& context)
|
|
|
|
|
{
|
|
|
|
|
return llvm_value_t<T>::get_type(context)->getPointerTo();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T, uint N>
|
|
|
|
|
struct llvm_value_t<T[N]> : llvm_value_t<T>
|
|
|
|
|
{
|
|
|
|
|
static_assert(!llvm_value_t<T>::is_vector, "llvm_value_t<> error: invalid multidimensional vector");
|
|
|
|
|
static_assert(!llvm_value_t<T>::is_pointer, "llvm_value_t<>: vector of pointers is not allowed");
|
|
|
|
|
|
|
|
|
|
using type = T[N];
|
|
|
|
|
using base = llvm_value_t<T>;
|
|
|
|
|
using base::base;
|
|
|
|
|
|
|
|
|
|
static constexpr uint is_vector = N;
|
|
|
|
|
static constexpr uint is_pointer = 0;
|
|
|
|
|
|
|
|
|
|
static llvm::Type* get_type(llvm::LLVMContext& context)
|
|
|
|
|
{
|
|
|
|
|
return llvm::VectorType::get(llvm_value_t<T>::get_type(context), N);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T>
|
|
|
|
|
using llvm_expr_t = std::decay_t<T>;
|
|
|
|
|
|
|
|
|
|
template <typename T, typename = void>
|
|
|
|
|
struct is_llvm_expr
|
|
|
|
|
{
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
struct is_llvm_expr<T, std::void_t<decltype(std::declval<T>().eval(std::declval<llvm::IRBuilder<>*>()))>>
|
|
|
|
|
{
|
|
|
|
|
using type = typename std::decay_t<T>::type;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T, typename Of, typename = void>
|
|
|
|
|
struct is_llvm_expr_of
|
|
|
|
|
{
|
|
|
|
|
static constexpr bool ok = false;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T, typename Of>
|
|
|
|
|
struct is_llvm_expr_of<T, Of, std::void_t<typename is_llvm_expr<T>::type, typename is_llvm_expr<Of>::type>>
|
|
|
|
|
{
|
|
|
|
|
static constexpr bool ok = std::is_same_v<typename is_llvm_expr<T>::type, typename is_llvm_expr<Of>::type>;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T, typename... Types>
|
|
|
|
|
using llvm_common_t = std::enable_if_t<(is_llvm_expr_of<T, Types>::ok && ...), typename is_llvm_expr<T>::type>;
|
|
|
|
|
|
|
|
|
|
template <typename T, bool ForceSigned = false>
|
|
|
|
|
struct llvm_const_int
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
using type = T;
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
u64 val;
|
|
|
|
|
|
|
|
|
|
llvm::Value* eval(llvm::IRBuilder<>* ir) const
|
|
|
|
|
{
|
|
|
|
|
static_assert(llvm_value_t<T>::is_int, "llvm_const_int<>: invalid type");
|
2018-01-29 22:31:38 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
return llvm::ConstantInt::get(llvm_value_t<T>::get_type(ir->getContext()), val, ForceSigned || llvm_value_t<T>::is_sint);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename A1, typename A2, typename T = llvm_common_t<A1, A2>>
|
|
|
|
|
struct llvm_add
|
|
|
|
|
{
|
|
|
|
|
using type = T;
|
|
|
|
|
|
|
|
|
|
llvm_expr_t<A1> a1;
|
|
|
|
|
llvm_expr_t<A2> a2;
|
|
|
|
|
|
|
|
|
|
static_assert(llvm_value_t<T>::is_sint || llvm_value_t<T>::is_uint || llvm_value_t<T>::is_float, "llvm_add<>: invalid type");
|
2018-01-29 22:31:38 +01:00
|
|
|
|
|
|
|
|
llvm::Value* eval(llvm::IRBuilder<>* ir) const
|
|
|
|
|
{
|
|
|
|
|
const auto v1 = a1.eval(ir);
|
|
|
|
|
const auto v2 = a2.eval(ir);
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_int)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateAdd(v1, v2);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_float)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateFAdd(v1, v2);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1, typename T2>
|
|
|
|
|
inline llvm_add<T1, T2> operator +(T1&& a1, T2&& a2)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return {a1, a2};
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1>
|
|
|
|
|
inline llvm_add<T1, llvm_const_int<typename is_llvm_expr<T1>::type>> operator +(T1&& a1, u64 c)
|
|
|
|
|
{
|
|
|
|
|
return {a1, {c}};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename A1, typename A2, typename A3, typename T = llvm_common_t<A1, A2, A3>>
|
|
|
|
|
struct llvm_sum
|
2018-05-01 12:21:45 +02:00
|
|
|
{
|
|
|
|
|
using type = T;
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
llvm_expr_t<A1> a1;
|
|
|
|
|
llvm_expr_t<A2> a2;
|
|
|
|
|
llvm_expr_t<A3> a3;
|
2018-05-01 12:21:45 +02:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
static_assert(llvm_value_t<T>::is_sint || llvm_value_t<T>::is_uint, "llvm_sum<>: invalid_type");
|
2018-05-01 12:21:45 +02:00
|
|
|
|
|
|
|
|
llvm::Value* eval(llvm::IRBuilder<>* ir) const
|
|
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
const auto v1 = a1.eval(ir);
|
|
|
|
|
const auto v2 = a2.eval(ir);
|
|
|
|
|
const auto v3 = a3.eval(ir);
|
|
|
|
|
|
|
|
|
|
if constexpr (llvm_value_t<T>::is_int)
|
|
|
|
|
{
|
|
|
|
|
return ir->CreateAdd(ir->CreateAdd(v1, v2), v3);
|
|
|
|
|
}
|
2018-05-01 12:21:45 +02:00
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1, typename T2, typename T3>
|
|
|
|
|
llvm_sum(T1&& a1, T2&& a2, T3&& a3) -> llvm_sum<T1, T2, T3>;
|
2018-05-01 12:21:45 +02:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename A1, typename A2, typename T = llvm_common_t<A1, A2>>
|
|
|
|
|
struct llvm_sub
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
using type = T;
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
llvm_expr_t<A1> a1;
|
|
|
|
|
llvm_expr_t<A2> a2;
|
2018-01-29 22:31:38 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
static_assert(llvm_value_t<T>::is_sint || llvm_value_t<T>::is_uint || llvm_value_t<T>::is_float, "llvm_sub<>: invalid type");
|
2018-01-29 22:31:38 +01:00
|
|
|
|
|
|
|
|
llvm::Value* eval(llvm::IRBuilder<>* ir) const
|
|
|
|
|
{
|
|
|
|
|
const auto v1 = a1.eval(ir);
|
|
|
|
|
const auto v2 = a2.eval(ir);
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_int)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateSub(v1, v2);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_float)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateFSub(v1, v2);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1, typename T2>
|
|
|
|
|
inline llvm_sub<T1, T2> operator -(T1&& a1, T2&& a2)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return {a1, a2};
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1>
|
|
|
|
|
inline llvm_sub<T1, llvm_const_int<typename is_llvm_expr<T1>::type>> operator -(T1&& a1, u64 c)
|
2018-05-01 12:21:45 +02:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
return {a1, {c}};
|
2018-05-01 12:21:45 +02:00
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1>
|
|
|
|
|
inline llvm_sub<llvm_const_int<typename is_llvm_expr<T1>::type>, T1> operator -(u64 c, T1&& a1)
|
2018-05-01 12:21:45 +02:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
return {{c}, a1};
|
2018-05-01 12:21:45 +02:00
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename A1, typename A2, typename T = llvm_common_t<A1, A2>>
|
|
|
|
|
struct llvm_mul
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
using type = T;
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
llvm_expr_t<A1> a1;
|
|
|
|
|
llvm_expr_t<A2> a2;
|
2018-01-29 22:31:38 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
static_assert(llvm_value_t<T>::is_sint || llvm_value_t<T>::is_uint || llvm_value_t<T>::is_float, "llvm_mul<>: invalid type");
|
2018-01-29 22:31:38 +01:00
|
|
|
|
|
|
|
|
llvm::Value* eval(llvm::IRBuilder<>* ir) const
|
|
|
|
|
{
|
|
|
|
|
const auto v1 = a1.eval(ir);
|
|
|
|
|
const auto v2 = a2.eval(ir);
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_int)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateMul(v1, v2);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_float)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateFMul(v1, v2);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1, typename T2>
|
|
|
|
|
inline llvm_mul<T1, T2> operator *(T1&& a1, T2&& a2)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return {a1, a2};
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename A1, typename A2, typename T = llvm_common_t<A1, A2>>
|
|
|
|
|
struct llvm_div
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
using type = T;
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
llvm_expr_t<A1> a1;
|
|
|
|
|
llvm_expr_t<A2> a2;
|
2018-01-29 22:31:38 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
static_assert(llvm_value_t<T>::is_sint || llvm_value_t<T>::is_uint || llvm_value_t<T>::is_float, "llvm_div<>: invalid type");
|
2018-01-29 22:31:38 +01:00
|
|
|
|
|
|
|
|
llvm::Value* eval(llvm::IRBuilder<>* ir) const
|
|
|
|
|
{
|
|
|
|
|
const auto v1 = a1.eval(ir);
|
|
|
|
|
const auto v2 = a2.eval(ir);
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_sint)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateSDiv(v1, v2);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_uint)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateUDiv(v1, v2);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_float)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateFDiv(v1, v2);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1, typename T2>
|
|
|
|
|
inline llvm_div<T1, T2> operator /(T1&& a1, T2&& a2)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return {a1, a2};
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename A1, typename T = llvm_common_t<A1>>
|
|
|
|
|
struct llvm_neg
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
using type = T;
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
llvm_expr_t<A1> a1;
|
2018-01-29 22:31:38 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
static_assert(llvm_value_t<T>::is_sint || llvm_value_t<T>::is_uint || llvm_value_t<T>::is_float, "llvm_neg<>: invalid type");
|
2018-01-29 22:31:38 +01:00
|
|
|
|
|
|
|
|
llvm::Value* eval(llvm::IRBuilder<>* ir) const
|
|
|
|
|
{
|
|
|
|
|
const auto v1 = a1.eval(ir);
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_int)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateNeg(v1);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_float)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateFNeg(v1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1>
|
|
|
|
|
inline llvm_neg<T1> operator -(T1 a1)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return {a1};
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename A1, typename A2, typename T = llvm_common_t<A1, A2>>
|
|
|
|
|
struct llvm_shl
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
using type = T;
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
llvm_expr_t<A1> a1;
|
|
|
|
|
llvm_expr_t<A2> a2;
|
2018-01-29 22:31:38 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
static_assert(llvm_value_t<T>::is_sint || llvm_value_t<T>::is_uint, "llvm_shl<>: invalid type");
|
2018-01-29 22:31:38 +01:00
|
|
|
|
|
|
|
|
llvm::Value* eval(llvm::IRBuilder<>* ir) const
|
|
|
|
|
{
|
|
|
|
|
const auto v1 = a1.eval(ir);
|
|
|
|
|
const auto v2 = a2.eval(ir);
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_sint)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateShl(v1, v2);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_uint)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateShl(v1, v2);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1, typename T2>
|
|
|
|
|
inline llvm_shl<T1, T2> operator <<(T1&& a1, T2&& a2)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return {a1, a2};
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1>
|
|
|
|
|
inline llvm_shl<T1, llvm_const_int<typename is_llvm_expr<T1>::type>> operator <<(T1&& a1, u64 c)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
return {a1, {c}};
|
2018-01-29 22:31:38 +01:00
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename A1, typename A2, typename T = llvm_common_t<A1, A2>>
|
|
|
|
|
struct llvm_shr
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
using type = T;
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
llvm_expr_t<A1> a1;
|
|
|
|
|
llvm_expr_t<A2> a2;
|
2018-01-29 22:31:38 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
static_assert(llvm_value_t<T>::is_sint || llvm_value_t<T>::is_uint, "llvm_shr<>: invalid type");
|
2018-01-29 22:31:38 +01:00
|
|
|
|
|
|
|
|
llvm::Value* eval(llvm::IRBuilder<>* ir) const
|
|
|
|
|
{
|
|
|
|
|
const auto v1 = a1.eval(ir);
|
|
|
|
|
const auto v2 = a2.eval(ir);
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_sint)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateAShr(v1, v2);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_uint)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateLShr(v1, v2);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1, typename T2>
|
|
|
|
|
inline llvm_shr<T1, T2> operator >>(T1&& a1, T2&& a2)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return {a1, a2};
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1>
|
|
|
|
|
inline llvm_shr<T1, llvm_const_int<typename is_llvm_expr<T1>::type>> operator >>(T1&& a1, u64 c)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
return {a1, {c}};
|
2018-01-29 22:31:38 +01:00
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename A1, typename A2, typename T = llvm_common_t<A1, A2>>
|
|
|
|
|
struct llvm_and
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
using type = T;
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
llvm_expr_t<A1> a1;
|
|
|
|
|
llvm_expr_t<A2> a2;
|
2018-01-29 22:31:38 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
static_assert(llvm_value_t<T>::is_int, "llvm_and<>: invalid type");
|
2018-01-29 22:31:38 +01:00
|
|
|
|
|
|
|
|
llvm::Value* eval(llvm::IRBuilder<>* ir) const
|
|
|
|
|
{
|
|
|
|
|
const auto v1 = a1.eval(ir);
|
|
|
|
|
const auto v2 = a2.eval(ir);
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_int)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateAnd(v1, v2);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1, typename T2>
|
|
|
|
|
inline llvm_and<T1, T2> operator &(T1&& a1, T2&& a2)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return {a1, a2};
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1>
|
|
|
|
|
inline llvm_and<T1, llvm_const_int<typename is_llvm_expr<T1>::type>> operator &(T1&& a1, u64 c)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
return {a1, {c}};
|
2018-01-29 22:31:38 +01:00
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename A1, typename A2, typename T = llvm_common_t<A1, A2>>
|
|
|
|
|
struct llvm_or
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
using type = T;
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
llvm_expr_t<A1> a1;
|
|
|
|
|
llvm_expr_t<A2> a2;
|
2018-01-29 22:31:38 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
static_assert(llvm_value_t<T>::is_int, "llvm_or<>: invalid type");
|
2018-01-29 22:31:38 +01:00
|
|
|
|
|
|
|
|
llvm::Value* eval(llvm::IRBuilder<>* ir) const
|
|
|
|
|
{
|
|
|
|
|
const auto v1 = a1.eval(ir);
|
|
|
|
|
const auto v2 = a2.eval(ir);
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_int)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateOr(v1, v2);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1, typename T2>
|
|
|
|
|
inline llvm_or<T1, T2> operator |(T1&& a1, T2&& a2)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return {a1, a2};
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1>
|
|
|
|
|
inline llvm_or<T1, llvm_const_int<typename is_llvm_expr<T1>::type>> operator |(T1&& a1, u64 c)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
return {a1, {c}};
|
2018-01-29 22:31:38 +01:00
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename A1, typename A2, typename T = llvm_common_t<A1, A2>>
|
|
|
|
|
struct llvm_xor
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
using type = T;
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
llvm_expr_t<A1> a1;
|
|
|
|
|
llvm_expr_t<A2> a2;
|
2018-01-29 22:31:38 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
static_assert(llvm_value_t<T>::is_int, "llvm_xor<>: invalid type");
|
2018-01-29 22:31:38 +01:00
|
|
|
|
|
|
|
|
llvm::Value* eval(llvm::IRBuilder<>* ir) const
|
|
|
|
|
{
|
|
|
|
|
const auto v1 = a1.eval(ir);
|
|
|
|
|
const auto v2 = a2.eval(ir);
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_int)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return ir->CreateXor(v1, v2);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1, typename T2>
|
|
|
|
|
inline llvm_xor<T1, T2> operator ^(T1&& a1, T2&& a2)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
|
|
|
|
return {a1, a2};
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1>
|
|
|
|
|
inline llvm_xor<T1, llvm_const_int<typename is_llvm_expr<T1>::type>> operator ^(T1&& a1, u64 c)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
return {a1, {c}};
|
2018-01-29 22:31:38 +01:00
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1>
|
|
|
|
|
inline llvm_xor<T1, llvm_const_int<typename is_llvm_expr<T1>::type, true>> operator ~(T1&& a1)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
return {a1, {UINT64_MAX}};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename A1, typename A2, llvm::CmpInst::Predicate UPred, typename T = llvm_common_t<A1, A2>>
|
|
|
|
|
struct llvm_cmp
|
|
|
|
|
{
|
|
|
|
|
using type = std::conditional_t<llvm_value_t<T>::is_vector != 0, bool[llvm_value_t<T>::is_vector], bool>;
|
2018-01-29 22:31:38 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
static constexpr bool is_float = llvm_value_t<T>::is_float;
|
|
|
|
|
|
|
|
|
|
llvm_expr_t<A1> a1;
|
|
|
|
|
llvm_expr_t<A2> a2;
|
|
|
|
|
|
|
|
|
|
static_assert(llvm_value_t<T>::is_int || is_float, "llvm_cmp<>: invalid type");
|
|
|
|
|
|
|
|
|
|
// Convert unsigned comparison predicate to signed if necessary
|
|
|
|
|
static constexpr llvm::CmpInst::Predicate pred = llvm_value_t<T>::is_uint ? UPred :
|
|
|
|
|
UPred == llvm::ICmpInst::ICMP_UGT ? llvm::ICmpInst::ICMP_SGT :
|
|
|
|
|
UPred == llvm::ICmpInst::ICMP_UGE ? llvm::ICmpInst::ICMP_SGE :
|
|
|
|
|
UPred == llvm::ICmpInst::ICMP_ULT ? llvm::ICmpInst::ICMP_SLT :
|
|
|
|
|
UPred == llvm::ICmpInst::ICMP_ULE ? llvm::ICmpInst::ICMP_SLE : UPred;
|
2018-01-29 22:31:38 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
static_assert(llvm_value_t<T>::is_sint || llvm_value_t<T>::is_uint || is_float || UPred == llvm::ICmpInst::ICMP_EQ || UPred == llvm::ICmpInst::ICMP_NE, "llvm_cmp<>: invalid operation on sign-undefined type");
|
2018-01-29 22:31:38 +01:00
|
|
|
|
|
|
|
|
llvm::Value* eval(llvm::IRBuilder<>* ir) const
|
|
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
static_assert(!is_float, "llvm_cmp<>: invalid operation (missing fcmp_ord or fcmp_uno)");
|
|
|
|
|
|
2018-01-29 22:31:38 +01:00
|
|
|
const auto v1 = a1.eval(ir);
|
2019-04-17 02:00:53 +02:00
|
|
|
const auto v2 = a2.eval(ir);
|
2018-01-29 22:31:38 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
if constexpr (llvm_value_t<T>::is_int)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
return ir->CreateICmp(pred, v1, v2);
|
2018-01-29 22:31:38 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T>
|
|
|
|
|
struct is_llvm_cmp : std::bool_constant<false>
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
};
|
2018-01-29 22:31:38 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename A1, typename A2, auto UPred, typename T>
|
|
|
|
|
struct is_llvm_cmp<llvm_cmp<A1, A2, UPred, T>> : std::bool_constant<true>
|
2018-02-12 17:35:04 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
};
|
2018-02-12 17:35:04 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename Cmp, typename T = llvm_common_t<Cmp>>
|
|
|
|
|
struct llvm_ord
|
|
|
|
|
{
|
|
|
|
|
using base = std::decay_t<Cmp>;
|
|
|
|
|
using type = typename base::type;
|
2018-02-12 17:35:04 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
llvm_expr_t<Cmp> cmp;
|
2018-02-12 17:35:04 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
// Convert comparison predicate to ordered
|
|
|
|
|
static constexpr llvm::CmpInst::Predicate pred =
|
|
|
|
|
base::pred == llvm::ICmpInst::ICMP_EQ ? llvm::ICmpInst::FCMP_OEQ :
|
|
|
|
|
base::pred == llvm::ICmpInst::ICMP_NE ? llvm::ICmpInst::FCMP_ONE :
|
|
|
|
|
base::pred == llvm::ICmpInst::ICMP_SGT ? llvm::ICmpInst::FCMP_OGT :
|
|
|
|
|
base::pred == llvm::ICmpInst::ICMP_SGE ? llvm::ICmpInst::FCMP_OGE :
|
|
|
|
|
base::pred == llvm::ICmpInst::ICMP_SLT ? llvm::ICmpInst::FCMP_OLT :
|
|
|
|
|
base::pred == llvm::ICmpInst::ICMP_SLE ? llvm::ICmpInst::FCMP_OLE : base::pred;
|
2018-02-12 17:35:04 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
static_assert(base::is_float, "llvm_ord<>: invalid type");
|
2018-06-25 12:58:39 +02:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
llvm::Value* eval(llvm::IRBuilder<>* ir) const
|
2018-02-12 17:35:04 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
const auto v1 = cmp.a1.eval(ir);
|
|
|
|
|
const auto v2 = cmp.a2.eval(ir);
|
|
|
|
|
return ir->CreateFCmp(pred, v1, v2);
|
2018-02-12 17:35:04 +01:00
|
|
|
}
|
2019-04-17 02:00:53 +02:00
|
|
|
};
|
2018-02-12 17:35:04 +01:00
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T>
|
|
|
|
|
llvm_ord(T&&) -> llvm_ord<std::enable_if_t<is_llvm_cmp<std::decay_t<T>>::value, T&&>>;
|
|
|
|
|
|
|
|
|
|
template <typename Cmp, typename T = llvm_common_t<Cmp>>
|
|
|
|
|
struct llvm_uno
|
|
|
|
|
{
|
|
|
|
|
using base = std::decay_t<Cmp>;
|
|
|
|
|
using type = typename base::type;
|
|
|
|
|
|
|
|
|
|
llvm_expr_t<Cmp> cmp;
|
|
|
|
|
|
|
|
|
|
// Convert comparison predicate to unordered
|
|
|
|
|
static constexpr llvm::CmpInst::Predicate pred =
|
|
|
|
|
base::pred == llvm::ICmpInst::ICMP_EQ ? llvm::ICmpInst::FCMP_UEQ :
|
|
|
|
|
base::pred == llvm::ICmpInst::ICMP_NE ? llvm::ICmpInst::FCMP_UNE :
|
|
|
|
|
base::pred == llvm::ICmpInst::ICMP_SGT ? llvm::ICmpInst::FCMP_UGT :
|
|
|
|
|
base::pred == llvm::ICmpInst::ICMP_SGE ? llvm::ICmpInst::FCMP_UGE :
|
|
|
|
|
base::pred == llvm::ICmpInst::ICMP_SLT ? llvm::ICmpInst::FCMP_ULT :
|
|
|
|
|
base::pred == llvm::ICmpInst::ICMP_SLE ? llvm::ICmpInst::FCMP_ULE : base::pred;
|
|
|
|
|
|
|
|
|
|
static_assert(base::is_float, "llvm_uno<>: invalid type");
|
2018-02-12 17:35:04 +01:00
|
|
|
|
|
|
|
|
llvm::Value* eval(llvm::IRBuilder<>* ir) const
|
|
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
const auto v1 = cmp.a1.eval(ir);
|
|
|
|
|
const auto v2 = cmp.a2.eval(ir);
|
|
|
|
|
return ir->CreateFCmp(pred, v1, v2);
|
2018-02-12 17:35:04 +01:00
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T>
|
|
|
|
|
llvm_uno(T&&) -> llvm_uno<std::enable_if_t<is_llvm_cmp<std::decay_t<T>>::value, T&&>>;
|
|
|
|
|
|
|
|
|
|
template <typename T1, typename T2>
|
|
|
|
|
inline llvm_cmp<T1, T2, llvm::ICmpInst::ICMP_EQ> operator ==(T1&& a1, T2&& a2)
|
2018-02-12 17:35:04 +01:00
|
|
|
{
|
|
|
|
|
return {a1, a2};
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1>
|
|
|
|
|
inline llvm_cmp<T1, llvm_const_int<typename is_llvm_expr<T1>::type>, llvm::ICmpInst::ICMP_EQ> operator ==(T1&& a1, u64 c)
|
2018-02-12 17:35:04 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
return {a1, {c}};
|
2018-02-12 17:35:04 +01:00
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1, typename T2>
|
|
|
|
|
inline llvm_cmp<T1, T2, llvm::ICmpInst::ICMP_NE> operator !=(T1&& a1, T2&& a2)
|
2018-02-12 17:35:04 +01:00
|
|
|
{
|
|
|
|
|
return {a1, a2};
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1>
|
|
|
|
|
inline llvm_cmp<T1, llvm_const_int<typename is_llvm_expr<T1>::type>, llvm::ICmpInst::ICMP_NE> operator !=(T1&& a1, u64 c)
|
2018-02-12 17:35:04 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
return {a1, {c}};
|
2018-02-12 17:35:04 +01:00
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1, typename T2>
|
|
|
|
|
inline llvm_cmp<T1, T2, llvm::ICmpInst::ICMP_UGT> operator >(T1&& a1, T2&& a2)
|
2018-02-12 17:35:04 +01:00
|
|
|
{
|
|
|
|
|
return {a1, a2};
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1>
|
|
|
|
|
inline llvm_cmp<T1, llvm_const_int<typename is_llvm_expr<T1>::type>, llvm::ICmpInst::ICMP_UGT> operator >(T1&& a1, u64 c)
|
2018-02-12 17:35:04 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
return {a1, {c}};
|
2018-02-12 17:35:04 +01:00
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1, typename T2>
|
|
|
|
|
inline llvm_cmp<T1, T2, llvm::ICmpInst::ICMP_UGE> operator >=(T1&& a1, T2&& a2)
|
2018-02-12 17:35:04 +01:00
|
|
|
{
|
|
|
|
|
return {a1, a2};
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1>
|
|
|
|
|
inline llvm_cmp<T1, llvm_const_int<typename is_llvm_expr<T1>::type>, llvm::ICmpInst::ICMP_UGE> operator >=(T1&& a1, u64 c)
|
2018-02-12 17:35:04 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
return {a1, {c}};
|
2018-02-12 17:35:04 +01:00
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1, typename T2>
|
|
|
|
|
inline llvm_cmp<T1, T2, llvm::ICmpInst::ICMP_ULT> operator <(T1&& a1, T2&& a2)
|
2018-02-12 17:35:04 +01:00
|
|
|
{
|
|
|
|
|
return {a1, a2};
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1>
|
|
|
|
|
inline llvm_cmp<T1, llvm_const_int<typename is_llvm_expr<T1>::type>, llvm::ICmpInst::ICMP_ULT> operator <(T1&& a1, u64 c)
|
2018-02-12 17:35:04 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
return {a1, {c}};
|
2018-02-12 17:35:04 +01:00
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1, typename T2>
|
|
|
|
|
inline llvm_cmp<T1, T2, llvm::ICmpInst::ICMP_ULE> operator <=(T1&& a1, T2&& a2)
|
2018-02-12 17:35:04 +01:00
|
|
|
{
|
|
|
|
|
return {a1, a2};
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T1>
|
|
|
|
|
inline llvm_cmp<T1, llvm_const_int<typename is_llvm_expr<T1>::type>, llvm::ICmpInst::ICMP_ULE> operator <=(T1&& a1, u64 c)
|
2018-02-12 17:35:04 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
return {a1, {c}};
|
2018-02-12 17:35:04 +01:00
|
|
|
}
|
|
|
|
|
|
2018-01-29 22:31:38 +01:00
|
|
|
class cpu_translator
|
|
|
|
|
{
|
|
|
|
|
protected:
|
2018-05-01 12:21:45 +02:00
|
|
|
cpu_translator(llvm::Module* module, bool is_be);
|
2018-01-29 22:31:38 +01:00
|
|
|
|
|
|
|
|
// LLVM context
|
2018-05-01 12:21:45 +02:00
|
|
|
std::reference_wrapper<llvm::LLVMContext> m_context;
|
2018-01-29 22:31:38 +01:00
|
|
|
|
|
|
|
|
// Module to which all generated code is output to
|
2018-05-01 12:21:45 +02:00
|
|
|
llvm::Module* m_module;
|
2018-01-29 22:31:38 +01:00
|
|
|
|
|
|
|
|
// Endianness, affects vector element numbering (TODO)
|
2018-05-01 12:21:45 +02:00
|
|
|
bool m_is_be;
|
2018-01-29 22:31:38 +01:00
|
|
|
|
2018-06-28 15:21:08 +02:00
|
|
|
// Allow PSHUFB intrinsic
|
|
|
|
|
bool m_use_ssse3;
|
|
|
|
|
|
2018-01-29 22:31:38 +01:00
|
|
|
// IR builder
|
|
|
|
|
llvm::IRBuilder<>* m_ir;
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
// Convert a C++ type to an LLVM type (TODO: remove)
|
|
|
|
|
template <typename T>
|
|
|
|
|
llvm::Type* GetType()
|
|
|
|
|
{
|
|
|
|
|
return llvm_value_t<T>::get_type(m_context);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
llvm::Type* get_type()
|
|
|
|
|
{
|
|
|
|
|
return llvm_value_t<T>::get_type(m_context);
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-25 19:31:16 +01:00
|
|
|
template <typename R, typename... Args>
|
|
|
|
|
llvm::FunctionType* get_ftype()
|
|
|
|
|
{
|
|
|
|
|
return llvm::FunctionType::get(get_type<R>(), {get_type<Args>()...}, false);
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-29 22:31:38 +01:00
|
|
|
template <typename T>
|
|
|
|
|
using value_t = llvm_value_t<T>;
|
|
|
|
|
|
2018-07-27 12:00:05 +02:00
|
|
|
template <typename T>
|
|
|
|
|
value_t<T> value(llvm::Value* value)
|
|
|
|
|
{
|
|
|
|
|
if (!value || value->getType() != get_type<T>())
|
|
|
|
|
{
|
|
|
|
|
fmt::throw_exception("cpu_translator::value<>(): invalid value type");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
value_t<T> result;
|
|
|
|
|
result.value = value;
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-29 22:31:38 +01:00
|
|
|
template <typename T>
|
2019-04-17 02:00:53 +02:00
|
|
|
auto eval(T&& expr)
|
2018-01-29 22:31:38 +01:00
|
|
|
{
|
2019-04-17 02:00:53 +02:00
|
|
|
value_t<typename std::decay_t<T>::type> result;
|
2018-01-29 22:31:38 +01:00
|
|
|
result.value = expr.eval(m_ir);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-17 02:00:53 +02:00
|
|
|
template <typename T, typename = std::enable_if_t<is_llvm_cmp<std::decay_t<T>>::value>>
|
|
|
|
|
static auto fcmp_ord(T&& cmp_expr)
|
|
|
|
|
{
|
|
|
|
|
return llvm_ord{std::forward<T>(cmp_expr)};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T, typename = std::enable_if_t<is_llvm_cmp<std::decay_t<T>>::value>>
|
|
|
|
|
static auto fcmp_uno(T&& cmp_expr)
|
|
|
|
|
{
|
|
|
|
|
return llvm_uno{std::forward<T>(cmp_expr)};
|
|
|
|
|
}
|
|
|
|
|
|
2018-02-10 13:57:01 +01:00
|
|
|
template <typename T, typename T2>
|
|
|
|
|
value_t<T> bitcast(T2 expr)
|
|
|
|
|
{
|
|
|
|
|
value_t<T> result;
|
|
|
|
|
result.value = m_ir->CreateBitCast(expr.eval(m_ir), result.get_type(m_context));
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T, typename T2>
|
|
|
|
|
value_t<T> trunc(T2 expr)
|
|
|
|
|
{
|
|
|
|
|
value_t<T> result;
|
|
|
|
|
result.value = m_ir->CreateTrunc(expr.eval(m_ir), result.get_type(m_context));
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T, typename T2>
|
|
|
|
|
value_t<T> sext(T2 expr)
|
|
|
|
|
{
|
|
|
|
|
value_t<T> result;
|
|
|
|
|
result.value = m_ir->CreateSExt(expr.eval(m_ir), result.get_type(m_context));
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T, typename T2>
|
|
|
|
|
value_t<T> zext(T2 expr)
|
|
|
|
|
{
|
|
|
|
|
value_t<T> result;
|
|
|
|
|
result.value = m_ir->CreateZExt(expr.eval(m_ir), result.get_type(m_context));
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-29 22:31:38 +01:00
|
|
|
// Get signed addition overflow into the sign bit (s = a + b)
|
|
|
|
|
template <typename T>
|
|
|
|
|
static inline auto scarry(T a, T b, T s)
|
|
|
|
|
{
|
|
|
|
|
return (b ^ s) & ~(a ^ b);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Bitwise select (c ? a : b)
|
|
|
|
|
template <typename T>
|
|
|
|
|
static inline auto merge(T c, T a, T b)
|
|
|
|
|
{
|
|
|
|
|
return (a & c) | (b & ~c);
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-01 12:21:45 +02:00
|
|
|
// Rotate left
|
|
|
|
|
template <typename T>
|
|
|
|
|
static inline auto rol(T a, T b)
|
|
|
|
|
{
|
|
|
|
|
static constexpr u64 mask = value_t<typename T::type>::esize - 1;
|
|
|
|
|
return a << (b & mask) | a >> (-b & mask);
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-29 14:35:00 +01:00
|
|
|
// Add with saturation
|
2018-05-01 12:21:45 +02:00
|
|
|
template <typename T>
|
2019-03-29 14:35:00 +01:00
|
|
|
inline auto add_sat(T a, T b)
|
2018-05-01 12:21:45 +02:00
|
|
|
{
|
2019-03-29 14:35:00 +01:00
|
|
|
value_t<typename T::type> result;
|
|
|
|
|
const auto eva = a.eval(m_ir);
|
|
|
|
|
const auto evb = b.eval(m_ir);
|
|
|
|
|
|
|
|
|
|
// Compute constant result immediately if possible
|
|
|
|
|
if (llvm::isa<llvm::Constant>(eva) && llvm::isa<llvm::Constant>(evb))
|
|
|
|
|
{
|
|
|
|
|
static_assert(result.is_sint || result.is_uint);
|
|
|
|
|
|
|
|
|
|
if constexpr (result.is_sint)
|
|
|
|
|
{
|
|
|
|
|
llvm::Type* cast_to = m_ir->getIntNTy(result.esize * 2);
|
|
|
|
|
if constexpr (result.is_vector != 0)
|
|
|
|
|
cast_to = llvm::VectorType::get(cast_to, result.is_vector);
|
|
|
|
|
|
|
|
|
|
const auto axt = m_ir->CreateSExt(eva, cast_to);
|
|
|
|
|
const auto bxt = m_ir->CreateSExt(evb, cast_to);
|
|
|
|
|
result.value = m_ir->CreateAdd(axt, bxt);
|
|
|
|
|
const auto _max = m_ir->getInt(llvm::APInt::getSignedMaxValue(result.esize * 2).ashr(result.esize));
|
|
|
|
|
const auto _min = m_ir->getInt(llvm::APInt::getSignedMinValue(result.esize * 2).ashr(result.esize));
|
|
|
|
|
const auto smax = result.is_vector != 0 ? llvm::ConstantVector::getSplat(result.is_vector, _max) : _max;
|
|
|
|
|
const auto smin = result.is_vector != 0 ? llvm::ConstantVector::getSplat(result.is_vector, _min) : _min;
|
|
|
|
|
result.value = m_ir->CreateSelect(m_ir->CreateICmpSGT(result.value, smax), smax, result.value);
|
|
|
|
|
result.value = m_ir->CreateSelect(m_ir->CreateICmpSLT(result.value, smin), smin, result.value);
|
|
|
|
|
result.value = m_ir->CreateTrunc(result.value, result.get_type(m_context));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
const auto _max = m_ir->getInt(llvm::APInt::getMaxValue(result.esize));
|
|
|
|
|
const auto ones = result.is_vector != 0 ? llvm::ConstantVector::getSplat(result.is_vector, _max) : _max;
|
|
|
|
|
result.value = m_ir->CreateAdd(eva, evb);
|
|
|
|
|
result.value = m_ir->CreateSelect(m_ir->CreateICmpULT(result.value, eva), ones, result.value);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
result.value = m_ir->CreateCall(get_intrinsic<typename T::type>(result.is_sint ? llvm::Intrinsic::sadd_sat : llvm::Intrinsic::uadd_sat), {eva, evb});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Subtract with saturation
|
|
|
|
|
template <typename T>
|
|
|
|
|
inline auto sub_sat(T a, T b)
|
|
|
|
|
{
|
|
|
|
|
value_t<typename T::type> result;
|
|
|
|
|
const auto eva = a.eval(m_ir);
|
|
|
|
|
const auto evb = b.eval(m_ir);
|
|
|
|
|
|
|
|
|
|
// Compute constant result immediately if possible
|
|
|
|
|
if (llvm::isa<llvm::Constant>(eva) && llvm::isa<llvm::Constant>(evb))
|
|
|
|
|
{
|
|
|
|
|
static_assert(result.is_sint || result.is_uint);
|
|
|
|
|
|
|
|
|
|
if constexpr (result.is_sint)
|
|
|
|
|
{
|
|
|
|
|
llvm::Type* cast_to = m_ir->getIntNTy(result.esize * 2);
|
|
|
|
|
if constexpr (result.is_vector != 0)
|
|
|
|
|
cast_to = llvm::VectorType::get(cast_to, result.is_vector);
|
|
|
|
|
|
|
|
|
|
const auto axt = m_ir->CreateSExt(eva, cast_to);
|
|
|
|
|
const auto bxt = m_ir->CreateSExt(evb, cast_to);
|
|
|
|
|
result.value = m_ir->CreateSub(axt, bxt);
|
|
|
|
|
const auto _max = m_ir->getInt(llvm::APInt::getSignedMaxValue(result.esize * 2).ashr(result.esize));
|
|
|
|
|
const auto _min = m_ir->getInt(llvm::APInt::getSignedMinValue(result.esize * 2).ashr(result.esize));
|
|
|
|
|
const auto smax = result.is_vector != 0 ? llvm::ConstantVector::getSplat(result.is_vector, _max) : _max;
|
|
|
|
|
const auto smin = result.is_vector != 0 ? llvm::ConstantVector::getSplat(result.is_vector, _min) : _min;
|
|
|
|
|
result.value = m_ir->CreateSelect(m_ir->CreateICmpSGT(result.value, smax), smax, result.value);
|
|
|
|
|
result.value = m_ir->CreateSelect(m_ir->CreateICmpSLT(result.value, smin), smin, result.value);
|
|
|
|
|
result.value = m_ir->CreateTrunc(result.value, result.get_type(m_context));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
const auto _min = m_ir->getInt(llvm::APInt::getMinValue(result.esize));
|
|
|
|
|
const auto zero = result.is_vector != 0 ? llvm::ConstantVector::getSplat(result.is_vector, _min) : _min;
|
|
|
|
|
result.value = m_ir->CreateSub(eva, evb);
|
|
|
|
|
result.value = m_ir->CreateSelect(m_ir->CreateICmpULT(eva, evb), zero, result.value);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
result.value = m_ir->CreateCall(get_intrinsic<typename T::type>(result.is_sint ? llvm::Intrinsic::ssub_sat : llvm::Intrinsic::usub_sat), {eva, evb});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
2018-05-01 12:21:45 +02:00
|
|
|
}
|
|
|
|
|
|
2018-06-27 14:44:06 +02:00
|
|
|
// Average: (a + b + 1) >> 1
|
|
|
|
|
template <typename T>
|
|
|
|
|
inline auto avg(T a, T b)
|
|
|
|
|
{
|
|
|
|
|
//return (a >> 1) + (b >> 1) + ((a | b) & 1);
|
|
|
|
|
|
|
|
|
|
value_t<typename T::type> result;
|
2019-03-29 14:35:00 +01:00
|
|
|
static_assert(result.is_sint || result.is_uint);
|
|
|
|
|
const auto cast_op = result.is_sint ? llvm::Instruction::SExt : llvm::Instruction::ZExt;
|
|
|
|
|
llvm::Type* cast_to = m_ir->getIntNTy(result.esize * 2);
|
|
|
|
|
if constexpr (result.is_vector != 0)
|
|
|
|
|
cast_to = llvm::VectorType::get(cast_to, result.is_vector);
|
|
|
|
|
|
|
|
|
|
const auto axt = m_ir->CreateCast(cast_op, a.eval(m_ir), cast_to);
|
|
|
|
|
const auto bxt = m_ir->CreateCast(cast_op, b.eval(m_ir), cast_to);
|
|
|
|
|
const auto cxt = llvm::ConstantInt::get(cast_to, 1, false);
|
2018-06-27 14:44:06 +02:00
|
|
|
const auto abc = m_ir->CreateAdd(m_ir->CreateAdd(axt, bxt), cxt);
|
|
|
|
|
result.value = m_ir->CreateTrunc(m_ir->CreateLShr(abc, 1), result.get_type(m_context));
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-24 16:18:51 +02:00
|
|
|
// Select (c ? a : b)
|
|
|
|
|
template <typename T, typename T2>
|
|
|
|
|
auto select(T2 c, T a, T b)
|
|
|
|
|
{
|
|
|
|
|
static_assert(value_t<typename T2::type>::esize == 1, "select: expected bool type (first argument)");
|
|
|
|
|
static_assert(value_t<typename T2::type>::is_vector == value_t<typename T::type>::is_vector, "select: incompatible arguments (vectors)");
|
|
|
|
|
T result;
|
|
|
|
|
result.value = m_ir->CreateSelect(c.eval(m_ir), a.eval(m_ir), b.eval(m_ir));
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-01 12:21:45 +02:00
|
|
|
template <typename T, typename E>
|
|
|
|
|
auto insert(T v, u64 i, E e)
|
|
|
|
|
{
|
|
|
|
|
value_t<typename T::type> result;
|
|
|
|
|
result.value = m_ir->CreateInsertElement(v.eval(m_ir), e.eval(m_ir), i);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
auto extract(T v, u64 i)
|
|
|
|
|
{
|
|
|
|
|
typename value_t<typename T::type>::base result;
|
|
|
|
|
result.value = m_ir->CreateExtractElement(v.eval(m_ir), i);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
auto splat(u64 c)
|
|
|
|
|
{
|
|
|
|
|
value_t<T> result;
|
|
|
|
|
result.value = llvm::ConstantInt::get(result.get_type(m_context), c, result.is_sint);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
auto fsplat(f64 c)
|
|
|
|
|
{
|
|
|
|
|
value_t<T> result;
|
|
|
|
|
result.value = llvm::ConstantFP::get(result.get_type(m_context), c);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-25 19:31:16 +01:00
|
|
|
template <typename T, typename V>
|
|
|
|
|
auto vsplat(V v)
|
|
|
|
|
{
|
|
|
|
|
value_t<T> result;
|
|
|
|
|
static_assert(result.is_vector);
|
|
|
|
|
result.value = m_ir->CreateVectorSplat(result.is_vector, v.eval(m_ir));
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-01 12:21:45 +02:00
|
|
|
// Min
|
|
|
|
|
template <typename T>
|
|
|
|
|
auto min(T a, T b)
|
|
|
|
|
{
|
|
|
|
|
T result;
|
|
|
|
|
result.value = m_ir->CreateSelect((a > b).eval(m_ir), b.eval(m_ir), a.eval(m_ir));
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Max
|
|
|
|
|
template <typename T>
|
|
|
|
|
auto max(T a, T b)
|
|
|
|
|
{
|
|
|
|
|
T result;
|
|
|
|
|
result.value = m_ir->CreateSelect((a > b).eval(m_ir), a.eval(m_ir), b.eval(m_ir));
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Shuffle single vector using all zeros second vector of the same size
|
|
|
|
|
template <typename T, typename T1, typename... Args>
|
|
|
|
|
auto zshuffle(T1 a, Args... args)
|
|
|
|
|
{
|
|
|
|
|
static_assert(sizeof(T) / sizeof(std::remove_extent_t<T>) == sizeof...(Args), "zshuffle: unexpected result type");
|
|
|
|
|
const u32 values[]{static_cast<u32>(args)...};
|
|
|
|
|
value_t<T> result;
|
|
|
|
|
result.value = a.eval(m_ir);
|
|
|
|
|
result.value = m_ir->CreateShuffleVector(result.value, llvm::ConstantInt::get(result.value->getType(), 0), values);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T, typename T1, typename T2, typename... Args>
|
|
|
|
|
auto shuffle2(T1 a, T2 b, Args... args)
|
|
|
|
|
{
|
|
|
|
|
static_assert(sizeof(T) / sizeof(std::remove_extent_t<T>) == sizeof...(Args), "shuffle2: unexpected result type");
|
|
|
|
|
const u32 values[]{static_cast<u32>(args)...};
|
|
|
|
|
value_t<T> result;
|
|
|
|
|
result.value = a.eval(m_ir);
|
|
|
|
|
result.value = m_ir->CreateShuffleVector(result.value, b.eval(m_ir), values);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-25 12:58:39 +02:00
|
|
|
template <typename T, typename... Args>
|
|
|
|
|
auto build(Args... args)
|
|
|
|
|
{
|
|
|
|
|
using value_type = std::remove_extent_t<T>;
|
|
|
|
|
const value_type values[]{static_cast<value_type>(args)...};
|
|
|
|
|
static_assert(sizeof(T) / sizeof(value_type) == sizeof...(Args), "build: unexpected number of arguments");
|
|
|
|
|
value_t<T> result;
|
|
|
|
|
result.value = llvm::ConstantDataVector::get(m_context, values);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-01 12:21:45 +02:00
|
|
|
template <typename... Types>
|
|
|
|
|
llvm::Function* get_intrinsic(llvm::Intrinsic::ID id)
|
|
|
|
|
{
|
|
|
|
|
const auto module = m_ir->GetInsertBlock()->getParent()->getParent();
|
|
|
|
|
return llvm::Intrinsic::getDeclaration(module, id, {get_type<Types>()...});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
auto ctlz(T a)
|
|
|
|
|
{
|
|
|
|
|
value_t<typename T::type> result;
|
|
|
|
|
result.value = m_ir->CreateCall(get_intrinsic<typename T::type>(llvm::Intrinsic::ctlz), {a.eval(m_ir), m_ir->getFalse()});
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
auto ctpop(T a)
|
|
|
|
|
{
|
|
|
|
|
value_t<typename T::type> result;
|
|
|
|
|
result.value = m_ir->CreateCall(get_intrinsic<typename T::type>(llvm::Intrinsic::ctpop), {a.eval(m_ir)});
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
auto sqrt(T a)
|
|
|
|
|
{
|
|
|
|
|
value_t<typename T::type> result;
|
|
|
|
|
result.value = m_ir->CreateCall(get_intrinsic<typename T::type>(llvm::Intrinsic::sqrt), {a.eval(m_ir)});
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
auto fabs(T a)
|
|
|
|
|
{
|
|
|
|
|
value_t<typename T::type> result;
|
|
|
|
|
result.value = m_ir->CreateCall(get_intrinsic<typename T::type>(llvm::Intrinsic::fabs), {a.eval(m_ir)});
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-27 12:00:05 +02:00
|
|
|
// Opportunistic hardware FMA, can be used if results are identical for all possible input values
|
|
|
|
|
template <typename T>
|
|
|
|
|
auto fmuladd(T a, T b, T c)
|
|
|
|
|
{
|
|
|
|
|
value_t<typename T::type> result;
|
|
|
|
|
const auto av = a.eval(m_ir);
|
|
|
|
|
const auto bv = b.eval(m_ir);
|
|
|
|
|
const auto cv = c.eval(m_ir);
|
|
|
|
|
result.value = m_ir->CreateCall(get_intrinsic<typename T::type>(llvm::Intrinsic::fmuladd), {av, bv, cv});
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-28 15:21:08 +02:00
|
|
|
template <typename T1, typename T2>
|
|
|
|
|
value_t<u8[16]> pshufb(T1 a, T2 b)
|
|
|
|
|
{
|
|
|
|
|
value_t<u8[16]> result;
|
|
|
|
|
|
|
|
|
|
const auto data0 = a.eval(m_ir);
|
|
|
|
|
const auto index = b.eval(m_ir);
|
|
|
|
|
const auto zeros = llvm::ConstantAggregateZero::get(get_type<u8[16]>());
|
|
|
|
|
|
|
|
|
|
if (auto c = llvm::dyn_cast<llvm::Constant>(index))
|
|
|
|
|
{
|
|
|
|
|
// Convert PSHUFB index back to LLVM vector shuffle mask
|
|
|
|
|
v128 mask{};
|
|
|
|
|
|
|
|
|
|
const auto cv = llvm::dyn_cast<llvm::ConstantDataVector>(c);
|
|
|
|
|
|
|
|
|
|
if (cv)
|
|
|
|
|
{
|
|
|
|
|
for (u32 i = 0; i < 16; i++)
|
|
|
|
|
{
|
|
|
|
|
const u64 b = cv->getElementAsInteger(i);
|
|
|
|
|
mask._u8[i] = b < 128 ? b % 16 : 16;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (cv || llvm::isa<llvm::ConstantAggregateZero>(c))
|
|
|
|
|
{
|
|
|
|
|
result.value = llvm::ConstantDataVector::get(m_context, llvm::makeArrayRef((const u8*)mask._bytes, 16));
|
|
|
|
|
result.value = m_ir->CreateZExt(result.value, get_type<u32[16]>());
|
|
|
|
|
result.value = m_ir->CreateShuffleVector(data0, zeros, result.value);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (m_use_ssse3)
|
|
|
|
|
{
|
|
|
|
|
result.value = m_ir->CreateCall(get_intrinsic(llvm::Intrinsic::x86_ssse3_pshuf_b_128), {data0, index});
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// Emulate PSHUFB (TODO)
|
|
|
|
|
const auto mask = m_ir->CreateAnd(index, 0xf);
|
|
|
|
|
const auto loop = llvm::BasicBlock::Create(m_context, "", m_ir->GetInsertBlock()->getParent());
|
|
|
|
|
const auto next = llvm::BasicBlock::Create(m_context, "", m_ir->GetInsertBlock()->getParent());
|
|
|
|
|
const auto prev = m_ir->GetInsertBlock();
|
|
|
|
|
|
|
|
|
|
m_ir->CreateBr(loop);
|
|
|
|
|
m_ir->SetInsertPoint(loop);
|
|
|
|
|
const auto i = m_ir->CreatePHI(get_type<u32>(), 2);
|
|
|
|
|
const auto v = m_ir->CreatePHI(get_type<u8[16]>(), 2);
|
|
|
|
|
i->addIncoming(m_ir->getInt32(0), prev);
|
|
|
|
|
i->addIncoming(m_ir->CreateAdd(i, m_ir->getInt32(1)), loop);
|
|
|
|
|
v->addIncoming(zeros, prev);
|
|
|
|
|
result.value = m_ir->CreateInsertElement(v, m_ir->CreateExtractElement(data0, m_ir->CreateExtractElement(mask, i)), i);
|
|
|
|
|
v->addIncoming(result.value, loop);
|
|
|
|
|
m_ir->CreateCondBr(m_ir->CreateICmpULT(i, m_ir->getInt32(16)), loop, next);
|
|
|
|
|
m_ir->SetInsertPoint(next);
|
|
|
|
|
result.value = m_ir->CreateSelect(m_ir->CreateICmpSLT(index, zeros), zeros, result.value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-25 19:31:16 +01:00
|
|
|
llvm::Value* load_const(llvm::GlobalVariable* g, llvm::Value* i)
|
|
|
|
|
{
|
|
|
|
|
return m_ir->CreateLoad(m_ir->CreateGEP(g, {m_ir->getInt64(0), m_ir->CreateZExtOrTrunc(i, get_type<u64>())}));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T, typename I>
|
|
|
|
|
value_t<T> load_const(llvm::GlobalVariable* g, I i)
|
|
|
|
|
{
|
|
|
|
|
value_t<T> result;
|
|
|
|
|
result.value = load_const(g, i.eval(m_ir));
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-25 12:58:39 +02:00
|
|
|
template <typename R = v128>
|
|
|
|
|
R get_const_vector(llvm::Constant*, u32 a, u32 b);
|
|
|
|
|
|
|
|
|
|
template <typename T = v128>
|
|
|
|
|
llvm::Constant* make_const_vector(T, llvm::Type*);
|
2018-01-29 22:31:38 +01:00
|
|
|
};
|
|
|
|
|
|
2018-05-01 12:21:45 +02:00
|
|
|
#endif
|