1 /*
2  * Copyright (C) 2015 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #ifndef BERBERIS_INTRINSICS_INTRINSICS_ARGS_H_
18 #define BERBERIS_INTRINSICS_INTRINSICS_ARGS_H_
19 
20 #include <cstddef>
21 #include <cstdio>
22 
23 #include "berberis/base/checks.h"
24 
25 namespace berberis {
26 
27 // Helper classes for the EmbedAsmInstruction "construction class".
28 //
29 // Constructor of this class takes prescribed arguments from IR insn and
30 // calls a single assembler [macro]instruction.  It creates required scratch
31 // register allocations and proper MOVs to preserve semantic of the IR insn.
32 //
33 // Name of a helper class describes argument of the assembler
34 // [macro]instruction.
35 //
36 // You could find many examples in intrinsics_x86.h file but EmbedAsmInstruction
37 // is not x86-specific: in theory it should work with any MachineIRBuilder.
38 //
39 // Each argument must by of one of the following types:
40 //
41 //   InArg<N> - argument comes from <N>th source of the IR insn.
42 //              Must be "use" argument of the assembler [macro]instruction.
43 //              Note: please don't use this argument for specific register
44 //              classes (such as "RDX" or "RCX").  If one operation returns
45 //              result, e.g., in "RCX" and another one accepts it in "RCX"
46 //              then register allocator will not be able to satisfy such
47 //              requirements.  Use InTmpArg<N> for such instructions.
48 //
49 //   OutArg<N> - argument comes from <N>th destination of the IR insn.
50 //               Must be "def" or "def_early_clobber" argument of the assembler
51 //               [macro]instruction.
52 //
53 //   OutTmpArg<N> - argument is copied from temporary register to <N>th
54 //                  destination of the IR insn.
55 //                  Must be "def" or "def_early_clobber" argument of the
56 //                  assembler [macro]instruction.
57 //
58 //   InOutArg<N, M> - argument is copied from <N>th source of the IR insn to
59 //                    <M>th destination of the IR insn, then is passed it to the
60 //                    [macro]instruction.
61 //                    Must be "use_def" argument of the assembler
62 //                    [macro]instruction.
63 //
64 //   InOutTmpArg<N, M> - argument is copied from <N>th source of the IR insn to
65 //                       temporary register, then it's passed to the
66 //                       [macro]instruction. Result is copied to the
67 //                       <M>th destination of the IR insn.
68 //                       Must be "use_def" argument of the assembler
69 //                       [macro]instruction.
70 //
71 //   InTmpArg<N> - argument is copied from <N>th source of the IR insn to the
72 //                 temporary register, then is passed to the [macro]instruction.
73 //                 Must be "use_def" argument of the assembler
74 //                 [macro]instruction.
75 //
76 //   ImmArg<N, uintXX_t> - argument is 8bit/16bit/32bit/64bit immediate and
77 //                         comes as <N>th source of IR insn.
78 //
79 //   TmpArg - argument is temporary register allocated for the
80 //            [macro]instruction.
81 //
82 // TODO(khim): investigate feasibility of adding unconditional copying of
83 // arguments and results. This way we could remove classes InOutTmpArg/InTmpArg
84 // and, more importantly, make sure InArg vs InTmpArg mixup will not lead to
85 // hard to debug errors.
86 
87 struct ArgInfo {
88  public:
89   enum ArgType {
90     IN_ARG,
91     IN_TMP_ARG,
92     OUT_ARG,
93     OUT_TMP_ARG,
94     IN_OUT_ARG,
95     IN_OUT_TMP_ARG,
96     TMP_ARG,
97     IMM_ARG
98   } arg_type;
HaveInputArgInfo99   friend constexpr bool HaveInput(const ArgInfo& arg) {
100     return arg.arg_type == ArgInfo::IN_ARG ||
101            arg.arg_type == ArgInfo::IN_TMP_ARG ||
102            arg.arg_type == ArgInfo::IN_OUT_ARG ||
103            arg.arg_type == ArgInfo::IN_OUT_TMP_ARG;
104   }
HaveOutputArgInfo105   friend constexpr bool HaveOutput(const ArgInfo& arg) {
106     return arg.arg_type == ArgInfo::IN_OUT_ARG ||
107            arg.arg_type == ArgInfo::IN_OUT_TMP_ARG ||
108            arg.arg_type == OUT_ARG ||
109            arg.arg_type == OUT_TMP_ARG;
110   }
IsImmediateArgInfo111   friend constexpr bool IsImmediate(const ArgInfo& arg) {
112     return arg.arg_type == IMM_ARG;
113   }
IsTemporaryArgInfo114   friend constexpr bool IsTemporary(const ArgInfo& arg) {
115     return arg.arg_type == TMP_ARG;
116   }
117   const int from = 0;
118   const int to = 0;
119 };
120 
121 template <int N, typename RegisterClass = void, typename Usage = void>
122 class InArg;
123 
124 template <int N, typename RegisterClass = void, typename Usage = void>
125 class OutArg;
126 
127 template <int N, typename RegisterClass = void, typename Usage = void>
128 class OutTmpArg;
129 
130 template <int N, int M, typename RegisterClass = void, typename Usage = void>
131 class InOutArg;
132 
133 template <int N, int M, typename RegisterClass = void, typename Usage = void>
134 class InOutTmpArg;
135 
136 template <int N, typename RegisterClass = void, typename Usage = void>
137 class InTmpArg;
138 
139 template <int N, typename ImmType, typename ImmediateClass = void>
140 class ImmArg;
141 
142 template <typename RegisterClass = void, typename Usage = void>
143 class TmpArg;
144 
145 template <typename ArgInfo>
146 class ArgTraits;
147 
148 template <int N, typename RegisterClassType, typename UsageType>
149 class ArgTraits<InArg<N, RegisterClassType, UsageType>> {
150  public:
151   using Class = RegisterClassType;
152   using RegisterClass = RegisterClassType;
153   using Usage = UsageType;
154   static constexpr ArgInfo arg_info{.arg_type = ArgInfo::IN_ARG, .from = N};
155 };
156 
157 template <int N, typename RegisterClassType, typename UsageType>
158 class ArgTraits<OutArg<N, RegisterClassType, UsageType>> {
159  public:
160   using Class = RegisterClassType;
161   using RegisterClass = RegisterClassType;
162   using Usage = UsageType;
163   static constexpr ArgInfo arg_info{.arg_type = ArgInfo::OUT_ARG, .to = N};
164 };
165 
166 template <int N, typename RegisterClassType, typename UsageType>
167 class ArgTraits<OutTmpArg<N, RegisterClassType, UsageType>> {
168  public:
169   using Class = RegisterClassType;
170   using RegisterClass = RegisterClassType;
171   using Usage = UsageType;
172   static constexpr ArgInfo arg_info{.arg_type = ArgInfo::OUT_TMP_ARG, .to = N};
173 };
174 
175 template <int N, int M, typename RegisterClassType, typename UsageType>
176 class ArgTraits<InOutArg<N, M, RegisterClassType, UsageType>> {
177  public:
178   using Class = RegisterClassType;
179   using RegisterClass = RegisterClassType;
180   using Usage = UsageType;
181   static constexpr ArgInfo arg_info{.arg_type = ArgInfo::IN_OUT_ARG, .from = N, .to = M};
182 };
183 
184 template <int N, int M, typename RegisterClassType, typename UsageType>
185 class ArgTraits<InOutTmpArg<N, M, RegisterClassType, UsageType>> {
186  public:
187   using Class = RegisterClassType;
188   using RegisterClass = RegisterClassType;
189   using Usage = UsageType;
190   static constexpr ArgInfo arg_info{.arg_type = ArgInfo::IN_OUT_TMP_ARG, .from = N, .to = M};
191 };
192 
193 template <int N, typename RegisterClassType, typename UsageType>
194 class ArgTraits<InTmpArg<N, RegisterClassType, UsageType>> {
195  public:
196   using Class = RegisterClassType;
197   using RegisterClass = RegisterClassType;
198   using Usage = UsageType;
199   static constexpr ArgInfo arg_info{.arg_type = ArgInfo::IN_TMP_ARG, .from = N};
200 };
201 
202 template <int N, typename ImmType, typename ImmediateClassType>
203 class ArgTraits<ImmArg<N, ImmType, ImmediateClassType>> {
204  public:
205   using Class = ImmediateClassType;
206   using ImmediateClass = ImmediateClassType;
207   static constexpr ArgInfo arg_info{.arg_type = ArgInfo::IMM_ARG, .from = N};
208 };
209 
210 template <typename RegisterClassType, typename UsageType>
211 class ArgTraits<TmpArg<RegisterClassType, UsageType>> {
212  public:
213   using Class = RegisterClassType;
214   using RegisterClass = RegisterClassType;
215   using Usage = UsageType;
216   static constexpr ArgInfo arg_info{.arg_type = ArgInfo::TMP_ARG};
217 };
218 
219 // We couldn't use standard "throw std::logic_error(...)" approach here because that code is
220 // compiled with -fno-exceptions.  Thankfully printf(...) produces very similar error messages.
221 //
222 // See https://stackoverflow.com/questions/8626055/c11-static-assert-within-constexpr-function
223 // is you need an explanation for how basic technique works.
224 template <typename MachineInsn, int arguments_count>
IsCompatible(const ArgInfo * arguments)225 constexpr bool IsCompatible(const ArgInfo* arguments) {
226   int reg_arguments = 0;
227   for (size_t argument = 0; argument < arguments_count; ++argument) {
228     if (arguments[argument].arg_type != ArgInfo::IMM_ARG) {
229       if ((arguments[argument].arg_type == ArgInfo::IN_ARG) &&
230           MachineInsn::RegKindAt(reg_arguments).IsDef()) {
231         fprintf(stderr, "Incorrect use of InArg for argument %d", argument);
232         return false;
233       } else if ((arguments[argument].arg_type == ArgInfo::IN_TMP_ARG) &&
234                  !MachineInsn::RegKindAt(reg_arguments).IsDef() &&
235                  !IsFixedRegClass(MachineInsn::RegKindAt(reg_arguments).RegClass())) {
236         fprintf(stderr, "Inefficient use of InTmpArg for argument %d", argument);
237         return false;
238       } else if ((arguments[argument].arg_type == ArgInfo::OUT_ARG) &&
239                  IsFixedRegClass(MachineInsn::RegKindAt(reg_arguments).RegClass())) {
240         fprintf(stderr, "Incorrect use of OutArg for argument %d", argument);
241         return false;
242       } else if ((arguments[argument].arg_type == ArgInfo::OUT_TMP_ARG) &&
243                  !IsFixedRegClass(MachineInsn::RegKindAt(reg_arguments).RegClass())) {
244         fprintf(stderr, "Inefficient use of OutTmpArg for argument %d", argument);
245         return false;
246       } else if ((arguments[argument].arg_type == ArgInfo::IN_OUT_ARG) &&
247                  IsFixedRegClass(MachineInsn::RegKindAt(reg_arguments).RegClass())) {
248         fprintf(stderr, "Incorrect use of InOutArg for argument %d", argument);
249         return false;
250       } else if ((arguments[argument].arg_type == ArgInfo::IN_OUT_TMP_ARG) &&
251                  !IsFixedRegClass(MachineInsn::RegKindAt(reg_arguments).RegClass())) {
252         fprintf(stderr, "Inefficient use of InOutTmpArg for argument %d", argument);
253         return false;
254       }
255       if (HaveInput(arguments[argument]) &&
256           !MachineInsn::RegKindAt(reg_arguments).IsInput()) {
257         fprintf(stderr, "Argument %d does not accept input!", argument);
258         return false;
259       } else if (!HaveInput(arguments[argument]) &&
260                  MachineInsn::RegKindAt(reg_arguments).IsInput()) {
261         fprintf(stderr, "Argument %d requires valid input!", argument);
262         return false;
263       }
264       ++reg_arguments;
265     }
266   }
267   if (MachineInsn::NumRegOperands() != reg_arguments) {
268     fprintf(stderr,
269             "expected %d arguments, got %d arguments",
270             MachineInsn::NumRegOperands(),
271             reg_arguments);
272     return false;
273   }
274   return true;
275 }
276 
277 template <typename MachineInsn, typename... Args>
IsCompatible()278 constexpr bool IsCompatible() {
279   const ArgInfo arguments[] = {ArgTraits<Args>::arg_info...};
280   // Note: we couldn't pass arguments as an array into IsCompatible by reference
281   // because this would cause compilation error in case where we have no arguments.
282   //
283   // Pass pointer and element count instead.
284   return IsCompatible<MachineInsn, sizeof...(Args)>(arguments);
285 }
286 
287 template <typename MachineIRBuilder, typename Arg>
288 class ArgGetterSetter;
289 
290 template <typename Instruction, typename... Args>
291 class EmbedAsmInstruction {
292  public:
293   template <typename MachineIRBuilder, typename IntrinsicInsn>
EmbedAsmInstruction(MachineIRBuilder * builder,const IntrinsicInsn * insn)294   EmbedAsmInstruction(MachineIRBuilder* builder, const IntrinsicInsn* insn) {
295     static_assert(IsCompatible<Instruction, Args...>(), "Incompatible intrinsic embedding");
296     builder->template Gen<Instruction>(ArgGetterSetter<MachineIRBuilder, Args>(builder, insn)...);
297   }
298 };
299 
300 }  // namespace berberis
301 
302 #endif  // BERBERIS_INTRINSICS_INTRINSICS_ARGS_H_
303