1 /*
2  * Copyright 2014 Advanced Micro Devices, Inc.
3  *
4  * Permission is hereby granted, free of charge, to any person obtaining a
5  * copy of this software and associated documentation files (the
6  * "Software"), to deal in the Software without restriction, including
7  * without limitation the rights to use, copy, modify, merge, publish,
8  * distribute, sub license, and/or sell copies of the Software, and to
9  * permit persons to whom the Software is furnished to do so, subject to
10  * the following conditions:
11  *
12  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
13  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
14  * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL
15  * THE COPYRIGHT HOLDERS, AUTHORS AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM,
16  * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
17  * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
18  * USE OR OTHER DEALINGS IN THE SOFTWARE.
19  *
20  * The above copyright notice and this permission notice (including the
21  * next paragraph) shall be included in all copies or substantial portions
22  * of the Software.
23  *
24  */
25 
26 /* based on Marek's patch to lp_bld_misc.cpp */
27 
28 // Workaround http://llvm.org/PR23628
29 #pragma push_macro("DEBUG")
30 #undef DEBUG
31 
32 #include "ac_llvm_util.h"
33 #include <llvm-c/Core.h>
34 #include <llvm/Target/TargetOptions.h>
35 #include <llvm/ExecutionEngine/ExecutionEngine.h>
36 #include <llvm/IR/Attributes.h>
37 #include <llvm/IR/CallSite.h>
38 #include <llvm/IR/IRBuilder.h>
39 
40 #if HAVE_LLVM < 0x0500
41 namespace llvm {
42 typedef AttributeSet AttributeList;
43 }
44 #endif
45 
ac_add_attr_dereferenceable(LLVMValueRef val,uint64_t bytes)46 void ac_add_attr_dereferenceable(LLVMValueRef val, uint64_t bytes)
47 {
48    llvm::Argument *A = llvm::unwrap<llvm::Argument>(val);
49 #if HAVE_LLVM < 0x0500
50    llvm::AttrBuilder B;
51    B.addDereferenceableAttr(bytes);
52    A->addAttr(llvm::AttributeList::get(A->getContext(), A->getArgNo() + 1,  B));
53 #else
54    A->addAttr(llvm::Attribute::getWithDereferenceableBytes(A->getContext(), bytes));
55 #endif
56 }
57 
ac_is_sgpr_param(LLVMValueRef arg)58 bool ac_is_sgpr_param(LLVMValueRef arg)
59 {
60 	llvm::Argument *A = llvm::unwrap<llvm::Argument>(arg);
61 	llvm::AttributeList AS = A->getParent()->getAttributes();
62 	unsigned ArgNo = A->getArgNo();
63 	return AS.hasAttribute(ArgNo + 1, llvm::Attribute::ByVal) ||
64 	       AS.hasAttribute(ArgNo + 1, llvm::Attribute::InReg);
65 }
66 
ac_llvm_get_called_value(LLVMValueRef call)67 LLVMValueRef ac_llvm_get_called_value(LLVMValueRef call)
68 {
69 #if HAVE_LLVM >= 0x0309
70 	return LLVMGetCalledValue(call);
71 #else
72 	return llvm::wrap(llvm::CallSite(llvm::unwrap<llvm::Instruction>(call)).getCalledValue());
73 #endif
74 }
75 
ac_llvm_is_function(LLVMValueRef v)76 bool ac_llvm_is_function(LLVMValueRef v)
77 {
78 #if HAVE_LLVM >= 0x0309
79 	return LLVMGetValueKind(v) == LLVMFunctionValueKind;
80 #else
81 	return llvm::isa<llvm::Function>(llvm::unwrap(v));
82 #endif
83 }
84 
ac_create_builder(LLVMContextRef ctx,enum ac_float_mode float_mode)85 LLVMBuilderRef ac_create_builder(LLVMContextRef ctx,
86 				 enum ac_float_mode float_mode)
87 {
88 	LLVMBuilderRef builder = LLVMCreateBuilderInContext(ctx);
89 
90 #if HAVE_LLVM >= 0x0308
91 	llvm::FastMathFlags flags;
92 
93 	switch (float_mode) {
94 	case AC_FLOAT_MODE_DEFAULT:
95 		break;
96 	case AC_FLOAT_MODE_NO_SIGNED_ZEROS_FP_MATH:
97 		flags.setNoSignedZeros();
98 		llvm::unwrap(builder)->setFastMathFlags(flags);
99 		break;
100 	case AC_FLOAT_MODE_UNSAFE_FP_MATH:
101 #if HAVE_LLVM >= 0x0600
102 		flags.setFast();
103 #else
104 		flags.setUnsafeAlgebra();
105 #endif
106 		llvm::unwrap(builder)->setFastMathFlags(flags);
107 		break;
108 	}
109 #endif
110 
111 	return builder;
112 }
113