clang 23.0.0git
RISCV.cpp
Go to the documentation of this file.
1//===-------- RISCV.cpp - Emit LLVM Code for builtins ---------------------===//
2//
3// 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
6//
7//===----------------------------------------------------------------------===//
8//
9// This contains code to emit Builtin calls as LLVM code.
10//
11//===----------------------------------------------------------------------===//
12
13#include "CodeGenFunction.h"
15#include "llvm/IR/IntrinsicsRISCV.h"
16#include "llvm/TargetParser/RISCVISAInfo.h"
17#include "llvm/TargetParser/RISCVTargetParser.h"
18
19using namespace clang;
20using namespace CodeGen;
21using namespace llvm;
22
23// The 0th bit simulates the `vta` of RVV
24// The 1st bit simulates the `vma` of RVV
25static constexpr unsigned RVV_VTA = 0x1;
26static constexpr unsigned RVV_VMA = 0x2;
27
28// RISC-V Vector builtin helper functions are marked NOINLINE to prevent
29// excessive inlining in CodeGenFunction::EmitRISCVBuiltinExpr's large switch
30// statement, which would significantly increase compilation time.
31static LLVM_ATTRIBUTE_NOINLINE Value *
33 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
34 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
35 int PolicyAttrs, bool IsMasked) {
36 auto &Builder = CGF->Builder;
37 auto &CGM = CGF->CGM;
39 if (IsMasked) {
40 // Move mask to right before vl.
41 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 1);
42 if ((PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA))
43 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
44 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
45 IntrinsicTypes = {ResultType, Ops[4]->getType(), Ops[2]->getType()};
46 } else {
47 if (PolicyAttrs & RVV_VTA)
48 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
49 IntrinsicTypes = {ResultType, Ops[3]->getType(), Ops[1]->getType()};
50 }
51 Value *NewVL = Ops[2];
52 Ops.erase(Ops.begin() + 2);
53 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
54 llvm::Value *LoadValue = Builder.CreateCall(F, Ops, "");
55 llvm::Value *V = Builder.CreateExtractValue(LoadValue, {0});
56 // Store new_vl.
57 clang::CharUnits Align;
58 if (IsMasked)
59 Align = CGM.getNaturalPointeeTypeAlignment(
60 E->getArg(E->getNumArgs() - 2)->getType());
61 else
62 Align = CGM.getNaturalPointeeTypeAlignment(E->getArg(1)->getType());
63 llvm::Value *Val = Builder.CreateExtractValue(LoadValue, {1});
64 Builder.CreateStore(Val, Address(NewVL, Val->getType(), Align));
65 return V;
66}
67
68static LLVM_ATTRIBUTE_NOINLINE Value *
70 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
71 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
72 int PolicyAttrs, bool IsMasked) {
73 auto &Builder = CGF->Builder;
74 auto &CGM = CGF->CGM;
76 if (IsMasked) {
77 // Builtin: (mask, ptr, stride, value, vl). Intrinsic: (value, ptr, stride,
78 // mask, vl)
79 std::swap(Ops[0], Ops[3]);
80 } else {
81 // Builtin: (ptr, stride, value, vl). Intrinsic: (value, ptr, stride, vl)
82 std::rotate(Ops.begin(), Ops.begin() + 2, Ops.begin() + 3);
83 }
84 if (IsMasked)
85 IntrinsicTypes = {Ops[0]->getType(), Ops[1]->getType(), Ops[4]->getType()};
86 else
87 IntrinsicTypes = {Ops[0]->getType(), Ops[1]->getType(), Ops[3]->getType()};
88 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
89 return Builder.CreateCall(F, Ops, "");
90}
91
92static LLVM_ATTRIBUTE_NOINLINE Value *
94 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
95 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
96 int PolicyAttrs, bool IsMasked) {
97 auto &Builder = CGF->Builder;
98 auto &CGM = CGF->CGM;
100 if (IsMasked) {
101 // Builtin: (mask, ptr, index, value, vl).
102 // Intrinsic: (value, ptr, index, mask, vl)
103 std::swap(Ops[0], Ops[3]);
104 } else {
105 // Builtin: (ptr, index, value, vl).
106 // Intrinsic: (value, ptr, index, vl)
107 std::rotate(Ops.begin(), Ops.begin() + 2, Ops.begin() + 3);
108 }
109 if (IsMasked)
110 IntrinsicTypes = {Ops[0]->getType(), Ops[1]->getType(), Ops[2]->getType(),
111 Ops[4]->getType()};
112 else
113 IntrinsicTypes = {Ops[0]->getType(), Ops[1]->getType(), Ops[2]->getType(),
114 Ops[3]->getType()};
115 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
116 return Builder.CreateCall(F, Ops, "");
117}
118
119static LLVM_ATTRIBUTE_NOINLINE Value *
121 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
122 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
123 int PolicyAttrs, bool IsMasked) {
124 auto &Builder = CGF->Builder;
125 auto &CGM = CGF->CGM;
127 if (IsMasked) {
128 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 1);
129 if ((PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA))
130 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
131 } else {
132 if (PolicyAttrs & RVV_VTA)
133 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
134 }
135 auto ElemTy = cast<llvm::VectorType>(ResultType)->getElementType();
136 Ops.insert(Ops.begin() + 2, llvm::Constant::getNullValue(ElemTy));
137 if (IsMasked) {
138 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
139 // maskedoff, op1, op2, mask, vl, policy
140 IntrinsicTypes = {ResultType, ElemTy, Ops[4]->getType()};
141 } else {
142 // passthru, op1, op2, vl
143 IntrinsicTypes = {ResultType, ElemTy, Ops[3]->getType()};
144 }
145 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
146 return Builder.CreateCall(F, Ops, "");
147}
148
149static LLVM_ATTRIBUTE_NOINLINE Value *
151 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
152 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
153 int PolicyAttrs, bool IsMasked) {
154 auto &Builder = CGF->Builder;
155 auto &CGM = CGF->CGM;
157 if (IsMasked) {
158 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 1);
159 if ((PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA))
160 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
161 } else {
162 if (PolicyAttrs & RVV_VTA)
163 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
164 }
165 auto ElemTy = cast<llvm::VectorType>(ResultType)->getElementType();
166 Ops.insert(Ops.begin() + 2, llvm::Constant::getAllOnesValue(ElemTy));
167 if (IsMasked) {
168 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
169 // maskedoff, op1, po2, mask, vl, policy
170 IntrinsicTypes = {ResultType, ElemTy, Ops[4]->getType()};
171 } else {
172 // passthru, op1, op2, vl
173 IntrinsicTypes = {ResultType, ElemTy, Ops[3]->getType()};
174 }
175 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
176 return Builder.CreateCall(F, Ops, "");
177}
178
179static LLVM_ATTRIBUTE_NOINLINE Value *
181 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
182 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
183 int PolicyAttrs, bool IsMasked) {
184 auto &Builder = CGF->Builder;
185 auto &CGM = CGF->CGM;
187 // op1, vl
188 IntrinsicTypes = {ResultType, Ops[1]->getType()};
189 Ops.insert(Ops.begin() + 1, Ops[0]);
190 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
191 return Builder.CreateCall(F, Ops, "");
192}
193
194static LLVM_ATTRIBUTE_NOINLINE Value *
196 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
197 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
198 int PolicyAttrs, bool IsMasked) {
199 auto &Builder = CGF->Builder;
200 auto &CGM = CGF->CGM;
202 if (IsMasked) {
203 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 1);
204 if ((PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA))
205 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
206 Ops.insert(Ops.begin() + 2, Ops[1]);
207 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
208 // maskedoff, op1, op2, mask, vl
209 IntrinsicTypes = {ResultType, Ops[2]->getType(), Ops.back()->getType()};
210 } else {
211 if (PolicyAttrs & RVV_VTA)
212 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
213 // op1, po2, vl
214 IntrinsicTypes = {ResultType, Ops[1]->getType(), Ops[2]->getType()};
215 Ops.insert(Ops.begin() + 2, Ops[1]);
216 }
217 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
218 return Builder.CreateCall(F, Ops, "");
219}
220
221static LLVM_ATTRIBUTE_NOINLINE Value *
223 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
224 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
225 int PolicyAttrs, bool IsMasked) {
226 auto &Builder = CGF->Builder;
227 auto &CGM = CGF->CGM;
229 if (IsMasked) {
230 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 1);
231 if ((PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA))
232 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
233 } else {
234 if (PolicyAttrs & RVV_VTA)
235 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
236 }
237 auto ElemTy = cast<llvm::VectorType>(Ops[1]->getType())->getElementType();
238 Ops.insert(Ops.begin() + 2, llvm::Constant::getNullValue(ElemTy));
239 if (IsMasked) {
240 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
241 // maskedoff, op1, op2, mask, vl, policy
242 IntrinsicTypes = {ResultType, Ops[1]->getType(), ElemTy, Ops[4]->getType()};
243 } else {
244 // passtru, op1, op2, vl
245 IntrinsicTypes = {ResultType, Ops[1]->getType(), ElemTy, Ops[3]->getType()};
246 }
247 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
248 return Builder.CreateCall(F, Ops, "");
249}
250
251static LLVM_ATTRIBUTE_NOINLINE Value *
253 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
254 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
255 int PolicyAttrs, bool IsMasked) {
256 auto &Builder = CGF->Builder;
257 auto &CGM = CGF->CGM;
259 if (IsMasked) {
260 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 1);
261 if ((PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA))
262 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
263 } else {
264 if (PolicyAttrs & RVV_VTA)
265 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
266 }
267 Ops.insert(Ops.begin() + 2,
268 llvm::Constant::getNullValue(Ops.back()->getType()));
269 if (IsMasked) {
270 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
271 // maskedoff, op1, xlen, mask, vl
272 IntrinsicTypes = {ResultType, Ops[1]->getType(), Ops[4]->getType(),
273 Ops[4]->getType()};
274 } else {
275 // passthru, op1, xlen, vl
276 IntrinsicTypes = {ResultType, Ops[1]->getType(), Ops[3]->getType(),
277 Ops[3]->getType()};
278 }
279 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
280 return Builder.CreateCall(F, Ops, "");
281}
282
283static LLVM_ATTRIBUTE_NOINLINE Value *
285 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
286 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
287 int PolicyAttrs, bool IsMasked) {
288 auto &Builder = CGF->Builder;
289 auto &CGM = CGF->CGM;
290 LLVMContext &Context = CGM.getLLVMContext();
291 llvm::MDBuilder MDHelper(Context);
292 llvm::Metadata *OpsMD[] = {llvm::MDString::get(Context, "vlenb")};
293 llvm::MDNode *RegName = llvm::MDNode::get(Context, OpsMD);
294 llvm::Value *Metadata = llvm::MetadataAsValue::get(Context, RegName);
295 llvm::Function *F =
296 CGM.getIntrinsic(llvm::Intrinsic::read_register, {CGF->SizeTy});
297 return Builder.CreateCall(F, Metadata);
298}
299
300static LLVM_ATTRIBUTE_NOINLINE Value *
302 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
303 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
304 int PolicyAttrs, bool IsMasked) {
305 auto &Builder = CGF->Builder;
306 auto &CGM = CGF->CGM;
307 llvm::Function *F = CGM.getIntrinsic(ID, {ResultType});
308 return Builder.CreateCall(F, Ops, "");
309}
310
311static LLVM_ATTRIBUTE_NOINLINE Value *
313 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
314 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
315 int PolicyAttrs, bool IsMasked) {
316 auto &Builder = CGF->Builder;
317 auto &CGM = CGF->CGM;
319 if (IsMasked) {
320 // Builtin: (mask, ptr, value, vl).
321 // Intrinsic: (value, ptr, mask, vl)
322 std::swap(Ops[0], Ops[2]);
323 } else {
324 // Builtin: (ptr, value, vl).
325 // Intrinsic: (value, ptr, vl)
326 std::swap(Ops[0], Ops[1]);
327 }
328 if (IsMasked)
329 IntrinsicTypes = {Ops[0]->getType(), Ops[1]->getType(), Ops[3]->getType()};
330 else
331 IntrinsicTypes = {Ops[0]->getType(), Ops[1]->getType(), Ops[2]->getType()};
332 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
333 return Builder.CreateCall(F, Ops, "");
334}
335
336static LLVM_ATTRIBUTE_NOINLINE Value *emitRVVUnitStridedSegLoadTupleBuiltin(
337 CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue,
338 llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
339 int PolicyAttrs, bool IsMasked, unsigned SegInstSEW) {
340 auto &Builder = CGF->Builder;
341 auto &CGM = CGF->CGM;
343 bool NoPassthru =
344 (IsMasked && (PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA)) |
345 (!IsMasked && (PolicyAttrs & RVV_VTA));
346 unsigned Offset = IsMasked ? NoPassthru ? 1 : 2 : NoPassthru ? 0 : 1;
347 if (IsMasked)
348 IntrinsicTypes = {ResultType, Ops[Offset]->getType(), Ops[0]->getType(),
349 Ops.back()->getType()};
350 else
351 IntrinsicTypes = {ResultType, Ops[Offset]->getType(),
352 Ops.back()->getType()};
353 if (IsMasked)
354 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 1);
355 if (NoPassthru)
356 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
357 if (IsMasked)
358 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
359 Ops.push_back(ConstantInt::get(Ops.back()->getType(), SegInstSEW));
360 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
361 llvm::Value *LoadValue = Builder.CreateCall(F, Ops, "");
362 if (ReturnValue.isNull())
363 return LoadValue;
364 return Builder.CreateStore(LoadValue, ReturnValue.getValue());
365}
366
367static LLVM_ATTRIBUTE_NOINLINE Value *emitRVVUnitStridedSegStoreTupleBuiltin(
368 CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue,
369 llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
370 int PolicyAttrs, bool IsMasked, unsigned SegInstSEW) {
371 auto &Builder = CGF->Builder;
372 auto &CGM = CGF->CGM;
374 // Masked
375 // Builtin: (mask, ptr, v_tuple, vl)
376 // Intrinsic: (tuple, ptr, mask, vl, SegInstSEW)
377 // Unmasked
378 // Builtin: (ptr, v_tuple, vl)
379 // Intrinsic: (tuple, ptr, vl, SegInstSEW)
380 if (IsMasked)
381 std::swap(Ops[0], Ops[2]);
382 else
383 std::swap(Ops[0], Ops[1]);
384 Ops.push_back(ConstantInt::get(Ops.back()->getType(), SegInstSEW));
385 if (IsMasked)
386 IntrinsicTypes = {Ops[0]->getType(), Ops[1]->getType(), Ops[2]->getType(),
387 Ops[3]->getType()};
388 else
389 IntrinsicTypes = {Ops[0]->getType(), Ops[1]->getType(), Ops[2]->getType()};
390 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
391 return Builder.CreateCall(F, Ops, "");
392}
393
394static LLVM_ATTRIBUTE_NOINLINE Value *emitRVVUnitStridedSegLoadFFTupleBuiltin(
395 CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue,
396 llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
397 int PolicyAttrs, bool IsMasked, unsigned SegInstSEW) {
398 auto &Builder = CGF->Builder;
399 auto &CGM = CGF->CGM;
401 bool NoPassthru =
402 (IsMasked && (PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA)) |
403 (!IsMasked && (PolicyAttrs & RVV_VTA));
404 unsigned Offset = IsMasked ? NoPassthru ? 1 : 2 : NoPassthru ? 0 : 1;
405 if (IsMasked)
406 IntrinsicTypes = {ResultType, Ops.back()->getType(), Ops[Offset]->getType(),
407 Ops[0]->getType()};
408 else
409 IntrinsicTypes = {ResultType, Ops.back()->getType(),
410 Ops[Offset]->getType()};
411 if (IsMasked)
412 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 1);
413 if (NoPassthru)
414 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
415 if (IsMasked)
416 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
417 Ops.push_back(ConstantInt::get(Ops.back()->getType(), SegInstSEW));
418 Value *NewVL = Ops[2];
419 Ops.erase(Ops.begin() + 2);
420 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
421 llvm::Value *LoadValue = Builder.CreateCall(F, Ops, "");
422 // Get alignment from the new vl operand
423 clang::CharUnits Align =
424 CGM.getNaturalPointeeTypeAlignment(E->getArg(Offset + 1)->getType());
425 llvm::Value *ReturnTuple = Builder.CreateExtractValue(LoadValue, 0);
426 // Store new_vl
427 llvm::Value *V = Builder.CreateExtractValue(LoadValue, 1);
428 Builder.CreateStore(V, Address(NewVL, V->getType(), Align));
429 if (ReturnValue.isNull())
430 return ReturnTuple;
431 return Builder.CreateStore(ReturnTuple, ReturnValue.getValue());
432}
433
434static LLVM_ATTRIBUTE_NOINLINE Value *emitRVVStridedSegLoadTupleBuiltin(
435 CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue,
436 llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
437 int PolicyAttrs, bool IsMasked, unsigned SegInstSEW) {
438 auto &Builder = CGF->Builder;
439 auto &CGM = CGF->CGM;
441 bool NoPassthru =
442 (IsMasked && (PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA)) |
443 (!IsMasked && (PolicyAttrs & RVV_VTA));
444 unsigned Offset = IsMasked ? NoPassthru ? 1 : 2 : NoPassthru ? 0 : 1;
445 if (IsMasked)
446 IntrinsicTypes = {ResultType, Ops[Offset]->getType(), Ops.back()->getType(),
447 Ops[0]->getType()};
448 else
449 IntrinsicTypes = {ResultType, Ops[Offset]->getType(),
450 Ops.back()->getType()};
451 if (IsMasked)
452 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 1);
453 if (NoPassthru)
454 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
455 if (IsMasked)
456 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
457 Ops.push_back(ConstantInt::get(Ops.back()->getType(), SegInstSEW));
458 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
459 llvm::Value *LoadValue = Builder.CreateCall(F, Ops, "");
460 if (ReturnValue.isNull())
461 return LoadValue;
462 return Builder.CreateStore(LoadValue, ReturnValue.getValue());
463}
464
465static LLVM_ATTRIBUTE_NOINLINE Value *emitRVVStridedSegStoreTupleBuiltin(
466 CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue,
467 llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
468 int PolicyAttrs, bool IsMasked, unsigned SegInstSEW) {
469 auto &Builder = CGF->Builder;
470 auto &CGM = CGF->CGM;
472 // Masked
473 // Builtin: (mask, ptr, stride, v_tuple, vl)
474 // Intrinsic: (tuple, ptr, stride, mask, vl, SegInstSEW)
475 // Unmasked
476 // Builtin: (ptr, stride, v_tuple, vl)
477 // Intrinsic: (tuple, ptr, stride, vl, SegInstSEW)
478 if (IsMasked)
479 std::swap(Ops[0], Ops[3]);
480 else
481 std::rotate(Ops.begin(), Ops.begin() + 2, Ops.begin() + 3);
482 Ops.push_back(ConstantInt::get(Ops.back()->getType(), SegInstSEW));
483 if (IsMasked)
484 IntrinsicTypes = {Ops[0]->getType(), Ops[1]->getType(), Ops[4]->getType(),
485 Ops[3]->getType()};
486 else
487 IntrinsicTypes = {Ops[0]->getType(), Ops[1]->getType(), Ops[3]->getType()};
488 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
489 return Builder.CreateCall(F, Ops, "");
490}
491
492static LLVM_ATTRIBUTE_NOINLINE Value *
494 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
495 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
496 int PolicyAttrs, bool IsMasked) {
497 auto &Builder = CGF->Builder;
498 auto &CGM = CGF->CGM;
499 // LLVM intrinsic
500 // Unmasked: (passthru, op0, op1, round_mode, vl)
501 // Masked: (passthru, vector_in, vector_in/scalar_in, mask, vxrm, vl,
502 // policy)
503
504 bool HasMaskedOff =
505 !((IsMasked && (PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA)) ||
506 (!IsMasked && PolicyAttrs & RVV_VTA));
507
508 if (IsMasked)
509 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 2);
510
511 if (!HasMaskedOff)
512 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
513
514 if (IsMasked)
515 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
516
517 llvm::Function *F = CGM.getIntrinsic(
518 ID, {ResultType, Ops[2]->getType(), Ops.back()->getType()});
519 return Builder.CreateCall(F, Ops, "");
520}
521
522static LLVM_ATTRIBUTE_NOINLINE Value *
524 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
525 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
526 int PolicyAttrs, bool IsMasked) {
527 auto &Builder = CGF->Builder;
528 auto &CGM = CGF->CGM;
529 // LLVM intrinsic
530 // Unmasked: (passthru, op0, op1, round_mode, vl)
531 // Masked: (passthru, vector_in, vector_in/scalar_in, mask, vxrm, vl,
532 // policy)
533
534 bool HasMaskedOff =
535 !((IsMasked && (PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA)) ||
536 (!IsMasked && PolicyAttrs & RVV_VTA));
537
538 if (IsMasked)
539 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 2);
540
541 if (!HasMaskedOff)
542 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
543
544 if (IsMasked)
545 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
546
547 llvm::Function *F =
548 CGM.getIntrinsic(ID, {ResultType, Ops[1]->getType(), Ops[2]->getType(),
549 Ops.back()->getType()});
550 return Builder.CreateCall(F, Ops, "");
551}
552
553static LLVM_ATTRIBUTE_NOINLINE Value *
555 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
556 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
557 int PolicyAttrs, bool IsMasked) {
558 auto &Builder = CGF->Builder;
559 auto &CGM = CGF->CGM;
560 // LLVM intrinsic
561 // Unmasked: (passthru, op0, op1, round_mode, vl)
562 // Masked: (passthru, vector_in, vector_in/scalar_in, mask, frm, vl, policy)
563
564 bool HasMaskedOff =
565 !((IsMasked && (PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA)) ||
566 (!IsMasked && PolicyAttrs & RVV_VTA));
567 bool HasRoundModeOp =
568 IsMasked ? (HasMaskedOff ? Ops.size() == 6 : Ops.size() == 5)
569 : (HasMaskedOff ? Ops.size() == 5 : Ops.size() == 4);
570
571 if (!HasRoundModeOp)
572 Ops.insert(Ops.end() - 1,
573 ConstantInt::get(Ops.back()->getType(), 7)); // frm
574
575 if (IsMasked)
576 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 2);
577
578 if (!HasMaskedOff)
579 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
580
581 if (IsMasked)
582 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
583
584 llvm::Function *F = CGM.getIntrinsic(
585 ID, {ResultType, Ops[2]->getType(), Ops.back()->getType()});
586 return Builder.CreateCall(F, Ops, "");
587}
588
589static LLVM_ATTRIBUTE_NOINLINE Value *emitRVVWideningFloatingPointBuiltin(
590 CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue,
591 llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
592 int PolicyAttrs, bool IsMasked) {
593 auto &Builder = CGF->Builder;
594 auto &CGM = CGF->CGM;
595 // LLVM intrinsic
596 // Unmasked: (passthru, op0, op1, round_mode, vl)
597 // Masked: (passthru, vector_in, vector_in/scalar_in, mask, frm, vl, policy)
598
599 bool HasMaskedOff =
600 !((IsMasked && (PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA)) ||
601 (!IsMasked && PolicyAttrs & RVV_VTA));
602 bool HasRoundModeOp =
603 IsMasked ? (HasMaskedOff ? Ops.size() == 6 : Ops.size() == 5)
604 : (HasMaskedOff ? Ops.size() == 5 : Ops.size() == 4);
605
606 if (!HasRoundModeOp)
607 Ops.insert(Ops.end() - 1,
608 ConstantInt::get(Ops.back()->getType(), 7)); // frm
609
610 if (IsMasked)
611 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 2);
612
613 if (!HasMaskedOff)
614 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
615
616 if (IsMasked)
617 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
618
619 llvm::Function *F =
620 CGM.getIntrinsic(ID, {ResultType, Ops[1]->getType(), Ops[2]->getType(),
621 Ops.back()->getType()});
622 return Builder.CreateCall(F, Ops, "");
623}
624
625static LLVM_ATTRIBUTE_NOINLINE Value *emitRVVIndexedSegLoadTupleBuiltin(
626 CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue,
627 llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
628 int PolicyAttrs, bool IsMasked, unsigned SegInstSEW) {
629 auto &Builder = CGF->Builder;
630 auto &CGM = CGF->CGM;
632
633 bool NoPassthru =
634 (IsMasked && (PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA)) |
635 (!IsMasked && (PolicyAttrs & RVV_VTA));
636
637 if (IsMasked)
638 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 1);
639 if (NoPassthru)
640 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
641
642 if (IsMasked)
643 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
644 Ops.push_back(ConstantInt::get(Ops.back()->getType(), SegInstSEW));
645
646 if (IsMasked)
647 IntrinsicTypes = {ResultType, Ops[1]->getType(), Ops[2]->getType(),
648 Ops[3]->getType(), Ops[4]->getType()};
649 else
650 IntrinsicTypes = {ResultType, Ops[1]->getType(), Ops[2]->getType(),
651 Ops[3]->getType()};
652 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
653 llvm::Value *LoadValue = Builder.CreateCall(F, Ops, "");
654
655 if (ReturnValue.isNull())
656 return LoadValue;
657 return Builder.CreateStore(LoadValue, ReturnValue.getValue());
658}
659
660static LLVM_ATTRIBUTE_NOINLINE Value *emitRVVIndexedSegStoreTupleBuiltin(
661 CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue,
662 llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
663 int PolicyAttrs, bool IsMasked, unsigned SegInstSEW) {
664 auto &Builder = CGF->Builder;
665 auto &CGM = CGF->CGM;
667 // Masked
668 // Builtin: (mask, ptr, index, v_tuple, vl)
669 // Intrinsic: (tuple, ptr, index, mask, vl, SegInstSEW)
670 // Unmasked
671 // Builtin: (ptr, index, v_tuple, vl)
672 // Intrinsic: (tuple, ptr, index, vl, SegInstSEW)
673
674 if (IsMasked)
675 std::swap(Ops[0], Ops[3]);
676 else
677 std::rotate(Ops.begin(), Ops.begin() + 2, Ops.begin() + 3);
678
679 Ops.push_back(ConstantInt::get(Ops.back()->getType(), SegInstSEW));
680
681 if (IsMasked)
682 IntrinsicTypes = {Ops[0]->getType(), Ops[1]->getType(), Ops[2]->getType(),
683 Ops[3]->getType(), Ops[4]->getType()};
684 else
685 IntrinsicTypes = {Ops[0]->getType(), Ops[1]->getType(), Ops[2]->getType(),
686 Ops[3]->getType()};
687 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
688 return Builder.CreateCall(F, Ops, "");
689}
690
691static LLVM_ATTRIBUTE_NOINLINE Value *
693 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
694 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
695 int PolicyAttrs, bool IsMasked) {
696 auto &Builder = CGF->Builder;
697 auto &CGM = CGF->CGM;
698 // LLVM intrinsic
699 // Unmasked: (vector_in, vector_in/scalar_in, vector_in, round_mode,
700 // vl, policy)
701 // Masked: (vector_in, vector_in/scalar_in, vector_in, mask, frm,
702 // vl, policy)
703
704 bool HasRoundModeOp = IsMasked ? Ops.size() == 6 : Ops.size() == 5;
705
706 if (!HasRoundModeOp)
707 Ops.insert(Ops.end() - 1,
708 ConstantInt::get(Ops.back()->getType(), 7)); // frm
709
710 if (IsMasked)
711 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 2);
712
713 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
714
715 llvm::Function *F = CGM.getIntrinsic(
716 ID, {ResultType, Ops[1]->getType(), Ops.back()->getType()});
717 return Builder.CreateCall(F, Ops, "");
718}
719
720static LLVM_ATTRIBUTE_NOINLINE Value *
722 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
723 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
724 int PolicyAttrs, bool IsMasked) {
725 auto &Builder = CGF->Builder;
726 auto &CGM = CGF->CGM;
727 // LLVM intrinsic
728 // Unmasked: (vector_in, vector_in/scalar_in, vector_in, round_mode, vl,
729 // policy) Masked: (vector_in, vector_in/scalar_in, vector_in, mask, frm,
730 // vl, policy)
731
732 bool HasRoundModeOp = IsMasked ? Ops.size() == 6 : Ops.size() == 5;
733
734 if (!HasRoundModeOp)
735 Ops.insert(Ops.end() - 1,
736 ConstantInt::get(Ops.back()->getType(), 7)); // frm
737
738 if (IsMasked)
739 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.begin() + 4);
740
741 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
742
743 llvm::Function *F =
744 CGM.getIntrinsic(ID, {ResultType, Ops[1]->getType(), Ops[2]->getType(),
745 Ops.back()->getType()});
746 return Builder.CreateCall(F, Ops, "");
747}
748
749static LLVM_ATTRIBUTE_NOINLINE Value *
751 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
752 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
753 int PolicyAttrs, bool IsMasked) {
754 auto &Builder = CGF->Builder;
755 auto &CGM = CGF->CGM;
757 // LLVM intrinsic
758 // Unmasked: (passthru, op0, round_mode, vl)
759 // Masked: (passthru, op0, mask, frm, vl, policy)
760
761 bool HasMaskedOff =
762 !((IsMasked && (PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA)) ||
763 (!IsMasked && PolicyAttrs & RVV_VTA));
764 bool HasRoundModeOp =
765 IsMasked ? (HasMaskedOff ? Ops.size() == 5 : Ops.size() == 4)
766 : (HasMaskedOff ? Ops.size() == 4 : Ops.size() == 3);
767
768 if (!HasRoundModeOp)
769 Ops.insert(Ops.end() - 1,
770 ConstantInt::get(Ops.back()->getType(), 7)); // frm
771
772 if (IsMasked)
773 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 2);
774
775 if (!HasMaskedOff)
776 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
777
778 if (IsMasked)
779 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
780
781 IntrinsicTypes = {ResultType, Ops.back()->getType()};
782 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
783 return Builder.CreateCall(F, Ops, "");
784}
785
786static LLVM_ATTRIBUTE_NOINLINE Value *
788 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
789 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
790 int PolicyAttrs, bool IsMasked) {
791 auto &Builder = CGF->Builder;
792 auto &CGM = CGF->CGM;
793 // LLVM intrinsic
794 // Unmasked: (passthru, op0, frm, vl)
795 // Masked: (passthru, op0, mask, frm, vl, policy)
796 bool HasMaskedOff =
797 !((IsMasked && (PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA)) ||
798 (!IsMasked && PolicyAttrs & RVV_VTA));
799 bool HasRoundModeOp =
800 IsMasked ? (HasMaskedOff ? Ops.size() == 5 : Ops.size() == 4)
801 : (HasMaskedOff ? Ops.size() == 4 : Ops.size() == 3);
802
803 if (!HasRoundModeOp)
804 Ops.insert(Ops.end() - 1,
805 ConstantInt::get(Ops.back()->getType(), 7)); // frm
806
807 if (IsMasked)
808 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 2);
809
810 if (!HasMaskedOff)
811 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
812
813 if (IsMasked)
814 Ops.push_back(ConstantInt::get(Ops.back()->getType(), PolicyAttrs));
815
816 llvm::Function *F = CGM.getIntrinsic(
817 ID, {ResultType, Ops[1]->getType(), Ops.back()->getType()});
818 return Builder.CreateCall(F, Ops, "");
819}
820
821static LLVM_ATTRIBUTE_NOINLINE Value *emitRVVFloatingReductionBuiltin(
822 CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue,
823 llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
824 int PolicyAttrs, bool IsMasked) {
825 auto &Builder = CGF->Builder;
826 auto &CGM = CGF->CGM;
827 // LLVM intrinsic
828 // Unmasked: (passthru, op0, op1, round_mode, vl)
829 // Masked: (passthru, vector_in, vector_in/scalar_in, mask, frm, vl, policy)
830
831 bool HasMaskedOff =
832 !((IsMasked && (PolicyAttrs & RVV_VTA) && (PolicyAttrs & RVV_VMA)) ||
833 (!IsMasked && PolicyAttrs & RVV_VTA));
834 bool HasRoundModeOp =
835 IsMasked ? (HasMaskedOff ? Ops.size() == 6 : Ops.size() == 5)
836 : (HasMaskedOff ? Ops.size() == 5 : Ops.size() == 4);
837
838 if (!HasRoundModeOp)
839 Ops.insert(Ops.end() - 1,
840 ConstantInt::get(Ops.back()->getType(), 7)); // frm
841
842 if (IsMasked)
843 std::rotate(Ops.begin(), Ops.begin() + 1, Ops.end() - 2);
844
845 if (!HasMaskedOff)
846 Ops.insert(Ops.begin(), llvm::PoisonValue::get(ResultType));
847
848 llvm::Function *F = CGM.getIntrinsic(
849 ID, {ResultType, Ops[1]->getType(), Ops.back()->getType()});
850 return Builder.CreateCall(F, Ops, "");
851}
852
853static LLVM_ATTRIBUTE_NOINLINE Value *
855 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
856 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
857 int PolicyAttrs, bool IsMasked) {
858 auto &Builder = CGF->Builder;
859 auto &CGM = CGF->CGM;
860
861 if (ResultType->isIntOrIntVectorTy(1) ||
862 Ops[0]->getType()->isIntOrIntVectorTy(1)) {
863 assert(isa<ScalableVectorType>(ResultType) &&
865
866 LLVMContext &Context = CGM.getLLVMContext();
867 ScalableVectorType *Boolean64Ty =
868 ScalableVectorType::get(llvm::Type::getInt1Ty(Context), 64);
869
870 if (ResultType->isIntOrIntVectorTy(1)) {
871 // Casting from m1 vector integer -> vector boolean
872 // Ex: <vscale x 8 x i8>
873 // --(bitcast)--------> <vscale x 64 x i1>
874 // --(vector_extract)-> <vscale x 8 x i1>
875 llvm::Value *BitCast = Builder.CreateBitCast(Ops[0], Boolean64Ty);
876 return Builder.CreateExtractVector(ResultType, BitCast,
877 ConstantInt::get(CGF->Int64Ty, 0));
878 } else {
879 // Casting from vector boolean -> m1 vector integer
880 // Ex: <vscale x 1 x i1>
881 // --(vector_insert)-> <vscale x 64 x i1>
882 // --(bitcast)-------> <vscale x 8 x i8>
883 llvm::Value *Boolean64Val = Builder.CreateInsertVector(
884 Boolean64Ty, llvm::PoisonValue::get(Boolean64Ty), Ops[0],
885 ConstantInt::get(CGF->Int64Ty, 0));
886 return Builder.CreateBitCast(Boolean64Val, ResultType);
887 }
888 }
889 return Builder.CreateBitCast(Ops[0], ResultType);
890}
891
892static LLVM_ATTRIBUTE_NOINLINE Value *
894 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
895 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
896 int PolicyAttrs, bool IsMasked) {
897 auto &Builder = CGF->Builder;
898 auto *VecTy = cast<ScalableVectorType>(ResultType);
899 if (auto *OpVecTy = dyn_cast<ScalableVectorType>(Ops[0]->getType())) {
900 unsigned MaxIndex =
901 OpVecTy->getMinNumElements() / VecTy->getMinNumElements();
902 assert(isPowerOf2_32(MaxIndex));
903 // Mask to only valid indices.
904 Ops[1] = Builder.CreateZExt(Ops[1], Builder.getInt64Ty());
905 Ops[1] = Builder.CreateAnd(Ops[1], MaxIndex - 1);
906 Ops[1] =
907 Builder.CreateMul(Ops[1], ConstantInt::get(Ops[1]->getType(),
908 VecTy->getMinNumElements()));
909 return Builder.CreateExtractVector(ResultType, Ops[0], Ops[1]);
910 }
911
912 return Builder.CreateIntrinsic(
913 Intrinsic::riscv_tuple_extract, {ResultType, Ops[0]->getType()},
914 {Ops[0], Builder.CreateTrunc(Ops[1], Builder.getInt32Ty())});
915}
916
917static LLVM_ATTRIBUTE_NOINLINE Value *
919 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
920 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
921 int PolicyAttrs, bool IsMasked) {
922 auto &Builder = CGF->Builder;
923 if (auto *ResVecTy = dyn_cast<ScalableVectorType>(ResultType)) {
924 auto *VecTy = cast<ScalableVectorType>(Ops[2]->getType());
925 unsigned MaxIndex =
926 ResVecTy->getMinNumElements() / VecTy->getMinNumElements();
927 assert(isPowerOf2_32(MaxIndex));
928 // Mask to only valid indices.
929 Ops[1] = Builder.CreateZExt(Ops[1], Builder.getInt64Ty());
930 Ops[1] = Builder.CreateAnd(Ops[1], MaxIndex - 1);
931 Ops[1] =
932 Builder.CreateMul(Ops[1], ConstantInt::get(Ops[1]->getType(),
933 VecTy->getMinNumElements()));
934 return Builder.CreateInsertVector(ResultType, Ops[0], Ops[2], Ops[1]);
935 }
936
937 return Builder.CreateIntrinsic(
938 Intrinsic::riscv_tuple_insert, {ResultType, Ops[2]->getType()},
939 {Ops[0], Ops[2], Builder.CreateTrunc(Ops[1], Builder.getInt32Ty())});
940}
941
942static LLVM_ATTRIBUTE_NOINLINE Value *
944 ReturnValueSlot ReturnValue, llvm::Type *ResultType,
945 Intrinsic::ID ID, SmallVectorImpl<Value *> &Ops,
946 int PolicyAttrs, bool IsMasked) {
947 auto &Builder = CGF->Builder;
948 llvm::Value *ReturnVector = llvm::PoisonValue::get(ResultType);
949 auto *VecTy = cast<ScalableVectorType>(Ops[0]->getType());
950 for (unsigned I = 0, N = Ops.size(); I < N; ++I) {
951 if (isa<ScalableVectorType>(ResultType)) {
952 llvm::Value *Idx = ConstantInt::get(Builder.getInt64Ty(),
953 VecTy->getMinNumElements() * I);
954 ReturnVector =
955 Builder.CreateInsertVector(ResultType, ReturnVector, Ops[I], Idx);
956 } else {
957 llvm::Value *Idx = ConstantInt::get(Builder.getInt32Ty(), I);
958 ReturnVector = Builder.CreateIntrinsic(Intrinsic::riscv_tuple_insert,
959 {ResultType, Ops[I]->getType()},
960 {ReturnVector, Ops[I], Idx});
961 }
962 }
963 return ReturnVector;
964}
965
967 llvm::FunctionType *FTy = llvm::FunctionType::get(VoidTy, {VoidPtrTy}, false);
968 llvm::FunctionCallee Func =
969 CGM.CreateRuntimeFunction(FTy, "__init_riscv_feature_bits");
970 auto *CalleeGV = cast<llvm::GlobalValue>(Func.getCallee());
971 CalleeGV->setDSOLocal(true);
972 CalleeGV->setDLLStorageClass(llvm::GlobalValue::DefaultStorageClass);
973 return Builder.CreateCall(Func, {llvm::ConstantPointerNull::get(VoidPtrTy)});
974}
975
977
978 const Expr *FeatureExpr = E->getArg(0)->IgnoreParenCasts();
979 StringRef FeatureStr = cast<StringLiteral>(FeatureExpr)->getString();
980 if (!getContext().getTargetInfo().validateCpuSupports(FeatureStr))
981 return Builder.getFalse();
982
983 return EmitRISCVCpuSupports(ArrayRef<StringRef>(FeatureStr));
984}
985
986static Value *loadRISCVFeatureBits(unsigned Index, CGBuilderTy &Builder,
987 CodeGenModule &CGM) {
988 llvm::Type *Int32Ty = Builder.getInt32Ty();
989 llvm::Type *Int64Ty = Builder.getInt64Ty();
990 llvm::ArrayType *ArrayOfInt64Ty =
991 llvm::ArrayType::get(Int64Ty, llvm::RISCVISAInfo::FeatureBitSize);
992 llvm::Type *StructTy = llvm::StructType::get(Int32Ty, ArrayOfInt64Ty);
993 llvm::Constant *RISCVFeaturesBits =
994 CGM.CreateRuntimeVariable(StructTy, "__riscv_feature_bits");
995 cast<llvm::GlobalValue>(RISCVFeaturesBits)->setDSOLocal(true);
996 Value *IndexVal = llvm::ConstantInt::get(Int32Ty, Index);
997 llvm::Value *GEPIndices[] = {Builder.getInt32(0), Builder.getInt32(1),
998 IndexVal};
999 Value *Ptr =
1000 Builder.CreateInBoundsGEP(StructTy, RISCVFeaturesBits, GEPIndices);
1001 Value *FeaturesBit =
1002 Builder.CreateAlignedLoad(Int64Ty, Ptr, CharUnits::fromQuantity(8));
1003 return FeaturesBit;
1004}
1005
1007 const unsigned RISCVFeatureLength = llvm::RISCVISAInfo::FeatureBitSize;
1008 uint64_t RequireBitMasks[RISCVFeatureLength] = {0};
1009
1010 for (auto Feat : FeaturesStrs) {
1011 auto [GroupID, BitPos] = RISCVISAInfo::getRISCVFeaturesBitsInfo(Feat);
1012
1013 // If there isn't BitPos for this feature, skip this version.
1014 // It also report the warning to user during compilation.
1015 if (BitPos == -1)
1016 return Builder.getFalse();
1017
1018 RequireBitMasks[GroupID] |= (1ULL << BitPos);
1019 }
1020
1021 Value *Result = nullptr;
1022 for (unsigned Idx = 0; Idx < RISCVFeatureLength; Idx++) {
1023 if (RequireBitMasks[Idx] == 0)
1024 continue;
1025
1026 Value *Mask = Builder.getInt64(RequireBitMasks[Idx]);
1027 Value *Bitset =
1028 Builder.CreateAnd(loadRISCVFeatureBits(Idx, Builder, CGM), Mask);
1029 Value *CmpV = Builder.CreateICmpEQ(Bitset, Mask);
1030 Result = (!Result) ? CmpV : Builder.CreateAnd(Result, CmpV);
1031 }
1032
1033 assert(Result && "Should have value here.");
1034
1035 return Result;
1036}
1037
1039 const Expr *CPUExpr = E->getArg(0)->IgnoreParenCasts();
1040 StringRef CPUStr = cast<clang::StringLiteral>(CPUExpr)->getString();
1041 return EmitRISCVCpuIs(CPUStr);
1042}
1043
1045 llvm::Type *Int32Ty = Builder.getInt32Ty();
1046 llvm::Type *Int64Ty = Builder.getInt64Ty();
1047 llvm::StructType *StructTy = llvm::StructType::get(Int32Ty, Int64Ty, Int64Ty);
1048 llvm::Constant *RISCVCPUModel =
1049 CGM.CreateRuntimeVariable(StructTy, "__riscv_cpu_model");
1050 cast<llvm::GlobalValue>(RISCVCPUModel)->setDSOLocal(true);
1051
1052 auto loadRISCVCPUID = [&](unsigned Index) {
1053 Value *Ptr = Builder.CreateStructGEP(StructTy, RISCVCPUModel, Index);
1054 Value *CPUID = Builder.CreateAlignedLoad(StructTy->getTypeAtIndex(Index),
1055 Ptr, llvm::MaybeAlign());
1056 return CPUID;
1057 };
1058
1059 const llvm::RISCV::CPUModel Model = llvm::RISCV::getCPUModel(CPUStr);
1060
1061 // Compare mvendorid.
1062 Value *VendorID = loadRISCVCPUID(0);
1063 Value *Result =
1064 Builder.CreateICmpEQ(VendorID, Builder.getInt32(Model.MVendorID));
1065
1066 // Compare marchid.
1067 Value *ArchID = loadRISCVCPUID(1);
1068 Result = Builder.CreateAnd(
1069 Result, Builder.CreateICmpEQ(ArchID, Builder.getInt64(Model.MArchID)));
1070
1071 // Compare mimpid.
1072 Value *ImpID = loadRISCVCPUID(2);
1073 Result = Builder.CreateAnd(
1074 Result, Builder.CreateICmpEQ(ImpID, Builder.getInt64(Model.MImpID)));
1075
1076 return Result;
1077}
1078
1080 const CallExpr *E,
1082
1083 if (BuiltinID == Builtin::BI__builtin_cpu_supports)
1084 return EmitRISCVCpuSupports(E);
1085 if (BuiltinID == Builtin::BI__builtin_cpu_init)
1086 return EmitRISCVCpuInit();
1087 if (BuiltinID == Builtin::BI__builtin_cpu_is)
1088 return EmitRISCVCpuIs(E);
1089
1091 llvm::Type *ResultType = ConvertType(E->getType());
1092
1093 // Find out if any arguments are required to be integer constant expressions.
1094 unsigned ICEArguments = 0;
1096 getContext().GetBuiltinType(BuiltinID, Error, &ICEArguments);
1098 // Vector intrinsics don't have a type string.
1099 assert(BuiltinID >= clang::RISCV::FirstRVVBuiltin &&
1100 BuiltinID <= clang::RISCV::LastRVVBuiltin);
1101 ICEArguments = 0;
1102 if (BuiltinID == RISCVVector::BI__builtin_rvv_vget_v ||
1103 BuiltinID == RISCVVector::BI__builtin_rvv_vset_v)
1104 ICEArguments = 1 << 1;
1105 } else {
1106 assert(Error == ASTContext::GE_None && "Unexpected error");
1107 }
1108
1109 if (BuiltinID == RISCV::BI__builtin_riscv_ntl_load)
1110 ICEArguments |= (1 << 1);
1111 if (BuiltinID == RISCV::BI__builtin_riscv_ntl_store)
1112 ICEArguments |= (1 << 2);
1113
1114 for (unsigned i = 0, e = E->getNumArgs(); i != e; i++) {
1115 // Handle aggregate argument, namely RVV tuple types in segment load/store
1118 llvm::Value *AggValue = Builder.CreateLoad(L.getAddress());
1119 Ops.push_back(AggValue);
1120 continue;
1121 }
1122 Ops.push_back(EmitScalarOrConstFoldImmArg(ICEArguments, i, E));
1123 }
1124
1125 Intrinsic::ID ID = Intrinsic::not_intrinsic;
1126 int PolicyAttrs = 0;
1127 bool IsMasked = false;
1128 // This is used by segment load/store to determine it's llvm type.
1129 unsigned SegInstSEW = 8;
1130 // This is used by XSfmm.
1131 unsigned TWiden = 0;
1132
1133 // Required for overloaded intrinsics.
1135 switch (BuiltinID) {
1136 default: llvm_unreachable("unexpected builtin ID");
1137 case RISCV::BI__builtin_riscv_orc_b_32:
1138 case RISCV::BI__builtin_riscv_orc_b_64:
1139 case RISCV::BI__builtin_riscv_clmul_32:
1140 case RISCV::BI__builtin_riscv_clmul_64:
1141 case RISCV::BI__builtin_riscv_clmulh_32:
1142 case RISCV::BI__builtin_riscv_clmulh_64:
1143 case RISCV::BI__builtin_riscv_clmulr_32:
1144 case RISCV::BI__builtin_riscv_clmulr_64:
1145 case RISCV::BI__builtin_riscv_xperm4_32:
1146 case RISCV::BI__builtin_riscv_xperm4_64:
1147 case RISCV::BI__builtin_riscv_xperm8_32:
1148 case RISCV::BI__builtin_riscv_xperm8_64:
1149 case RISCV::BI__builtin_riscv_brev8_32:
1150 case RISCV::BI__builtin_riscv_brev8_64:
1151 case RISCV::BI__builtin_riscv_zip_32:
1152 case RISCV::BI__builtin_riscv_unzip_32: {
1153 switch (BuiltinID) {
1154 default: llvm_unreachable("unexpected builtin ID");
1155 // Zbb
1156 case RISCV::BI__builtin_riscv_orc_b_32:
1157 case RISCV::BI__builtin_riscv_orc_b_64:
1158 ID = Intrinsic::riscv_orc_b;
1159 break;
1160
1161 // Zbc
1162 case RISCV::BI__builtin_riscv_clmul_32:
1163 case RISCV::BI__builtin_riscv_clmul_64:
1164 ID = Intrinsic::clmul;
1165 break;
1166 case RISCV::BI__builtin_riscv_clmulh_32:
1167 case RISCV::BI__builtin_riscv_clmulh_64:
1168 ID = Intrinsic::riscv_clmulh;
1169 break;
1170 case RISCV::BI__builtin_riscv_clmulr_32:
1171 case RISCV::BI__builtin_riscv_clmulr_64:
1172 ID = Intrinsic::riscv_clmulr;
1173 break;
1174
1175 // Zbkx
1176 case RISCV::BI__builtin_riscv_xperm8_32:
1177 case RISCV::BI__builtin_riscv_xperm8_64:
1178 ID = Intrinsic::riscv_xperm8;
1179 break;
1180 case RISCV::BI__builtin_riscv_xperm4_32:
1181 case RISCV::BI__builtin_riscv_xperm4_64:
1182 ID = Intrinsic::riscv_xperm4;
1183 break;
1184
1185 // Zbkb
1186 case RISCV::BI__builtin_riscv_brev8_32:
1187 case RISCV::BI__builtin_riscv_brev8_64:
1188 ID = Intrinsic::riscv_brev8;
1189 break;
1190 case RISCV::BI__builtin_riscv_zip_32:
1191 ID = Intrinsic::riscv_zip;
1192 break;
1193 case RISCV::BI__builtin_riscv_unzip_32:
1194 ID = Intrinsic::riscv_unzip;
1195 break;
1196 }
1197
1198 IntrinsicTypes = {ResultType};
1199 break;
1200 }
1201
1202 // Zk builtins
1203
1204 // Zknh
1205 case RISCV::BI__builtin_riscv_sha256sig0:
1206 ID = Intrinsic::riscv_sha256sig0;
1207 break;
1208 case RISCV::BI__builtin_riscv_sha256sig1:
1209 ID = Intrinsic::riscv_sha256sig1;
1210 break;
1211 case RISCV::BI__builtin_riscv_sha256sum0:
1212 ID = Intrinsic::riscv_sha256sum0;
1213 break;
1214 case RISCV::BI__builtin_riscv_sha256sum1:
1215 ID = Intrinsic::riscv_sha256sum1;
1216 break;
1217
1218 // Zksed
1219 case RISCV::BI__builtin_riscv_sm4ks:
1220 ID = Intrinsic::riscv_sm4ks;
1221 break;
1222 case RISCV::BI__builtin_riscv_sm4ed:
1223 ID = Intrinsic::riscv_sm4ed;
1224 break;
1225
1226 // Zksh
1227 case RISCV::BI__builtin_riscv_sm3p0:
1228 ID = Intrinsic::riscv_sm3p0;
1229 break;
1230 case RISCV::BI__builtin_riscv_sm3p1:
1231 ID = Intrinsic::riscv_sm3p1;
1232 break;
1233
1234 case RISCV::BI__builtin_riscv_clz_32:
1235 case RISCV::BI__builtin_riscv_clz_64: {
1236 Function *F = CGM.getIntrinsic(Intrinsic::ctlz, Ops[0]->getType());
1237 Value *Result = Builder.CreateCall(F, {Ops[0], Builder.getInt1(false)});
1238 if (Result->getType() != ResultType)
1239 Result =
1240 Builder.CreateIntCast(Result, ResultType, /*isSigned*/ false, "cast");
1241 return Result;
1242 }
1243 case RISCV::BI__builtin_riscv_ctz_32:
1244 case RISCV::BI__builtin_riscv_ctz_64: {
1245 Function *F = CGM.getIntrinsic(Intrinsic::cttz, Ops[0]->getType());
1246 Value *Result = Builder.CreateCall(F, {Ops[0], Builder.getInt1(false)});
1247 if (Result->getType() != ResultType)
1248 Result =
1249 Builder.CreateIntCast(Result, ResultType, /*isSigned*/ false, "cast");
1250 return Result;
1251 }
1252
1253 // Zihintntl
1254 case RISCV::BI__builtin_riscv_ntl_load: {
1255 llvm::Type *ResTy = ConvertType(E->getType());
1256 unsigned DomainVal = 5; // Default __RISCV_NTLH_ALL
1257 if (Ops.size() == 2)
1258 DomainVal = cast<ConstantInt>(Ops[1])->getZExtValue();
1259
1260 llvm::MDNode *RISCVDomainNode = llvm::MDNode::get(
1262 llvm::ConstantAsMetadata::get(Builder.getInt32(DomainVal)));
1263 llvm::MDNode *NontemporalNode = llvm::MDNode::get(
1264 getLLVMContext(), llvm::ConstantAsMetadata::get(Builder.getInt32(1)));
1265
1266 int Width;
1267 if(ResTy->isScalableTy()) {
1268 const ScalableVectorType *SVTy = cast<ScalableVectorType>(ResTy);
1269 llvm::Type *ScalarTy = ResTy->getScalarType();
1270 Width = ScalarTy->getPrimitiveSizeInBits() *
1271 SVTy->getElementCount().getKnownMinValue();
1272 } else
1273 Width = ResTy->getPrimitiveSizeInBits();
1274 LoadInst *Load = Builder.CreateLoad(
1275 Address(Ops[0], ResTy, CharUnits::fromQuantity(Width / 8)));
1276
1277 Load->setMetadata(llvm::LLVMContext::MD_nontemporal, NontemporalNode);
1278 Load->setMetadata(CGM.getModule().getMDKindID("riscv-nontemporal-domain"),
1279 RISCVDomainNode);
1280
1281 return Load;
1282 }
1283 case RISCV::BI__builtin_riscv_ntl_store: {
1284 unsigned DomainVal = 5; // Default __RISCV_NTLH_ALL
1285 if (Ops.size() == 3)
1286 DomainVal = cast<ConstantInt>(Ops[2])->getZExtValue();
1287
1288 llvm::MDNode *RISCVDomainNode = llvm::MDNode::get(
1290 llvm::ConstantAsMetadata::get(Builder.getInt32(DomainVal)));
1291 llvm::MDNode *NontemporalNode = llvm::MDNode::get(
1292 getLLVMContext(), llvm::ConstantAsMetadata::get(Builder.getInt32(1)));
1293
1294 StoreInst *Store = Builder.CreateDefaultAlignedStore(Ops[1], Ops[0]);
1295 Store->setMetadata(llvm::LLVMContext::MD_nontemporal, NontemporalNode);
1296 Store->setMetadata(CGM.getModule().getMDKindID("riscv-nontemporal-domain"),
1297 RISCVDomainNode);
1298
1299 return Store;
1300 }
1301 // Zihintpause
1302 case RISCV::BI__builtin_riscv_pause: {
1303 llvm::Function *Fn = CGM.getIntrinsic(llvm::Intrinsic::riscv_pause);
1304 return Builder.CreateCall(Fn, {});
1305 }
1306
1307 // XCValu
1308 case RISCV::BI__builtin_riscv_cv_alu_addN:
1309 ID = Intrinsic::riscv_cv_alu_addN;
1310 break;
1311 case RISCV::BI__builtin_riscv_cv_alu_addRN:
1312 ID = Intrinsic::riscv_cv_alu_addRN;
1313 break;
1314 case RISCV::BI__builtin_riscv_cv_alu_adduN:
1315 ID = Intrinsic::riscv_cv_alu_adduN;
1316 break;
1317 case RISCV::BI__builtin_riscv_cv_alu_adduRN:
1318 ID = Intrinsic::riscv_cv_alu_adduRN;
1319 break;
1320 case RISCV::BI__builtin_riscv_cv_alu_clip:
1321 ID = Intrinsic::riscv_cv_alu_clip;
1322 break;
1323 case RISCV::BI__builtin_riscv_cv_alu_clipu:
1324 ID = Intrinsic::riscv_cv_alu_clipu;
1325 break;
1326 case RISCV::BI__builtin_riscv_cv_alu_extbs:
1327 return Builder.CreateSExt(Builder.CreateTrunc(Ops[0], Int8Ty), Int32Ty,
1328 "extbs");
1329 case RISCV::BI__builtin_riscv_cv_alu_extbz:
1330 return Builder.CreateZExt(Builder.CreateTrunc(Ops[0], Int8Ty), Int32Ty,
1331 "extbz");
1332 case RISCV::BI__builtin_riscv_cv_alu_exths:
1333 return Builder.CreateSExt(Builder.CreateTrunc(Ops[0], Int16Ty), Int32Ty,
1334 "exths");
1335 case RISCV::BI__builtin_riscv_cv_alu_exthz:
1336 return Builder.CreateZExt(Builder.CreateTrunc(Ops[0], Int16Ty), Int32Ty,
1337 "exthz");
1338 case RISCV::BI__builtin_riscv_cv_alu_sle:
1339 return Builder.CreateZExt(Builder.CreateICmpSLE(Ops[0], Ops[1]), Int32Ty,
1340 "sle");
1341 case RISCV::BI__builtin_riscv_cv_alu_sleu:
1342 return Builder.CreateZExt(Builder.CreateICmpULE(Ops[0], Ops[1]), Int32Ty,
1343 "sleu");
1344 case RISCV::BI__builtin_riscv_cv_alu_subN:
1345 ID = Intrinsic::riscv_cv_alu_subN;
1346 break;
1347 case RISCV::BI__builtin_riscv_cv_alu_subRN:
1348 ID = Intrinsic::riscv_cv_alu_subRN;
1349 break;
1350 case RISCV::BI__builtin_riscv_cv_alu_subuN:
1351 ID = Intrinsic::riscv_cv_alu_subuN;
1352 break;
1353 case RISCV::BI__builtin_riscv_cv_alu_subuRN:
1354 ID = Intrinsic::riscv_cv_alu_subuRN;
1355 break;
1356
1357 // XAndesPerf
1358 case RISCV::BI__builtin_riscv_nds_ffb_32:
1359 case RISCV::BI__builtin_riscv_nds_ffb_64:
1360 IntrinsicTypes = {ResultType};
1361 ID = Intrinsic::riscv_nds_ffb;
1362 break;
1363 case RISCV::BI__builtin_riscv_nds_ffzmism_32:
1364 case RISCV::BI__builtin_riscv_nds_ffzmism_64:
1365 IntrinsicTypes = {ResultType};
1366 ID = Intrinsic::riscv_nds_ffzmism;
1367 break;
1368 case RISCV::BI__builtin_riscv_nds_ffmism_32:
1369 case RISCV::BI__builtin_riscv_nds_ffmism_64:
1370 IntrinsicTypes = {ResultType};
1371 ID = Intrinsic::riscv_nds_ffmism;
1372 break;
1373 case RISCV::BI__builtin_riscv_nds_flmism_32:
1374 case RISCV::BI__builtin_riscv_nds_flmism_64:
1375 IntrinsicTypes = {ResultType};
1376 ID = Intrinsic::riscv_nds_flmism;
1377 break;
1378
1379 // XAndesBFHCvt
1380 case RISCV::BI__builtin_riscv_nds_fcvt_s_bf16:
1381 return Builder.CreateFPExt(Ops[0], FloatTy);
1382 case RISCV::BI__builtin_riscv_nds_fcvt_bf16_s:
1383 return Builder.CreateFPTrunc(Ops[0], BFloatTy);
1384
1385 // Vector builtins are handled from here.
1386#include "clang/Basic/riscv_vector_builtin_cg.inc"
1387
1388 // SiFive Vector builtins are handled from here.
1389#include "clang/Basic/riscv_sifive_vector_builtin_cg.inc"
1390
1391 // Andes Vector builtins are handled from here.
1392#include "clang/Basic/riscv_andes_vector_builtin_cg.inc"
1393 }
1394
1395 assert(ID != Intrinsic::not_intrinsic);
1396
1397 llvm::Function *F = CGM.getIntrinsic(ID, IntrinsicTypes);
1398 return Builder.CreateCall(F, Ops, "");
1399}
#define V(N, I)
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVStridedSegLoadTupleBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked, unsigned SegInstSEW)
Definition RISCV.cpp:434
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVPseudoUnaryBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:120
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVCreateBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:943
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVGetBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:893
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVFloatingPointBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:554
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVFloatingReductionBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:821
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVPseudoMaskBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:180
static constexpr unsigned RVV_VMA
Definition RISCV.cpp:26
static Value * loadRISCVFeatureBits(unsigned Index, CGBuilderTy &Builder, CodeGenModule &CGM)
Definition RISCV.cpp:986
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVFloatingUnaryBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:750
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVFMABuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:692
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVVLEFFBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:32
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVPseudoVNotBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:150
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVStridedSegStoreTupleBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked, unsigned SegInstSEW)
Definition RISCV.cpp:465
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVVSEMaskBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:312
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVNarrowingClipBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:523
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVVsetvliBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:301
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVVSSEBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:69
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVIndexedSegLoadTupleBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked, unsigned SegInstSEW)
Definition RISCV.cpp:625
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVFloatingConvBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:787
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVUnitStridedSegLoadFFTupleBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked, unsigned SegInstSEW)
Definition RISCV.cpp:394
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVAveragingBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:493
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVUnitStridedSegStoreTupleBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked, unsigned SegInstSEW)
Definition RISCV.cpp:367
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVWideningFMABuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:721
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVIndexedStoreBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:93
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVSetBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:918
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVIndexedSegStoreTupleBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked, unsigned SegInstSEW)
Definition RISCV.cpp:660
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVReinterpretBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:854
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVPseudoVWCVTBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:222
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVPseudoVNCVTBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:252
static constexpr unsigned RVV_VTA
Definition RISCV.cpp:25
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVWideningFloatingPointBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:589
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVVlenbBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:284
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVUnitStridedSegLoadTupleBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked, unsigned SegInstSEW)
Definition RISCV.cpp:336
static LLVM_ATTRIBUTE_NOINLINE Value * emitRVVPseudoVFUnaryBuiltin(CodeGenFunction *CGF, const CallExpr *E, ReturnValueSlot ReturnValue, llvm::Type *ResultType, Intrinsic::ID ID, SmallVectorImpl< Value * > &Ops, int PolicyAttrs, bool IsMasked)
Definition RISCV.cpp:195
TokenType getType() const
Returns the token's type, e.g.
Enumerates target-specific builtins in their own namespaces within namespace clang.
QualType GetBuiltinType(unsigned ID, GetBuiltinTypeError &Error, unsigned *IntegerConstantArgs=nullptr) const
Return the type for the specified builtin.
@ GE_None
No error.
@ GE_Missing_type
Missing a type.
CallExpr - Represents a function call (C99 6.5.2.2, C++ [expr.call]).
Definition Expr.h:2946
Expr * getArg(unsigned Arg)
getArg - Return the specified argument.
Definition Expr.h:3150
unsigned getNumArgs() const
getNumArgs - Return the number of actual arguments to this call.
Definition Expr.h:3137
CharUnits - This is an opaque type for sizes expressed in character units.
Definition CharUnits.h:38
static CharUnits fromQuantity(QuantityType Quantity)
fromQuantity - Construct a CharUnits quantity from a raw integer type.
Definition CharUnits.h:63
Like RawAddress, an abstract representation of an aligned address, but the pointer contained in this ...
Definition Address.h:128
CodeGenFunction - This class organizes the per-function state that is used while generating LLVM code...
llvm::Value * EmitScalarOrConstFoldImmArg(unsigned ICEArguments, unsigned Idx, const CallExpr *E)
llvm::Value * EmitRISCVCpuSupports(const CallExpr *E)
Definition RISCV.cpp:976
llvm::Value * EmitRISCVCpuInit()
Definition RISCV.cpp:966
llvm::Type * ConvertType(QualType T)
llvm::Value * EmitRISCVBuiltinExpr(unsigned BuiltinID, const CallExpr *E, ReturnValueSlot ReturnValue)
Definition RISCV.cpp:1079
llvm::Value * EmitRISCVCpuIs(const CallExpr *E)
Definition RISCV.cpp:1038
LValue EmitAggExprToLValue(const Expr *E)
EmitAggExprToLValue - Emit the computation of the specified expression of aggregate type into a tempo...
static bool hasAggregateEvaluationKind(QualType T)
Address ReturnValue
ReturnValue - The temporary alloca to hold the return value.
llvm::LLVMContext & getLLVMContext()
This class organizes the cross-function state that is used while generating LLVM code.
llvm::Constant * CreateRuntimeVariable(llvm::Type *Ty, StringRef Name)
Create a new runtime global variable with the specified type and name.
llvm::LLVMContext & getLLVMContext()
llvm::Function * getIntrinsic(unsigned IID, ArrayRef< llvm::Type * > Tys={})
LValue - This represents an lvalue references.
Definition CGValue.h:183
Address getAddress() const
Definition CGValue.h:373
ReturnValueSlot - Contains the address where the return value of a function can be stored,...
Definition CGCall.h:382
This represents one expression.
Definition Expr.h:112
Expr * IgnoreParenCasts() LLVM_READONLY
Skip past any parentheses and casts which might surround this expression until reaching a fixed point...
Definition Expr.cpp:3102
QualType getType() const
Definition Expr.h:144
QualType getType() const
Definition Value.cpp:238
The JSON file list parser is used to communicate input to InstallAPI.
bool isa(CodeGen::Address addr)
Definition Address.h:330
@ Result
The result type of a method or function.
Definition TypeBase.h:905
U cast(CodeGen::Address addr)
Definition Address.h:327
Diagnostic wrappers for TextAPI types for error reporting.
Definition Dominators.h:30
llvm::IntegerType * Int8Ty
i8, i16, i32, and i64