forked from llvm/clangir
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathCIRGenBuilder.h
889 lines (765 loc) · 33.6 KB
/
CIRGenBuilder.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
//===-- CIRGenBuilder.h - CIRBuilder implementation ------------*- C++ -*-===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#ifndef LLVM_CLANG_LIB_CIR_CIRGENBUILDER_H
#define LLVM_CLANG_LIB_CIR_CIRGENBUILDER_H
#include "Address.h"
#include "CIRDataLayout.h"
#include "CIRGenRecordLayout.h"
#include "CIRGenTypeCache.h"
#include "UnimplementedFeatureGuarding.h"
#include "clang/AST/Decl.h"
#include "clang/AST/Type.h"
#include "clang/CIR/Dialect/Builder/CIRBaseBuilder.h"
#include "clang/CIR/Dialect/IR/CIRAttrs.h"
#include "clang/CIR/Dialect/IR/CIRDialect.h"
#include "clang/CIR/Dialect/IR/CIROpsEnums.h"
#include "clang/CIR/Dialect/IR/CIRTypes.h"
#include "clang/CIR/Dialect/IR/FPEnv.h"
#include "mlir/IR/Attributes.h"
#include "mlir/IR/Builders.h"
#include "mlir/IR/BuiltinAttributes.h"
#include "mlir/IR/BuiltinOps.h"
#include "mlir/IR/BuiltinTypes.h"
#include "mlir/IR/Location.h"
#include "mlir/IR/Types.h"
#include "llvm/ADT/APSInt.h"
#include "llvm/ADT/ArrayRef.h"
#include "llvm/ADT/FloatingPointMode.h"
#include "llvm/ADT/StringMap.h"
#include "llvm/ADT/StringSet.h"
#include "llvm/Support/ErrorHandling.h"
#include <cassert>
#include <optional>
#include <string>
namespace cir {
class CIRGenFunction;
class CIRGenBuilderTy : public CIRBaseBuilderTy {
const CIRGenTypeCache &typeCache;
bool IsFPConstrained = false;
fp::ExceptionBehavior DefaultConstrainedExcept = fp::ebStrict;
llvm::RoundingMode DefaultConstrainedRounding = llvm::RoundingMode::Dynamic;
llvm::StringMap<unsigned> GlobalsVersioning;
llvm::StringSet<> anonRecordNames;
public:
CIRGenBuilderTy(mlir::MLIRContext &C, const CIRGenTypeCache &tc)
: CIRBaseBuilderTy(C), typeCache(tc) {}
std::string getUniqueAnonRecordName() {
std::string name = "anon." + std::to_string(anonRecordNames.size());
anonRecordNames.insert(name);
return name;
}
//
// Floating point specific helpers
// -------------------------------
//
/// Enable/Disable use of constrained floating point math. When enabled the
/// CreateF<op>() calls instead create constrained floating point intrinsic
/// calls. Fast math flags are unaffected by this setting.
void setIsFPConstrained(bool IsCon) {
if (IsCon)
llvm_unreachable("Constrained FP NYI");
IsFPConstrained = IsCon;
}
/// Query for the use of constrained floating point math
bool getIsFPConstrained() {
if (IsFPConstrained)
llvm_unreachable("Constrained FP NYI");
return IsFPConstrained;
}
/// Set the exception handling to be used with constrained floating point
void setDefaultConstrainedExcept(fp::ExceptionBehavior NewExcept) {
#ifndef NDEBUG
std::optional<llvm::StringRef> ExceptStr =
convertExceptionBehaviorToStr(NewExcept);
assert(ExceptStr && "Garbage strict exception behavior!");
#endif
DefaultConstrainedExcept = NewExcept;
}
/// Set the rounding mode handling to be used with constrained floating point
void setDefaultConstrainedRounding(llvm::RoundingMode NewRounding) {
#ifndef NDEBUG
std::optional<llvm::StringRef> RoundingStr =
convertRoundingModeToStr(NewRounding);
assert(RoundingStr && "Garbage strict rounding mode!");
#endif
DefaultConstrainedRounding = NewRounding;
}
/// Get the exception handling used with constrained floating point
fp::ExceptionBehavior getDefaultConstrainedExcept() {
return DefaultConstrainedExcept;
}
/// Get the rounding mode handling used with constrained floating point
llvm::RoundingMode getDefaultConstrainedRounding() {
return DefaultConstrainedRounding;
}
//
// Attribute helpers
// -----------------
//
/// Get constant address of a global variable as an MLIR attribute.
/// This wrapper infers the attribute type through the global op.
mlir::cir::GlobalViewAttr getGlobalViewAttr(mlir::cir::GlobalOp globalOp,
mlir::ArrayAttr indices = {}) {
auto type = getPointerTo(globalOp.getSymType());
return getGlobalViewAttr(type, globalOp, indices);
}
/// Get constant address of a global variable as an MLIR attribute.
mlir::cir::GlobalViewAttr getGlobalViewAttr(mlir::cir::PointerType type,
mlir::cir::GlobalOp globalOp,
mlir::ArrayAttr indices = {}) {
auto symbol = mlir::FlatSymbolRefAttr::get(globalOp.getSymNameAttr());
return mlir::cir::GlobalViewAttr::get(type, symbol, indices);
}
mlir::TypedAttr getZeroAttr(mlir::Type t) {
return mlir::cir::ZeroAttr::get(getContext(), t);
}
mlir::cir::BoolAttr getCIRBoolAttr(bool state) {
return mlir::cir::BoolAttr::get(getContext(), getBoolTy(), state);
}
mlir::TypedAttr getConstNullPtrAttr(mlir::Type t) {
assert(t.isa<mlir::cir::PointerType>() && "expected cir.ptr");
return mlir::cir::ConstPtrAttr::get(getContext(), t, 0);
}
mlir::Attribute getString(llvm::StringRef str, mlir::Type eltTy,
unsigned size = 0) {
unsigned finalSize = size ? size : str.size();
// If the string is full of null bytes, emit a #cir.zero rather than
// a #cir.const_array.
if (str.count('\0') == str.size()) {
auto arrayTy = mlir::cir::ArrayType::get(getContext(), eltTy, finalSize);
return getZeroAttr(arrayTy);
}
auto arrayTy = mlir::cir::ArrayType::get(getContext(), eltTy, finalSize);
return getConstArray(mlir::StringAttr::get(str, arrayTy), arrayTy);
}
mlir::cir::ConstArrayAttr getConstArray(mlir::Attribute attrs,
mlir::cir::ArrayType arrayTy) {
return mlir::cir::ConstArrayAttr::get(arrayTy, attrs);
}
mlir::Attribute getConstStructOrZeroAttr(mlir::ArrayAttr arrayAttr,
bool packed = false,
mlir::Type type = {}) {
llvm::SmallVector<mlir::Type, 8> members;
auto structTy = type.dyn_cast<mlir::cir::StructType>();
assert(structTy && "expected cir.struct");
// Collect members and check if they are all zero.
bool isZero = true;
for (auto &attr : arrayAttr) {
const auto typedAttr = attr.dyn_cast<mlir::TypedAttr>();
members.push_back(typedAttr.getType());
isZero &= isNullValue(typedAttr);
}
// Struct type not specified: create anon struct type from members.
if (!structTy)
structTy = getType<mlir::cir::StructType>(members, packed,
mlir::cir::StructType::Struct,
/*ast=*/nullptr);
// Return zero or anonymous constant struct.
if (isZero)
return mlir::cir::ZeroAttr::get(getContext(), structTy);
return mlir::cir::ConstStructAttr::get(structTy, arrayAttr);
}
mlir::cir::ConstStructAttr getAnonConstStruct(mlir::ArrayAttr arrayAttr,
bool packed = false,
mlir::Type ty = {}) {
llvm::SmallVector<mlir::Type, 4> members;
for (auto &f : arrayAttr) {
auto ta = f.dyn_cast<mlir::TypedAttr>();
assert(ta && "expected typed attribute member");
members.push_back(ta.getType());
}
if (!ty)
ty = getAnonStructTy(members, packed);
auto sTy = ty.dyn_cast<mlir::cir::StructType>();
assert(sTy && "expected struct type");
return mlir::cir::ConstStructAttr::get(sTy, arrayAttr);
}
mlir::cir::TypeInfoAttr getTypeInfo(mlir::ArrayAttr fieldsAttr) {
auto anonStruct = getAnonConstStruct(fieldsAttr);
return mlir::cir::TypeInfoAttr::get(anonStruct.getType(), fieldsAttr);
}
mlir::cir::CmpThreeWayInfoAttr getCmpThreeWayInfoStrongOrdering(
const llvm::APSInt <, const llvm::APSInt &eq, const llvm::APSInt >) {
return mlir::cir::CmpThreeWayInfoAttr::get(
getContext(), lt.getSExtValue(), eq.getSExtValue(), gt.getSExtValue());
}
mlir::cir::CmpThreeWayInfoAttr getCmpThreeWayInfoPartialOrdering(
const llvm::APSInt <, const llvm::APSInt &eq, const llvm::APSInt >,
const llvm::APSInt &unordered) {
return mlir::cir::CmpThreeWayInfoAttr::get(
getContext(), lt.getSExtValue(), eq.getSExtValue(), gt.getSExtValue(),
unordered.getSExtValue());
}
mlir::cir::DataMemberAttr getDataMemberAttr(mlir::cir::DataMemberType ty,
size_t memberIndex) {
return mlir::cir::DataMemberAttr::get(getContext(), ty, memberIndex);
}
mlir::cir::DataMemberAttr
getNullDataMemberAttr(mlir::cir::DataMemberType ty) {
return mlir::cir::DataMemberAttr::get(getContext(), ty, std::nullopt);
}
mlir::TypedAttr getZeroInitAttr(mlir::Type ty) {
if (ty.isa<mlir::cir::IntType>())
return mlir::cir::IntAttr::get(ty, 0);
if (auto fltType = ty.dyn_cast<mlir::cir::SingleType>())
return mlir::cir::FPAttr::getZero(fltType);
if (auto fltType = ty.dyn_cast<mlir::cir::DoubleType>())
return mlir::cir::FPAttr::getZero(fltType);
if (auto arrTy = ty.dyn_cast<mlir::cir::ArrayType>())
return getZeroAttr(arrTy);
if (auto ptrTy = ty.dyn_cast<mlir::cir::PointerType>())
return getConstPtrAttr(ptrTy, 0);
if (auto structTy = ty.dyn_cast<mlir::cir::StructType>())
return getZeroAttr(structTy);
if (ty.isa<mlir::cir::BoolType>()) {
return getCIRBoolAttr(false);
}
llvm_unreachable("Zero initializer for given type is NYI");
}
// TODO(cir): Once we have CIR float types, replace this by something like a
// NullableValueInterface to allow for type-independent queries.
bool isNullValue(mlir::Attribute attr) const {
if (attr.isa<mlir::cir::ZeroAttr>())
return true;
if (const auto ptrVal = attr.dyn_cast<mlir::cir::ConstPtrAttr>())
return ptrVal.isNullValue();
if (attr.isa<mlir::cir::GlobalViewAttr>())
return false;
// TODO(cir): introduce char type in CIR and check for that instead.
if (const auto intVal = attr.dyn_cast<mlir::cir::IntAttr>())
return intVal.isNullValue();
if (const auto boolVal = attr.dyn_cast<mlir::cir::BoolAttr>())
return !boolVal.getValue();
if (auto fpAttr = attr.dyn_cast<mlir::cir::FPAttr>()) {
auto fpVal = fpAttr.getValue();
bool ignored;
llvm::APFloat FV(+0.0);
FV.convert(fpVal.getSemantics(), llvm::APFloat::rmNearestTiesToEven,
&ignored);
return FV.bitwiseIsEqual(fpVal);
}
if (const auto structVal = attr.dyn_cast<mlir::cir::ConstStructAttr>()) {
for (const auto elt : structVal.getMembers()) {
// FIXME(cir): the struct's ID should not be considered a member.
if (elt.isa<mlir::StringAttr>())
continue;
if (!isNullValue(elt))
return false;
}
return true;
}
if (const auto arrayVal = attr.dyn_cast<mlir::cir::ConstArrayAttr>()) {
if (arrayVal.getElts().isa<mlir::StringAttr>())
return false;
for (const auto elt : arrayVal.getElts().cast<mlir::ArrayAttr>()) {
if (!isNullValue(elt))
return false;
}
return true;
}
llvm_unreachable("NYI");
}
//
// Type helpers
// ------------
//
mlir::cir::IntType getUIntNTy(int N) {
switch (N) {
case 8:
return getUInt8Ty();
case 16:
return getUInt16Ty();
case 32:
return getUInt32Ty();
case 64:
return getUInt64Ty();
default:
return mlir::cir::IntType::get(getContext(), N, false);
}
}
mlir::cir::IntType getSIntNTy(int N) {
switch (N) {
case 8:
return getSInt8Ty();
case 16:
return getSInt16Ty();
case 32:
return getSInt32Ty();
case 64:
return getSInt64Ty();
default:
return mlir::cir::IntType::get(getContext(), N, true);
}
}
mlir::cir::VoidType getVoidTy() { return typeCache.VoidTy; }
mlir::cir::IntType getSInt8Ty() { return typeCache.SInt8Ty; }
mlir::cir::IntType getSInt16Ty() { return typeCache.SInt16Ty; }
mlir::cir::IntType getSInt32Ty() { return typeCache.SInt32Ty; }
mlir::cir::IntType getSInt64Ty() { return typeCache.SInt64Ty; }
mlir::cir::IntType getUInt8Ty() { return typeCache.UInt8Ty; }
mlir::cir::IntType getUInt16Ty() { return typeCache.UInt16Ty; }
mlir::cir::IntType getUInt32Ty() { return typeCache.UInt32Ty; }
mlir::cir::IntType getUInt64Ty() { return typeCache.UInt64Ty; }
bool isInt8Ty(mlir::Type i) {
return i == typeCache.UInt8Ty || i == typeCache.SInt8Ty;
}
bool isInt16Ty(mlir::Type i) {
return i == typeCache.UInt16Ty || i == typeCache.SInt16Ty;
}
bool isInt32Ty(mlir::Type i) {
return i == typeCache.UInt32Ty || i == typeCache.SInt32Ty;
}
bool isInt64Ty(mlir::Type i) {
return i == typeCache.UInt64Ty || i == typeCache.SInt64Ty;
}
bool isInt(mlir::Type i) { return i.isa<mlir::cir::IntType>(); }
mlir::cir::LongDoubleType
getLongDoubleTy(const llvm::fltSemantics &format) const {
if (&format == &llvm::APFloat::IEEEdouble())
return mlir::cir::LongDoubleType::get(getContext(), typeCache.DoubleTy);
if (&format == &llvm::APFloat::x87DoubleExtended())
return mlir::cir::LongDoubleType::get(getContext(), typeCache.FP80Ty);
if (&format == &llvm::APFloat::IEEEquad())
llvm_unreachable("NYI");
if (&format == &llvm::APFloat::PPCDoubleDouble())
llvm_unreachable("NYI");
llvm_unreachable("unsupported long double format");
}
mlir::Type getVirtualFnPtrType(bool isVarArg = false) {
// FIXME: replay LLVM codegen for now, perhaps add a vtable ptr special
// type so it's a bit more clear and C++ idiomatic.
auto fnTy = mlir::cir::FuncType::get({}, getUInt32Ty(), isVarArg);
assert(!UnimplementedFeature::isVarArg());
return getPointerTo(getPointerTo(fnTy));
}
mlir::cir::FuncType getFuncType(llvm::ArrayRef<mlir::Type> params,
mlir::Type retTy, bool isVarArg = false) {
return mlir::cir::FuncType::get(params, retTy, isVarArg);
}
// Fetch the type representing a pointer to unsigned int values.
mlir::cir::PointerType getUInt8PtrTy(unsigned AddrSpace = 0) {
return typeCache.UInt8PtrTy;
}
mlir::cir::PointerType getUInt32PtrTy(unsigned AddrSpace = 0) {
return mlir::cir::PointerType::get(getContext(), typeCache.UInt32Ty);
}
/// Get a CIR anonymous struct type.
mlir::cir::StructType
getAnonStructTy(llvm::ArrayRef<mlir::Type> members, bool packed = false,
const clang::RecordDecl *ast = nullptr) {
mlir::cir::ASTRecordDeclAttr astAttr = nullptr;
auto kind = mlir::cir::StructType::RecordKind::Struct;
if (ast) {
astAttr = getAttr<mlir::cir::ASTRecordDeclAttr>(ast);
kind = getRecordKind(ast->getTagKind());
}
return getType<mlir::cir::StructType>(members, packed, kind, astAttr);
}
/// Get a CIR record kind from a AST declaration tag.
mlir::cir::StructType::RecordKind
getRecordKind(const clang::TagTypeKind kind) {
switch (kind) {
case clang::TagTypeKind::Struct:
return mlir::cir::StructType::Struct;
case clang::TagTypeKind::Union:
return mlir::cir::StructType::Union;
case clang::TagTypeKind::Class:
return mlir::cir::StructType::Class;
case clang::TagTypeKind::Interface:
llvm_unreachable("interface records are NYI");
case clang::TagTypeKind::Enum:
llvm_unreachable("enum records are NYI");
}
}
/// Get a incomplete CIR struct type.
mlir::cir::StructType getIncompleteStructTy(llvm::StringRef name,
const clang::RecordDecl *ast) {
const auto nameAttr = getStringAttr(name);
auto kind = mlir::cir::StructType::RecordKind::Struct;
if (ast)
kind = getRecordKind(ast->getTagKind());
return getType<mlir::cir::StructType>(nameAttr, kind);
}
/// Get a CIR named struct type.
///
/// If a struct already exists and is complete, but the client tries to fetch
/// it with a different set of attributes, this method will crash.
mlir::cir::StructType getCompleteStructTy(llvm::ArrayRef<mlir::Type> members,
llvm::StringRef name, bool packed,
const clang::RecordDecl *ast) {
const auto nameAttr = getStringAttr(name);
mlir::cir::ASTRecordDeclAttr astAttr = nullptr;
auto kind = mlir::cir::StructType::RecordKind::Struct;
if (ast) {
astAttr = getAttr<mlir::cir::ASTRecordDeclAttr>(ast);
kind = getRecordKind(ast->getTagKind());
}
// Create or get the struct.
auto type = getType<mlir::cir::StructType>(members, nameAttr, packed, kind,
astAttr);
// Complete an incomplete struct or ensure the existing complete struct
// matches the requested attributes.
type.complete(members, packed, astAttr);
return type;
}
mlir::cir::StructType
getCompleteStructType(mlir::ArrayAttr fields, bool packed = false,
llvm::StringRef name = "",
const clang::RecordDecl *ast = nullptr) {
llvm::SmallVector<mlir::Type, 8> members;
for (auto &attr : fields) {
const auto typedAttr = attr.dyn_cast<mlir::TypedAttr>();
members.push_back(typedAttr.getType());
}
if (name.empty())
return getAnonStructTy(members, packed, ast);
else
return getCompleteStructTy(members, name, packed, ast);
}
mlir::cir::ArrayType getArrayType(mlir::Type eltType, unsigned size) {
return mlir::cir::ArrayType::get(getContext(), eltType, size);
}
bool isSized(mlir::Type ty) {
if (ty.isa<mlir::cir::PointerType, mlir::cir::StructType,
mlir::cir::ArrayType, mlir::cir::BoolType, mlir::cir::IntType,
mlir::cir::CIRFPTypeInterface>())
return true;
assert(0 && "Unimplemented size for type");
return false;
}
//
// Constant creation helpers
// -------------------------
//
mlir::cir::ConstantOp getSInt32(uint32_t c, mlir::Location loc) {
auto sInt32Ty = getSInt32Ty();
return create<mlir::cir::ConstantOp>(loc, sInt32Ty,
mlir::cir::IntAttr::get(sInt32Ty, c));
}
mlir::cir::ConstantOp getUInt32(uint32_t C, mlir::Location loc) {
auto uInt32Ty = getUInt32Ty();
return create<mlir::cir::ConstantOp>(loc, uInt32Ty,
mlir::cir::IntAttr::get(uInt32Ty, C));
}
mlir::cir::ConstantOp getSInt64(uint64_t C, mlir::Location loc) {
auto sInt64Ty = getSInt64Ty();
return create<mlir::cir::ConstantOp>(loc, sInt64Ty,
mlir::cir::IntAttr::get(sInt64Ty, C));
}
mlir::cir::ConstantOp getUInt64(uint64_t C, mlir::Location loc) {
auto uInt64Ty = getUInt64Ty();
return create<mlir::cir::ConstantOp>(loc, uInt64Ty,
mlir::cir::IntAttr::get(uInt64Ty, C));
}
mlir::cir::ConstantOp getConstInt(mlir::Location loc, llvm::APSInt intVal) {
bool isSigned = intVal.isSigned();
auto width = intVal.getBitWidth();
mlir::cir::IntType t = isSigned ? getSIntNTy(width) : getUIntNTy(width);
return getConstInt(
loc, t, isSigned ? intVal.getSExtValue() : intVal.getZExtValue());
}
mlir::cir::ConstantOp getConstInt(mlir::Location loc, mlir::Type t,
uint64_t C) {
auto intTy = t.dyn_cast<mlir::cir::IntType>();
assert(intTy && "expected mlir::cir::IntType");
return create<mlir::cir::ConstantOp>(loc, intTy,
mlir::cir::IntAttr::get(t, C));
}
mlir::cir::ConstantOp getBool(bool state, mlir::Location loc) {
return create<mlir::cir::ConstantOp>(loc, getBoolTy(),
getCIRBoolAttr(state));
}
mlir::cir::ConstantOp getFalse(mlir::Location loc) {
return getBool(false, loc);
}
mlir::cir::ConstantOp getTrue(mlir::Location loc) {
return getBool(true, loc);
}
/// Create constant nullptr for pointer-to-data-member type ty.
mlir::cir::ConstantOp getNullDataMemberPtr(mlir::cir::DataMemberType ty,
mlir::Location loc) {
return create<mlir::cir::ConstantOp>(loc, ty, getNullDataMemberAttr(ty));
}
// Creates constant null value for integral type ty.
mlir::cir::ConstantOp getNullValue(mlir::Type ty, mlir::Location loc) {
return create<mlir::cir::ConstantOp>(loc, ty, getZeroInitAttr(ty));
}
mlir::cir::ConstantOp getZero(mlir::Location loc, mlir::Type ty) {
// TODO: dispatch creation for primitive types.
assert(
(ty.isa<mlir::cir::StructType>() || ty.isa<mlir::cir::ArrayType>()) &&
"NYI for other types");
return create<mlir::cir::ConstantOp>(loc, ty, getZeroAttr(ty));
}
//
// Operation creation helpers
// --------------------------
//
/// Create a copy with inferred length.
mlir::cir::CopyOp createCopy(mlir::Value dst, mlir::Value src) {
return create<mlir::cir::CopyOp>(dst.getLoc(), dst, src);
}
/// Create a break operation.
mlir::cir::BreakOp createBreak(mlir::Location loc) {
return create<mlir::cir::BreakOp>(loc);
}
/// Create a continue operation.
mlir::cir::ContinueOp createContinue(mlir::Location loc) {
return create<mlir::cir::ContinueOp>(loc);
}
mlir::cir::MemCpyOp createMemCpy(mlir::Location loc, mlir::Value dst,
mlir::Value src, mlir::Value len) {
return create<mlir::cir::MemCpyOp>(loc, dst, src, len);
}
mlir::Value createNeg(mlir::Value value) {
if (auto intTy = value.getType().dyn_cast<mlir::cir::IntType>()) {
// Source is a unsigned integer: first cast it to signed.
if (intTy.isUnsigned())
value = createIntCast(value, getSIntNTy(intTy.getWidth()));
return create<mlir::cir::UnaryOp>(value.getLoc(), value.getType(),
mlir::cir::UnaryOpKind::Minus, value);
}
llvm_unreachable("negation for the given type is NYI");
}
// TODO: split this to createFPExt/createFPTrunc when we have dedicated cast
// operations.
mlir::Value createFloatingCast(mlir::Value v, mlir::Type destType) {
if (getIsFPConstrained())
llvm_unreachable("constrainedfp NYI");
return create<mlir::cir::CastOp>(v.getLoc(), destType,
mlir::cir::CastKind::floating, v);
}
mlir::Value createFSub(mlir::Value lhs, mlir::Value rhs) {
assert(!UnimplementedFeature::metaDataNode());
if (IsFPConstrained)
llvm_unreachable("Constrained FP NYI");
assert(!UnimplementedFeature::foldBinOpFMF());
return create<mlir::cir::BinOp>(lhs.getLoc(), mlir::cir::BinOpKind::Sub,
lhs, rhs);
}
mlir::Value createDynCast(mlir::Location loc, mlir::Value src,
mlir::cir::PointerType destType, bool isRefCast,
mlir::cir::DynamicCastInfoAttr info) {
auto castKind = isRefCast ? mlir::cir::DynamicCastKind::ref
: mlir::cir::DynamicCastKind::ptr;
return create<mlir::cir::DynamicCastOp>(loc, destType, castKind, src, info,
/*relative_layout=*/false);
}
mlir::Value createDynCastToVoid(mlir::Location loc, mlir::Value src,
bool vtableUseRelativeLayout) {
auto srcTy = src.getType().cast<mlir::cir::PointerType>();
auto destTy = getVoidPtrTy(srcTy.getAddrSpace());
return create<mlir::cir::DynamicCastOp>(
loc, destTy, mlir::cir::DynamicCastKind::ptr, src,
mlir::cir::DynamicCastInfoAttr{}, vtableUseRelativeLayout);
}
cir::Address createBaseClassAddr(mlir::Location loc, cir::Address addr,
mlir::Type destType) {
if (destType == addr.getElementType())
return addr;
auto ptrTy = getPointerTo(destType);
auto baseAddr =
create<mlir::cir::BaseClassAddrOp>(loc, ptrTy, addr.getPointer());
return Address(baseAddr, ptrTy, addr.getAlignment());
}
// FIXME(cir): CIRGenBuilder class should have an attribute with a reference
// to the module so that we don't have search for it or pass it around.
// FIXME(cir): Track a list of globals, or at least the last one inserted, so
// that we can insert globals in the same order they are defined by CIRGen.
/// Creates a versioned global variable. If the symbol is already taken, an ID
/// will be appended to the symbol. The returned global must always be queried
/// for its name so it can be referenced correctly.
[[nodiscard]] mlir::cir::GlobalOp
createVersionedGlobal(mlir::ModuleOp module, mlir::Location loc,
mlir::StringRef name, mlir::Type type, bool isConst,
mlir::cir::GlobalLinkageKind linkage) {
mlir::OpBuilder::InsertionGuard guard(*this);
setInsertionPointToStart(module.getBody());
// Create a unique name if the given name is already taken.
std::string uniqueName;
if (unsigned version = GlobalsVersioning[name.str()]++)
uniqueName = name.str() + "." + std::to_string(version);
else
uniqueName = name.str();
return create<mlir::cir::GlobalOp>(loc, uniqueName, type, isConst, linkage);
}
mlir::Value createGetGlobal(mlir::cir::GlobalOp global,
bool threadLocal = false) {
return create<mlir::cir::GetGlobalOp>(global.getLoc(),
getPointerTo(global.getSymType()),
global.getName(), threadLocal);
}
mlir::Value createGetBitfield(mlir::Location loc, mlir::Type resultType,
mlir::Value addr, mlir::Type storageType,
const CIRGenBitFieldInfo &info,
bool isLvalueVolatile, bool useVolatile) {
auto offset = useVolatile ? info.VolatileOffset : info.Offset;
return create<mlir::cir::GetBitfieldOp>(loc, resultType, addr, storageType,
info.Name, info.Size, offset,
info.IsSigned, isLvalueVolatile);
}
mlir::Value createSetBitfield(mlir::Location loc, mlir::Type resultType,
mlir::Value dstAddr, mlir::Type storageType,
mlir::Value src, const CIRGenBitFieldInfo &info,
bool isLvalueVolatile, bool useVolatile) {
auto offset = useVolatile ? info.VolatileOffset : info.Offset;
return create<mlir::cir::SetBitfieldOp>(
loc, resultType, dstAddr, storageType, src, info.Name, info.Size,
offset, info.IsSigned, isLvalueVolatile);
}
/// Create a pointer to a record member.
mlir::Value createGetMember(mlir::Location loc, mlir::Type result,
mlir::Value base, llvm::StringRef name,
unsigned index) {
return create<mlir::cir::GetMemberOp>(loc, result, base, name, index);
}
/// Cast the element type of the given address to a different type,
/// preserving information like the alignment.
cir::Address createElementBitCast(mlir::Location loc, cir::Address addr,
mlir::Type destType) {
if (destType == addr.getElementType())
return addr;
auto ptrTy = getPointerTo(destType);
return Address(createBitcast(loc, addr.getPointer(), ptrTy), destType,
addr.getAlignment());
}
mlir::Value createLoad(mlir::Location loc, Address addr) {
auto ptrTy = addr.getPointer().getType().dyn_cast<mlir::cir::PointerType>();
if (addr.getElementType() != ptrTy.getPointee())
addr = addr.withPointer(
createPtrBitcast(addr.getPointer(), addr.getElementType()));
return create<mlir::cir::LoadOp>(loc, addr.getElementType(),
addr.getPointer());
}
mlir::Value createAlignedLoad(mlir::Location loc, mlir::Type ty,
mlir::Value ptr,
[[maybe_unused]] llvm::MaybeAlign align,
[[maybe_unused]] bool isVolatile) {
assert(!UnimplementedFeature::volatileLoadOrStore());
assert(!UnimplementedFeature::alignedLoad());
return create<mlir::cir::LoadOp>(loc, ty, ptr);
}
mlir::Value createAlignedLoad(mlir::Location loc, mlir::Type ty,
mlir::Value ptr, llvm::MaybeAlign align) {
return createAlignedLoad(loc, ty, ptr, align, /*isVolatile=*/false);
}
mlir::Value
createAlignedLoad(mlir::Location loc, mlir::Type ty, mlir::Value addr,
clang::CharUnits align = clang::CharUnits::One()) {
return createAlignedLoad(loc, ty, addr, align.getAsAlign());
}
mlir::cir::StoreOp createStore(mlir::Location loc, mlir::Value val,
Address dst, bool _volatile = false,
::mlir::cir::MemOrderAttr order = {}) {
return CIRBaseBuilderTy::createStore(loc, val, dst.getPointer(), _volatile,
order);
}
mlir::cir::StoreOp createFlagStore(mlir::Location loc, bool val,
mlir::Value dst) {
auto flag = getBool(val, loc);
return CIRBaseBuilderTy::createStore(loc, flag, dst);
}
// Convert byte offset to sequence of high-level indices suitable for
// GlobalViewAttr. Ideally we shouldn't deal with low-level offsets at all
// but currently some parts of Clang AST, which we don't want to touch just
// yet, return them.
void computeGlobalViewIndicesFromFlatOffset(
int64_t Offset, mlir::Type Ty, CIRDataLayout Layout,
llvm::SmallVectorImpl<int64_t> &Indices) {
if (!Offset)
return;
mlir::Type SubType;
if (auto ArrayTy = Ty.dyn_cast<mlir::cir::ArrayType>()) {
auto EltSize = Layout.getTypeAllocSize(ArrayTy.getEltType());
Indices.push_back(Offset / EltSize);
SubType = ArrayTy.getEltType();
Offset %= EltSize;
} else if (auto PtrTy = Ty.dyn_cast<mlir::cir::PointerType>()) {
auto EltSize = Layout.getTypeAllocSize(PtrTy.getPointee());
Indices.push_back(Offset / EltSize);
SubType = PtrTy.getPointee();
Offset %= EltSize;
} else if (auto StructTy = Ty.dyn_cast<mlir::cir::StructType>()) {
auto Elts = StructTy.getMembers();
unsigned Pos = 0;
for (size_t I = 0; I < Elts.size(); ++I) {
auto EltSize = Layout.getTypeAllocSize(Elts[I]);
unsigned AlignMask = Layout.getABITypeAlign(Elts[I]) - 1;
Pos = (Pos + AlignMask) & ~AlignMask;
if (Offset < Pos + EltSize) {
Indices.push_back(I);
SubType = Elts[I];
Offset -= Pos;
break;
}
Pos += EltSize;
}
} else {
llvm_unreachable("unexpected type");
}
assert(SubType);
computeGlobalViewIndicesFromFlatOffset(Offset, SubType, Layout, Indices);
}
mlir::cir::StackSaveOp createStackSave(mlir::Location loc, mlir::Type ty) {
return create<mlir::cir::StackSaveOp>(loc, ty);
}
mlir::cir::StackRestoreOp createStackRestore(mlir::Location loc,
mlir::Value v) {
return create<mlir::cir::StackRestoreOp>(loc, v);
}
// TODO(cir): Change this to hoist alloca to the parent *scope* instead.
/// Move alloca operation to the parent region.
void hoistAllocaToParentRegion(mlir::cir::AllocaOp alloca) {
auto &block = alloca->getParentOp()->getParentRegion()->front();
const auto allocas = block.getOps<mlir::cir::AllocaOp>();
if (allocas.empty()) {
alloca->moveBefore(&block, block.begin());
} else {
alloca->moveAfter(*std::prev(allocas.end()));
}
}
mlir::cir::CmpThreeWayOp
createThreeWayCmpStrong(mlir::Location loc, mlir::Value lhs, mlir::Value rhs,
const llvm::APSInt <Res, const llvm::APSInt &eqRes,
const llvm::APSInt >Res) {
assert(ltRes.getBitWidth() == eqRes.getBitWidth() &&
ltRes.getBitWidth() == gtRes.getBitWidth() &&
"the three comparison results must have the same bit width");
auto cmpResultTy = getSIntNTy(ltRes.getBitWidth());
auto infoAttr = getCmpThreeWayInfoStrongOrdering(ltRes, eqRes, gtRes);
return create<mlir::cir::CmpThreeWayOp>(loc, cmpResultTy, lhs, rhs,
infoAttr);
}
mlir::cir::CmpThreeWayOp
createThreeWayCmpPartial(mlir::Location loc, mlir::Value lhs, mlir::Value rhs,
const llvm::APSInt <Res, const llvm::APSInt &eqRes,
const llvm::APSInt >Res,
const llvm::APSInt &unorderedRes) {
assert(ltRes.getBitWidth() == eqRes.getBitWidth() &&
ltRes.getBitWidth() == gtRes.getBitWidth() &&
ltRes.getBitWidth() == unorderedRes.getBitWidth() &&
"the four comparison results must have the same bit width");
auto cmpResultTy = getSIntNTy(ltRes.getBitWidth());
auto infoAttr =
getCmpThreeWayInfoPartialOrdering(ltRes, eqRes, gtRes, unorderedRes);
return create<mlir::cir::CmpThreeWayOp>(loc, cmpResultTy, lhs, rhs,
infoAttr);
}
mlir::cir::GetRuntimeMemberOp createGetIndirectMember(mlir::Location loc,
mlir::Value objectPtr,
mlir::Value memberPtr) {
auto objectPtrTy = objectPtr.getType().cast<mlir::cir::PointerType>();
auto memberPtrTy = memberPtr.getType().cast<mlir::cir::DataMemberType>();
auto resultTy =
getPointerTo(memberPtrTy.getMemberTy(), objectPtrTy.getAddrSpace());
return create<mlir::cir::GetRuntimeMemberOp>(loc, resultTy, objectPtr,
memberPtr);
}
};
} // namespace cir
#endif